SpatialDB Advisor
Many years ago I wrote a PL/SQL function to move any sdo_geometry when supplied a “delta” in terms of deltaX, deltaY and optionally deltaZ. This function also allowed a user to supply an envelope, box or minimum bounding rectangle (MBR) to the function so that only those vertices that lie within an sdo_geometry object would be moved.
I often play with PostGIS and have found the writings on Regina Obe very enlightening. PostGIS contains a function called ST_Translate (not part of the SQL/MM standard) which is identical to my original Move function.
So, recently, I decided to re-visit my Move function to improve its implementation, and provide ST_Translate wrappers for it.
This article summarises this recent work.
PostGIS’s ST_Translate function is described, neatly, by its documentation so I will duplicate the examples it for my Oracle implementation.
/* Move a point 1 degree longitude */
/* PostGIS */
SELECT ST_AsText(ST_Translate(ST_GeomFromText('POINT(-71.01 42.37)',4326),1,0)) As wgs_transgeomtxt;
OK, so my functions pass the “PostGIS” test.
That is all well and good with the standard parameters, but is that all?
What I have found is that, sometimes, you want to move only part of a geometry object. For example, at Forestry Tasmania, when I was GIS Manager, we had a situation, before my time, where King Island (see image below) had been artificially moved from its geographic position to an artificial position for the purposes of map production (it was also related to single precision vs double precision ArcInfo coverages but that is another story). When implementing a change from the Australian Map Grid (AMG, based on the Australian Geodetic Datum 1966) to the Map Grid of Australia (based on the Australian Geodetic Datum 1994), we had to first move King Island back to its normal AMG position before projecting it to MGA.

In this situation, I found that the Move function needed to be enhanced to allow for the provision of a minimum bounding rectangle (MBR) which the function tested against each vertex in an sdo_geometry object such that it moved only those which the INSIDE topological relationship.
In summary, the ST_Translate PostGIS function needs enhancing to allow for this possibility. I will now present the Move and ST_Translate functions which are implemented in the GEOM package which is a part of my free PL/SQL functions. I will only present the relevant package specifications:
/* ----------------------------------------------------------------------------------------
*
function : MOVE
* precis : Function which updates all coordinates in a shape by applying x/y shift.
* version : 2.0
* description: In order to be able to apply the King Island move to the shapes stored
* in an Oracle Spatial shape, a function is needed that iterates though
* all the coordinates of a shape and applys a deltaX and deltaY to them.
* If p_envelope is provided, only those coordinate that fall within it are moved:
* otherwise all are moved.
* usage : Function move ( p_geometry IN MDSYS.SDO_GEOMETRY,
* p_deltaX IN number,
* p_deltaY IN number,
* p_envelope IN MDSYS.SDO_GEOMETRY ) RETURN MDSYS.SDO_GEOMETRY DETERMINISTIC;
* eg fixedShape := CODESYS.geom.tolerance(shape,diminfo);
* param : p_geometry : The shape to move.
* paramtype : p_geomery : MDSYS.SDO_GEOMETRY
* param : p_deltaX : Shift to be applied to the X coordinate.
* paramtype : p_deltaX : number
* param : p_deltaY : Shift to be applied to the Y coordinate.
* paramtype : p_deltaY : number
* param : p_deltaZ : Shift to be applied to the Z coordinate.
* paramtype : p_deltaZ : number
* param : p_mbr : minimum bounding rectangle/envelope in which p_geometry's coordinates have to be within to be able to be moved.
* paramtype : p_mbr : MBR object
* param : p_filter_geom : A shape defining a sub-area in which p_geometry's coordinates have to be within for move.
* paramtype : p_filter_geom : MDSYS.SDO_GEOMETRY
* param : p_filter_mask : A mask for use with SDO_GEOM.RELATE()
* paramtype : p_filer_mask : varchar
* param : p_tolerance : Tolerance used when processing vertices with SDO_GEOM.RELATE
* param : p_tolerance : Number * requires : CODESYS.GF package.
* return : newShape : Shape whose coordinates are 'moved'.
* rtnType : newShape : MDSYS.SDO_GEOMETRY
* history : Simon Greener - Mar 2003 - Original coding.
* history : Simon Greener - Jul 2006 - Migrated to GF package and made 3D aware.
* history : Simon Greener - Sep 2007 - Removed need for SDO_GEOM.RELATE via use of MBR type.
* history : Simon Greener - Jun 2008 - Removed modification of ordinates to precision of diminfo/tolerance as duplicates Tolerance() function.
* @copyright : Free for public use
**/
FUNCTION Move( p_geometry IN MDSYS.SDO_GEOMETRY,
p_deltaX IN NUMBER,
p_deltaY IN NUMBER,
p_deltaZ IN NUMBER := NULL,
p_mbr IN CODESYS.MBR := NULL,
p_filter_geom IN MDSYS.SDO_GEOMETRY := NULL,
p_filter_mask IN VARCHAR2 := 'INSIDE',
p_tolerance IN NUMBER := NULL
)
Return MDSYS.SDO_GEOMETRY Deterministic;
So, first, let’s move “King Island”: to a new location.
update King_Island a
set a.geometry = Geom.St_Translate(a.geometry,40000,-9000,NULL,Codesys.MBR(225000,5548000,257000,5616000))
where a.name = 'King Island';
commit;
Notice how King Island has been moved to a new location in the following image.

Now, let’s move it back:
update King_Island a
set a.geometry = Geom.St_Translate(a.geometry,-40000,9000,NULL,Codesys.MBR(265000,5539000,297000,5607000))
where a.name = 'King Island';
commit;
Here it is moved back to its original position:

Now, above we used an MBR parameter to constrain the move. But what if we wanted to move vertices that won’t neatly fit into an MBR object? The Move/ST_Translate functions need parameters to enable a move from any input SDO_GEOMETRY and topological mask. If the following we use a polygon to move King Island and Hunter Island (in blue in the following image).

update King_Island a
set a.geometry = Geom.St_Translate(a.geometry,40000,10000,NULL,NULL,mdsys.sdo_geometry('POLYGON((225647 5617686, 227451 5503583, 316749 5500877, 308180 5624902, 225647 5617686))',28355),'INSIDE',0.05);
commit;
This image shows the two islands only being moved.

Finally, Now, move King Island and Hunter Island back to their original position.
update King_Island a
set a.geometry = Geom.St_Translate(a.geometry,-40000,-10000,NULL,NULL)
where a.name in ('King Island','Hunter Island');
commit;

The implementation of the Move and ST_Translate functions are in the GEOM package of my free PL/SQL utilities.
I hope this is useful to someone.


















<<Elem_Info_Array Processing: An alternative to SDO_UTIL.GetNumRings and querying SDO_ELEM_INFO itself >>ST_Azimuth for Oracle: AKA Cogo.Bearing
Dose this package(actually the move function) support oracle 9i,I can’t compile it in 9i, could you please tell me how to fix it or can you provide me an oracle 9i version of the package, many thanks!
— Benny May 13, 01:33 pm #