All Downloads are FREE. Search and download functionalities are using the official Maven repository.

edu.mines.jtk.mosaic.Transcaler Maven / Gradle / Ivy

The newest version!
/****************************************************************************
Copyright 2004, Colorado School of Mines and others.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
****************************************************************************/
package edu.mines.jtk.mosaic;

import edu.mines.jtk.util.Check;
import static edu.mines.jtk.util.MathPlus.log10;

/**
 * Translates and scales (maps) user coordinates to/from device coordinates.
 * The mapping is specified by two rectangles, one in user coordinates and 
 * the other in device coordinates.
 * 

* Device coordinates are ints, and the device coordinate rectangle * typically corresponds to device bounds. User coordinates are doubles. *

* In conversion from user to device coordinates, the latter are clipped * to lie in the range [-32768,32767], which is the range of a 16-bit * short integer. Although device coordinates are represented by ints, * they are often limited by an underlying graphics systems to the 16-bit * range of shorts. *

* Conversion from/to user coordinates to/from device coordinates behaves * robustly in the cases where the mapping is degenerate. For example, if * the device coordinate rectangle has width one, then conversion from * any device x-coordinate to user x-coordinate yields the average of the * user x-coordinate bounds. Likewise, if the user coordinate rectangle * has zero width, then conversion from any user x-coordinate to device * x-coordinate yields the average of the device x-coordinate bounds. * * @author Dave Hale, Colorado School of Mines * @version 2004.12.23 */ public class Transcaler { /** * Constructs a transcaler with identity coordinate mapping. * In this mapping, device coordinates equal user coordinates, * rounded to the nearest integer. */ public Transcaler() { this(0.0,0.0,1.0,1.0,0,0,1,1); } /** * Constructs a transcaler with specified device width and height. * Maps user coordinates (0.0,0.0) to device coordinates (0,0) and * user coordinates (1.0,1.0) to device coordinates (width-1,height-1). * @param width the width, in device coordinates. * @param height the height, in device coordinates. */ public Transcaler(int width, int height) { this(0.0,0.0,1.0,1.0,0,0,width-1,height-1); Check.argument(width>0,"width>0"); Check.argument(height>0,"height>0"); } /** * Constructs a transcaler with specified coordinate mapping. * @param x1u the user x-coordinate corresponding to x1d. * @param y1u the user y-coordinate corresponding to y1d. * @param x2u the user x-coordinate corresponding to x2d. * @param y2u the user y-coordinate corresponding to y2d. * @param x1d the device x-coordinate corresponding to x1u. * @param y1d the device y-coordinate corresponding to y1u. * @param x2d the device x-coordinate corresponding to x2u. * @param y2d the device y-coordinate corresponding to y2u. */ public Transcaler( double x1u, double y1u, double x2u, double y2u, int x1d, int y1d, int x2d, int y2d) { setMapping(x1u,y1u,x2u,y2u,x1d,y1d,x2d,y2d); } /** * Sets the coordinate mapping for this transcaler. * @param x1u the user x-coordinate corresponding to x1d. * @param y1u the user y-coordinate corresponding to y1d. * @param x2u the user x-coordinate corresponding to x2d. * @param y2u the user y-coordinate corresponding to y2d. * @param x1d the device x-coordinate corresponding to x1u. * @param y1d the device y-coordinate corresponding to y1u. * @param x2d the device x-coordinate corresponding to x2u. * @param y2d the device y-coordinate corresponding to y2u. */ public void setMapping( double x1u, double y1u, double x2u, double y2u, int x1d, int y1d, int x2d, int y2d) { _x1u = x1u; _x2u = x2u; _y1u = y1u; _y2u = y2u; _x1d = x1d; _x2d = x2d; _y1d = y1d; _y2d = y2d; computeShiftAndScale(); } /** * Sets the user-coordinate part of the mapping for this transcaler. * @param x1u the user x-coordinate corresponding to the current x1d. * @param y1u the user y-coordinate corresponding to the current y1d. * @param x2u the user x-coordinate corresponding to the current x2d. * @param y2u the user y-coordinate corresponding to the current y2d. */ public void setMapping(double x1u, double y1u, double x2u, double y2u) { _x1u = x1u; _x2u = x2u; _y1u = y1u; _y2u = y2u; computeShiftAndScale(); } /** * Sets the device-coordinate part of the mapping for this transcaler. * @param x1d the device x-coordinate corresponding to the current x1u. * @param y1d the device y-coordinate corresponding to the current y1u. * @param x2d the device x-coordinate corresponding to the current x2u. * @param y2d the device y-coordinate corresponding to the current y2u. */ public void setMapping(int x1d, int y1d, int x2d, int y2d) { _x1d = x1d; _x2d = x2d; _y1d = y1d; _y2d = y2d; computeShiftAndScale(); } /** * Sets the device-coordinate width and height. Maps the current user * coordinates (x1u,y1u) to device coordinates (0,0) and user coordinates * (x2u,y2u) to device coordinates (width-1,height-1). * @param width device-coordinate width. * @param height device-coordinate height. */ public void setMapping(int width, int height) { Check.argument(width>0,"width>0"); Check.argument(height>0,"height>0"); setMapping(0,0,width-1,height-1); } /** * Returns a new transcaler that combines this transcaler with projectors. * The returned transcaler includes the transforms of the projectors. * Does not change this transcaler. * @param xp the projector for x coordinates. * @param yp the projector for y coordinates. * @return the new transcaler. */ public Transcaler combineWith(Projector xp, Projector yp) { AxisScale xsc = xp.getScale(); AxisScale ysc = yp.getScale(); double x1v = (xsc==AxisScale.LOG10)?log10(xp.v(_x1u)):xp.v(_x1u); double x2v = (xsc==AxisScale.LOG10)?log10(xp.v(_x2u)):xp.v(_x2u); double y1v = (ysc==AxisScale.LOG10)?log10(yp.v(_y1u)):yp.v(_y1u); double y2v = (ysc==AxisScale.LOG10)?log10(yp.v(_y2u)):yp.v(_y2u); return new Transcaler(x1v,y1v,x2v,y2v,_x1d,_y1d,_x2d,_y2d, xsc, ysc); } /** * Converts the specified user x-coordinate to device x-coordinate. * @param xu the user x-coordinate. * @return the device x-coordinate. */ public int x(double xu) { if(_xpScale==AxisScale.LOG10) xu = log10(xu); double xd = _xushift+_xuscale*xu; if (xdDMAX) { xd = DMAX; } return (int)(xd); } /** * Converts the specified user y-coordinate to device y-coordinate. * @param yu the user y-coordinate. * @return the device y-coordinate. */ public int y(double yu) { if(_ypScale==AxisScale.LOG10) yu = log10(yu); double yd = _yushift+_yuscale*yu; if (ydDMAX) { yd = DMAX; } return (int)(yd); } /** * Converts the specified user-coordinate width to device-coordinate width. * @param wu the user-coordinate width. * @return the device-coordinate width. */ public int width(double wu) { double wd = _xuscale*wu; if (wdDMAXW) { wd = DMAXW; } return (int)(wd+1.5); } /** * Converts the specified user-coordinate height to device-coordinate height. * @param hu the user-coordinate height. * @return the device-coordinate height. */ public int height(double hu) { double hd = _yuscale*hu; if (hdDMAXW) { hd = DMAXW; } return (int)(hd+1.5); } /** * Converts the specified device x-coordinate to user x-coordinate. * @param xd the device x-coordinate. * @return the user x-coordinate. */ public double x(int xd) { return _xdshift+_xdscale*xd; } /** * Converts the specified device y-coordinate to user y-coordinate. * @param yd the device y-coordinate. * @return the user y-coordinate. */ public double y(int yd) { return _ydshift+_ydscale*yd; } /** * Converts the specified device-coordinate width to user-coordinate width. * @param wd the device-coordinate width. * @return the user-coordinate width. */ public double width(int wd) { return _xdscale*(wd-1); } /** * Converts the specified device-coordinate height to user-coordinate height. * @param hd the device-coordinate height. * @return the user-coordinate height. */ public double height(int hd) { return _ydscale*(hd-1); } /////////////////////////////////////////////////////////////////////////// // private private static final double DMIN = -32768.0; // device min coordinate private static final double DMAX = 32767.0; // device max coordinate private static final double DMINW = DMIN-DMAX; // device min width private static final double DMAXW = DMAX-DMIN; // device max width private double _x1u,_y1u,_x2u,_y2u; private int _x1d,_y1d,_x2d,_y2d; private double _xushift,_xuscale,_yushift,_yuscale; private double _xdshift,_xdscale,_ydshift,_ydscale; private AxisScale _xpScale = AxisScale.LINEAR; // Scale of hor. projector received from combineWith private AxisScale _ypScale = AxisScale.LINEAR; // Scale of ver. projector received from combineWith private void computeShiftAndScale() { if (_x1u!=_x2u) { _xuscale = (_x2d-_x1d)/(_x2u-_x1u); _xushift = _x1d-_x1u*_xuscale+0.5; } else { _xushift = 0.5*(_x1d+_x2d)+0.5; _xuscale = 0.0; } if (_x1d!=_x2d) { _xdscale = (_x2u-_x1u)/(_x2d-_x1d); _xdshift = _x1u-_x1d*_xdscale; } else { _xdshift = 0.5*(_x1u+_x2u); _xdscale = 0.0; } if (_y1u!=_y2u) { _yuscale = (_y2d-_y1d)/(_y2u-_y1u); _yushift = _y1d-_y1u*_yuscale+0.5; } else { _yushift = 0.5*(_y1d+_y2d)+0.5; _yuscale = 0.0; } if (_y1d!=_y2d) { _ydscale = (_y2u-_y1u)/(_y2d-_y1d); _ydshift = _y1u-_y1d*_ydscale; } else { _ydshift = 0.5*(_y1u+_y2u); _ydscale = 0.0; } } /** * Constructs a transcaler with specified coordinate mapping * and specified Scales. This should only be used with the * combineWith() method when receiving projectors that * may be LOG scale. It is not the intention to be able * to create arbitrary LOG scaled Transcalers. * @param x1u the user x-coordinate corresponding to x1d. * @param y1u the user y-coordinate corresponding to y1d. * @param x2u the user x-coordinate corresponding to x2d. * @param y2u the user y-coordinate corresponding to y2d. * @param x1d the device x-coordinate corresponding to x1u. * @param y1d the device y-coordinate corresponding to y1u. * @param x2d the device x-coordinate corresponding to x2u. * @param y2d the device y-coordinate corresponding to y2u. * @param hscale the horizontal projector Scale * @param vscale the vertical projector Scale */ private Transcaler( double x1u, double y1u, double x2u, double y2u, int x1d, int y1d, int x2d, int y2d, AxisScale hscale, AxisScale vscale) { _xpScale = hscale; _ypScale = vscale; setMapping(x1u,y1u,x2u,y2u,x1d,y1d,x2d,y2d); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy