Statistics
| Revision:

gvsig-3d / 2.1 / branches / org.gvsig.view3d_vector_and_extrusion_2.3 / org.gvsig.view3d / org.gvsig.view3d / org.gvsig.view3d.swing / org.gvsig.view3d.swing.impl / src / main / java / org / gvsig / view3d / swing / impl / layers / DefaultLayerConverter.java @ 708

History | View | Annotate | Download (17.6 KB)

1
/**
2
 * gvSIG. Desktop Geographic Information System.
3
 *
4
 * Copyright � 2007-2015 gvSIG Association
5
 *
6
 * This program is free software; you can redistribute it and/or
7
 * modify it under the terms of the GNU General Public License
8
 * as published by the Free Software Foundation; either version 2
9
 * of the License, or (at your option) any later version.
10
 *
11
 * This program is distributed in the hope that it will be useful,
12
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14
 * GNU General Public License for more details.
15
 *
16
 * You should have received a copy of the GNU General Public License
17
 * along with this program; if not, write to the Free Software
18
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
19
 * MA  02110-1301, USA.
20
 *
21
 * For any additional information, do not hesitate to contact us
22
 * at info AT gvsig.com, or visit our website www.gvsig.com.
23
 */
24

    
25
package org.gvsig.view3d.swing.impl.layers;
26

    
27
import java.awt.geom.Point2D;
28
import java.security.MessageDigest;
29
import java.security.NoSuchAlgorithmException;
30

    
31
import org.cresques.cts.ICoordTrans;
32
import org.cresques.cts.IProjection;
33
import org.slf4j.Logger;
34
import org.slf4j.LoggerFactory;
35

    
36
import org.gvsig.fmap.crs.CRSFactory;
37
import org.gvsig.fmap.dal.coverage.dataset.Buffer;
38
import org.gvsig.fmap.dal.coverage.store.props.Statistics;
39
import org.gvsig.fmap.dal.exception.DataException;
40
import org.gvsig.fmap.geom.primitive.Envelope;
41
import org.gvsig.fmap.mapcontext.layers.FLayer;
42
import org.gvsig.fmap.mapcontext.layers.operations.SingleLayer;
43
import org.gvsig.fmap.mapcontext.layers.vectorial.FLyrVect;
44
import org.gvsig.raster.fmap.layers.FLyrRaster;
45
import org.gvsig.view3d.lib.api.View3DLocator;
46
import org.gvsig.view3d.lib.api.View3DManager;
47
import org.gvsig.view3d.lib.api.loadingmodes.NwwLoaderFactory;
48
import org.gvsig.view3d.lib.api.properties.ElevationLayerProperties3D;
49
import org.gvsig.view3d.lib.api.properties.LayerProperties3D;
50
import org.gvsig.view3d.lib.api.properties.RasterLayerProperties3D;
51
import org.gvsig.view3d.lib.impl.layers.properties.DefaultElevationLayerProperties3D;
52
import org.gvsig.view3d.swing.api.MapControl3D;
53
import org.gvsig.view3d.swing.api.properties.NwwRasterLoaderFactory;
54
import org.gvsig.view3d.swing.impl.DefaultMapControl3D;
55
import org.gvsig.view3d.swing.impl.layers.loaders.ElevationLayerLoaderFactory;
56

    
57
import gov.nasa.worldwind.avlist.AVKey;
58
import gov.nasa.worldwind.avlist.AVList;
59
import gov.nasa.worldwind.avlist.AVListImpl;
60
import gov.nasa.worldwind.geom.Angle;
61
import gov.nasa.worldwind.geom.LatLon;
62
import gov.nasa.worldwind.geom.Sector;
63
import gov.nasa.worldwind.globes.ElevationModel;
64
import gov.nasa.worldwind.layers.Layer;
65
import gov.nasa.worldwind.terrain.BasicElevationModel;
66
import gov.nasa.worldwind.util.WWMath;
67

    
68
/**
69
 * Default implementation of {@link LayerConverter}.
70
 *
71
 * @author <a href="mailto:lmarques@disid.com">Lluis Marques</a>
72
 */
73
public class DefaultLayerConverter implements LayerConverter {
74

    
75
    private static final Logger LOG = LoggerFactory.getLogger(DefaultLayerConverter.class);
76

    
77
    private final int DEFAULT_RASTERIZED_LEVEL_ZERO = 20;
78
    private final int DEFAULT_TILE_WIDTH = 512;
79
    private final int DEFAULT_TILE_HEIGHT = 512;
80

    
81
    private View3DManager manager = View3DLocator.getManager();
82

    
83
    public Layer convertToLayer(MapControl3D mapControl, FLayer layer) throws DataException {
84
        LayerProperties3D layerProperties = manager.getLayerProperties(layer, null);
85

    
86
        NwwLoaderFactory loadingFactory = manager.getLayerLoadingFactory(layer);
87
        if (loadingFactory instanceof NwwRasterLoaderFactory) {
88
            NwwRasterLoaderFactory rasterLoaderFactory = (NwwRasterLoaderFactory) loadingFactory;
89
            rasterLoaderFactory.setMapControl(mapControl);
90
        }
91
        Layer nwwLayer = loadingFactory.loadLayer(layerProperties);
92
        return nwwLayer;
93
    }
94

    
95
    public ElevationModel convertToElevationModel(MapControl3D mapControl, FLayer layer) {
96
        LayerProperties3D layerProperties = manager.getLayerProperties(layer, null);
97

    
98
        NwwLoaderFactory loadingFactory = manager.getLayerLoadingFactory(layer);
99
        if (loadingFactory instanceof NwwRasterLoaderFactory) {
100
            NwwRasterLoaderFactory rasterLoaderFactory = (NwwRasterLoaderFactory) loadingFactory;
101
            rasterLoaderFactory.setMapControl(mapControl);
102
        }
103
        if (loadingFactory instanceof ElevationLayerLoaderFactory) {
104
            BasicElevationModel elevModel =
105
                ((ElevationLayerLoaderFactory) loadingFactory).loadElevationModel(layerProperties);
106
            return elevModel;
107
        }
108

    
109
        return null;
110
    }
111

    
112
    private String getUniqueCacheId(FLayer layer) {
113
        // TODO This is not valid. We have to take in account more properties
114
        // to get unique cache id
115
        try {
116
            // Create MD5 Hash
117
            MessageDigest digest = java.security.MessageDigest.getInstance("MD5");
118

    
119
            String name = layer.getName();
120
            long drawVersion = layer.getDrawVersion();
121
            String dataStoreFullName = null;
122
            if (layer instanceof SingleLayer) {
123
                dataStoreFullName = ((SingleLayer) layer).getDataStore().getFullName();
124
            }
125
            int hashCode = layer.hashCode();
126

    
127
            StringBuilder stb = new StringBuilder();
128
            stb.append(name);
129
            stb.append("$");
130
            stb.append(drawVersion);
131
            stb.append("$");
132
            stb.append(hashCode);
133
            stb.append("$");
134
            if (dataStoreFullName != null) {
135
                stb.append(dataStoreFullName);
136
                stb.append("$");
137
            }
138

    
139
            String id = stb.toString();
140
            digest.update(id.getBytes());
141
            byte messageDigest[] = digest.digest();
142

    
143
            // Create Hex String
144
            StringBuilder hexString = new StringBuilder();
145
            hexString.append(layer.getName());
146
            hexString.append("-");
147
            for (int i = 0; i < messageDigest.length; i++)
148
                hexString.append(Integer.toHexString(0xFF & messageDigest[i]));
149
            return hexString.toString();
150

    
151
        } catch (NoSuchAlgorithmException e) {
152
            LOG.warn("MD5 algorith is not available", e);
153

    
154
            String name = layer.getName();
155
            long drawVersion = layer.getDrawVersion();
156
            String dataStoreFullName = null;
157
            if (layer instanceof SingleLayer) {
158
                dataStoreFullName = ((SingleLayer) layer).getDataStore().getFullName();
159
            }
160
            int hashCode = layer.hashCode();
161

    
162
            StringBuilder stb = new StringBuilder();
163
            stb.append(name);
164
            stb.append("$");
165
            stb.append(drawVersion);
166
            stb.append("$");
167
            stb.append(hashCode);
168
            stb.append("$");
169
            if (dataStoreFullName != null) {
170
                stb.append(dataStoreFullName);
171
                stb.append("$");
172
            }
173

    
174
            return stb.toString();
175
        }
176
    }
177

    
178
    private void createRasterTiledImageLayerParams(FLayer layer, AVList params,
179
        RasterLayerProperties3D layerProperties) {
180

    
181
        if (params == null) {
182
            params = new AVListImpl();
183
        }
184

    
185
        setBasicParameters(layer, params);
186

    
187
        setSectorParameters(layer, params);
188

    
189
        // Set resolution of layer
190
        params.setValue(AVKey.WIDTH, Integer.MAX_VALUE);
191
        params.setValue(AVKey.HEIGHT, Integer.MAX_VALUE);
192
        params.setValue(AVKey.IMAGE_FORMAT, "image/png");
193
        params.setValue(AVKey.PIXEL_FORMAT, AVKey.IMAGE);
194
        params.setValue(AVKey.TEXTURE_FORMAT, "image/dds");
195
        params.setValue(AVKey.FORMAT_SUFFIX, ".png");
196
        params.setValue(AVKey.AVAILABLE_IMAGE_FORMATS, new String[] { params.getValue(AVKey.IMAGE_FORMAT).toString() });
197
        params.setValue(AVKey.NETWORK_RETRIEVAL_ENABLED, false);
198
        params.setValue(AVKey.USE_MIP_MAPS, true);
199
        params.setValue(AVKey.USE_TRANSPARENT_TEXTURES, true);
200

    
201
        Sector sector = (Sector) params.getValue(AVKey.SECTOR);
202

    
203
        setTileParameters(layer, sector, params, layerProperties);
204

    
205
        setDetailLevelParameters(layer, sector, params, layerProperties);
206
    }
207

    
208
    private void createElevationModelParams(FLayer layer, AVList params) {
209

    
210
        if (params == null) {
211
            params = new AVListImpl();
212
        }
213

    
214
        setBasicParameters(layer, params);
215

    
216
        setSectorParameters(layer, params);
217

    
218
        if (layer instanceof FLyrRaster) {
219

    
220
            FLyrRaster rasterLayer = (FLyrRaster) layer;
221
            int[] dataType = rasterLayer.getDataStore().getDataType();
222

    
223
            switch (dataType[0]) {
224
            case Buffer.TYPE_BYTE:
225
                params.setValue(AVKey.DATA_TYPE, AVKey.INT8);
226
                break;
227
            case Buffer.TYPE_SHORT:
228
            case Buffer.TYPE_USHORT:
229
                params.setValue(AVKey.DATA_TYPE, AVKey.INT16);
230
                break;
231
            case Buffer.TYPE_INT:
232
                params.setValue(AVKey.DATA_TYPE, AVKey.INT32);
233
                break;
234
            case Buffer.TYPE_FLOAT:
235
                params.setValue(AVKey.DATA_TYPE, AVKey.FLOAT32);
236
                break;
237
            case Buffer.TYPE_DOUBLE:
238
                params.setValue(AVKey.DATA_TYPE, AVKey.FLOAT64);
239
                break;
240
            case Buffer.TYPE_UNDEFINED:
241
            default:
242
                return;
243
            }
244

    
245
            params.setValue(AVKey.WIDTH, (int) rasterLayer.getDataStore().getWidth());
246
            params.setValue(AVKey.HEIGHT, (int) rasterLayer.getDataStore().getHeight());
247

    
248
            Statistics statistics = rasterLayer.getDataStore().getStatistics();
249
            params.setValue(AVKey.ELEVATION_MIN, statistics.getMin()[0]);
250
            params.setValue(AVKey.ELEVATION_MAX, statistics.getMax()[0]);
251

    
252
        } else {
253

    
254
            // TODO Vectorial elevation
255
        }
256

    
257
        View3DManager manager = View3DLocator.getManager();
258
        LayerProperties3D layerProperties = manager.getLayerProperties(layer, null);
259
        ElevationLayerProperties3D elevLayerProperties;
260
        if (!(layerProperties instanceof ElevationLayerProperties3D)) {
261
            elevLayerProperties = new DefaultElevationLayerProperties3D();
262
        } else {
263
            elevLayerProperties = (ElevationLayerProperties3D) layerProperties;
264
        }
265

    
266
        params.setValue(AVKey.PIXEL_FORMAT, AVKey.ELEVATION);
267
        params.setValue(AVKey.IMAGE_FORMAT, "application/bil32");
268
        params.setValue(AVKey.MISSING_DATA_SIGNAL, elevLayerProperties.getNoDataValue());
269
        params.setValue(AVKey.ELEVATION_UNIT, elevLayerProperties.getElevationUnits());
270
        params.setValue(AVKey.BYTE_ORDER, AVKey.LITTLE_ENDIAN);
271
        params.setValue(AVKey.AVAILABLE_IMAGE_FORMATS, new String[] { params.getValue(AVKey.IMAGE_FORMAT).toString() });
272

    
273
        Sector sector = (Sector) params.getValue(AVKey.SECTOR);
274

    
275
        setTileParameters(layer, sector, params, elevLayerProperties);
276

    
277
        setDetailLevelParameters(layer, sector, params, elevLayerProperties);
278
    }
279

    
280
    private void setDetailLevelParameters(FLayer layer, Sector sector, AVList params,
281
        RasterLayerProperties3D layerProperties) {
282

    
283
        int numLevels;
284
        // If deatail levels are 0, compute best level of detail number
285
        if (layerProperties.getMinLevel() == 0 && layerProperties.getMaxLevel() == 0) {
286

    
287
            LatLon rasterTileDelta = this.computeRasterTileDelta(params);
288
            LatLon desiredLevelZeroDelta = this.computeDesiredTileDelta(sector);
289
            numLevels = computeNumLevels(desiredLevelZeroDelta, rasterTileDelta);
290

    
291
        } else {
292
            numLevels = layerProperties.getMaxLevel() - layerProperties.getMinLevel();
293
            params.setValue(AVKey.NUM_EMPTY_LEVELS, layerProperties.getMinLevel());
294

    
295
        }
296
        params.setValue(AVKey.NUM_LEVELS, numLevels);
297

    
298
        if (layer instanceof FLyrVect) {
299

    
300
            // Only rasterized vectorial layers
301
            double degrees = DEFAULT_RASTERIZED_LEVEL_ZERO * layerProperties.getLevelZeroResolutionMultiplier();
302
            params.setValue(AVKey.LEVEL_ZERO_TILE_DELTA,
303
                new LatLon(Angle.fromDegrees(degrees), Angle.fromDegrees(degrees)));
304

    
305
        } else {
306

    
307
            LatLon rasterTileDelta = this.computeRasterTileDelta(params);
308
            double scale = Math.pow(2d, numLevels - 1);
309
            LatLon levelZeroTileDelta = LatLon.fromDegrees(
310
                scale * layerProperties.getLevelZeroResolutionMultiplier() * rasterTileDelta.getLatitude().degrees,
311
                scale * layerProperties.getLevelZeroResolutionMultiplier() * rasterTileDelta.getLongitude().degrees);
312
            params.setValue(AVKey.LEVEL_ZERO_TILE_DELTA, levelZeroTileDelta);
313
        }
314

    
315
    }
316

    
317
    private void setSectorParameters(FLayer layer, AVList params) {
318
        MapControl3D mapControl3D = (MapControl3D) params.getValue(DefaultMapControl3D.GVSIG_MAPCONTROL3D);
319
        Sector sector = null;
320
        IProjection projection = mapControl3D.getMapContext().getProjection();
321
        try {
322
            sector = getSector(layer, projection);
323
        } catch (DataException e) {
324
            LOG.info("Can't create sector from layer {} with projection {}", new Object[] { layer, projection }, e);
325
            return;
326
        }
327

    
328
        params.setValue(AVKey.SECTOR, sector);
329
    }
330

    
331
    private void setTileParameters(FLayer layer, Sector sector, AVList params,
332
        RasterLayerProperties3D layerProperties) {
333

    
334
        params.setValue(AVKey.TILE_ORIGIN, new LatLon(sector.getMinLatitude(), sector.getMinLongitude()));
335
        // If tile sizes are 0, set defautlt values
336
        params.setValue(AVKey.TILE_WIDTH,
337
            layerProperties.getTileWidth() == 0 ? DEFAULT_TILE_WIDTH : layerProperties.getTileWidth());
338
        params.setValue(AVKey.TILE_HEIGHT,
339
            layerProperties.getTileHeight() == 0 ? DEFAULT_TILE_HEIGHT : layerProperties.getTileHeight());
340

    
341
    }
342

    
343
    private void setBasicParameters(FLayer layer, AVList params) {
344
        params.setValue(AVKey.DATASET_NAME, layer.getName());
345
        params.setValue(AVKey.DATA_CACHE_NAME, getUniqueCacheId(layer));
346
        params.setValue(AVKey.DISPLAY_NAME, layer.getName());
347
        params.setValue(AVKey.SERVICE_NAME, "Offline");
348
    }
349

    
350
    private Sector getSector(FLayer layer, IProjection projection) throws DataException {
351

    
352
        ICoordTrans ct = projection.getCT(CRSFactory.getCRS("EPSG:4326"));
353
        Point2D p1 = null;
354
        Point2D p2 = null;
355
        Envelope envelope = layer.getFullEnvelope();
356

    
357
        p1 = convert(ct, new Point2D.Double(envelope.getMinimum(0), envelope.getMinimum(1)));
358
        p2 = convert(ct, new Point2D.Double(envelope.getMaximum(0), envelope.getMaximum(1)));
359

    
360
        double minLatitude =
361
            Double.isInfinite(p1.getY()) || p1.getY() < Angle.NEG90.degrees ? Angle.NEG90.degrees : p1.getY();
362
        double maxLatitude =
363
            Double.isInfinite(p2.getY()) || p2.getY() > Angle.POS90.degrees ? Angle.POS90.degrees : p2.getY();
364
        double minLongitude =
365
            Double.isInfinite(p1.getX()) || p1.getX() < Angle.NEG180.degrees ? Angle.NEG180.degrees : p1.getX();
366
        double maxLongitude =
367
            Double.isInfinite(p2.getX()) || p2.getX() > Angle.POS180.degrees ? Angle.POS180.degrees : p2.getX();
368

    
369
        Sector sector = new Sector(Sector.fromDegrees(minLatitude, maxLatitude, minLongitude, maxLongitude));
370

    
371
        return sector;
372
    }
373

    
374
    private int computeNumLevels(LatLon levelZeroDelta, LatLon lastLevelDelta) {
375
        // Compute the number of levels needed to achieve the given last level
376
        // tile delta, starting from the given
377
        // level zero tile delta.
378
        double numLatLevels = WWMath.logBase2(levelZeroDelta.getLatitude().getDegrees())
379
            - WWMath.logBase2(lastLevelDelta.getLatitude().getDegrees());
380
        double numLonLevels = WWMath.logBase2(levelZeroDelta.getLongitude().getDegrees())
381
            - WWMath.logBase2(lastLevelDelta.getLongitude().getDegrees());
382

    
383
        // Compute the maximum number of levels needed, but limit the number of
384
        // levels to positive integers greater
385
        // than or equal to one.
386
        int numLevels = (int) Math.ceil(Math.max(numLatLevels, numLonLevels));
387
        if (numLevels < 1)
388
            numLevels = 1;
389

    
390
        return numLevels;
391
    }
392

    
393
    private LatLon computeRasterTileDelta(AVList parameters) {
394

    
395
        Sector sector = (Sector) parameters.getValue(AVKey.SECTOR);
396
        int width = (Integer) parameters.getValue(AVKey.WIDTH);
397
        int height = (Integer) parameters.getValue(AVKey.HEIGHT);
398
        int tileWidth = (Integer) parameters.getValue(AVKey.TILE_WIDTH);
399
        int tileHeight = (Integer) parameters.getValue(AVKey.TILE_HEIGHT);
400

    
401
        LatLon pixelSize = this.computeRasterPixelSize(sector, width, height);
402
        // Compute the tile size in latitude and longitude, given a raster's
403
        // sector and dimension, and the tile
404
        // dimensions. In this computation a pixel is assumed to cover a finite
405
        // area.
406
        double latDelta = tileHeight * pixelSize.getLatitude().degrees;
407
        double lonDelta = tileWidth * pixelSize.getLongitude().degrees;
408
        return LatLon.fromDegrees(latDelta, lonDelta);
409
    }
410

    
411
    private LatLon computeRasterPixelSize(Sector sector, int width, int height) {
412
        // Compute the raster's pixel dimension in latitude and longitude. In
413
        // this computation a pixel is assumed to
414
        // cover a finite area.
415
        return LatLon.fromDegrees(sector.getDeltaLatDegrees() / height, sector.getDeltaLonDegrees() / width);
416
    }
417

    
418
    private LatLon computeDesiredTileDelta(Sector sector) {
419
        return LatLon.fromDegrees(sector.getDeltaLatDegrees(), sector.getDeltaLonDegrees());
420
    }
421

    
422
    private Point2D convert(ICoordTrans ct, Point2D point) {
423
        if (ct == null) {
424
            return point;
425
        }
426

    
427
        return ct.convert(point, null);
428
    }
429

    
430
}