تحويل الإحداثيات بين Mercator العرضية المخصصة و WGS84 LON-LAT
-
24-09-2019 - |
سؤال
أحاول وضع مجموعة من النقاط (X ، Y) على الخريطة ، والتي لها أصل Lon-Lat. أرغب في إعداد إسقاط Mercator عرضيًا مخصصًا ، وتركز على أصله ، واستخدام ذلك لإسقاط نقاطتي في Lon-Lat.
هذا يبدو وكأنه نوع من الأشياء التي يجب أن تكون Geotoolkit (أو Geotools) قادرة على القيام بها ، لكنني أواجه صعوبة كبيرة في تحديد أفضل طريقة لاستخدام مكتبة المرجعية.
لقد تمكنت من كتابة بعض التعليمات البرمجية التي تعمل في بعض الأحيان ، لكن في أوقات أخرى أحصل على استثناء من Classcast ، حيث يحاول Geotoolkit تحميل MathTransformFactory. أي مساعدة سيكون محل تقدير كبير. هذا هو الكود الخاص بي:
public class LonLatConverter {
private static final double WGS84_AXIS_MINOR = 6356752.314;
private static final double WGS84_AXIS_MAJOR = 6378137.000;
private GeoLocation origin;
private static MathTransformFactory factory;
public LonLatConverter(GeoLocation origin) {
this.origin = origin;
}
public GeoLocation toLonLat(Vector2D location) {
double lon = 0;
double lat = 0;
try {
MathTransformFactory factory = getMathTransformFactory();
MathTransform tr = getMathTransform(factory);
DirectPosition sourcePt = new GeneralDirectPosition(
location.getX(), location.getY());
DirectPosition targetPt = tr.transform(sourcePt, null);
lon = targetPt.getOrdinate(0);
lat = targetPt.getOrdinate(1);
} catch (Exception e) {
throw new RuntimeException(e);
}
return new GeoLocation(lon, lat);
}
private MathTransform getMathTransform(MathTransformFactory factory)
throws NoninvertibleTransformException, FactoryException {
ParameterValueGroup p = factory
.getDefaultParameters("Transverse_Mercator");
p.parameter("semi_major").setValue(WGS84_AXIS_MAJOR);
p.parameter("semi_minor").setValue(WGS84_AXIS_MINOR);
p.parameter("central_meridian").setValue(origin.getLongitude());
p.parameter("latitude_of_origin").setValue(origin.getLatitude());
MathTransform tr = factory.createParameterizedTransform(p).inverse();
return tr;
}
private MathTransformFactory getMathTransformFactory() {
if (factory == null) {
FactoryRegistry registry = new FactoryRegistry(
MathTransformFactory.class);
factory = registry.getServiceProvider(MathTransformFactory.class,
null, null, Hints.MATH_TRANSFORM_FACTORY);
}
return factory;
}
}
شكرًا،
دان
المحلول
اتضح أن الاستثناء ناتج عن تفاعل غريب بين GeotOlkit و GWT (Google Web Toolkit) "DevMode" (يحدث تحويل الإحداثيات في نهاية تطبيق الويب GWT/Java). للسجل ، هذا هو الاستثناء:
Caused by: java.lang.RuntimeException: java.lang.ClassCastException: class org.geotoolkit.referencing.operation.DefaultMathTransformFactory
at com.example.LonLatConverter.toLonLat(LonLatConverter.java:47)
...
Caused by: java.lang.ClassCastException: class org.geotoolkit.referencing.operation.DefaultMathTransformFactory
at java.lang.Class.asSubclass(Class.java:3018)
at org.geotoolkit.factory.FactoryRegistry.register(FactoryRegistry.java:921)
at org.geotoolkit.factory.FactoryRegistry.scanForPlugins(FactoryRegistry.java:793)
at org.geotoolkit.factory.FactoryRegistry.scanForPluginsIfNeeded(FactoryRegistry.java:843)
at org.geotoolkit.factory.FactoryRegistry.getServiceProviders(FactoryRegistry.java:236)
at com.example.LonLatConverter.getMathTransformFactory(LonLatConverter.java:71)
at com.example.LonLatConverter.toLonLat(LonLatConverter.java:35)
...
عملي الحالي هو استخدام المكون الإضافي GWT Eclipse. هذا لا يرمي الاستثناء ، ولا بناء الحرب ونشر تطبيق الويب على Tomcat أو Jetty. كل شيء غريب للغاية ، لكنني سعيد بتجاهله في الوقت الحالي.