Statistics
| Revision:

svn-gvsig-desktop / trunk / org.gvsig.desktop / org.gvsig.desktop.compat.cdc / org.gvsig.fmap.mapcontext / org.gvsig.fmap.mapcontext.api / src / main / java / org / gvsig / fmap / mapcontext / layers / FLayers.java @ 42543

History | View | Annotate | Download (43.9 KB)

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

    
25
import java.awt.Graphics2D;
26
import java.awt.Point;
27
import java.awt.image.BufferedImage;
28
import java.util.ArrayList;
29
import java.util.Collection;
30
import java.util.Collections;
31
import java.util.Iterator;
32
import java.util.List;
33
import java.util.ListIterator;
34
import java.util.Set;
35
import java.util.TreeSet;
36

    
37
import org.cresques.cts.ICoordTrans;
38
import org.cresques.cts.IProjection;
39
import org.slf4j.Logger;
40
import org.slf4j.LoggerFactory;
41

    
42
import org.gvsig.compat.print.PrintAttributes;
43
import org.gvsig.fmap.dal.exception.DataException;
44
import org.gvsig.fmap.dal.exception.ReadException;
45
import org.gvsig.fmap.geom.primitive.Envelope;
46
import org.gvsig.fmap.mapcontext.MapContext;
47
import org.gvsig.fmap.mapcontext.MapContextRuntimeException;
48
import org.gvsig.fmap.mapcontext.Messages;
49
import org.gvsig.fmap.mapcontext.ViewPort;
50
import org.gvsig.fmap.mapcontext.exceptions.LoadLayerException;
51
import org.gvsig.fmap.mapcontext.layers.operations.ComposedLayer;
52
import org.gvsig.fmap.mapcontext.layers.operations.InfoByPoint;
53
import org.gvsig.fmap.mapcontext.layers.operations.LayerCollection;
54
import org.gvsig.fmap.mapcontext.layers.operations.LayerNotFoundInCollectionException;
55
import org.gvsig.fmap.mapcontext.layers.operations.LayersVisitable;
56
import org.gvsig.fmap.mapcontext.layers.operations.LayersVisitor;
57
import org.gvsig.fmap.mapcontext.layers.vectorial.FLyrVect;
58
import org.gvsig.fmap.mapcontext.rendering.legend.styling.ILabelable;
59
import org.gvsig.metadata.exceptions.MetadataException;
60
import org.gvsig.tools.ToolsLocator;
61
import org.gvsig.tools.dispose.Disposable;
62
import org.gvsig.tools.dynobject.DynObjectSet;
63
import org.gvsig.tools.dynobject.DynStruct;
64
import org.gvsig.tools.dynobject.impl.MultiDynObjectSet;
65
import org.gvsig.tools.exception.BaseException;
66
import org.gvsig.tools.persistence.PersistenceManager;
67
import org.gvsig.tools.persistence.PersistentState;
68
import org.gvsig.tools.persistence.exception.PersistenceException;
69
import org.gvsig.tools.task.Cancellable;
70
import org.gvsig.tools.util.Callable;
71
import org.gvsig.tools.visitor.VisitCanceledException;
72
import org.gvsig.tools.visitor.Visitor;
73

    
74
/**
75
 * <p>
76
 * Represents a generic collection of layers, that can be represented as a node
77
 * in a tree of nodes of layers.</p>
78
 *
79
 * <p>
80
 * Adapts the basic functionality implemented for a layer in the abstract class
81
 * <code>FLyrDefault</code>, to a collection of layers, implementing, as well,
82
 * specific methods for this kind of object, defined in the interfaces
83
 * <code>VectorialData</code>, <code>LayerCollection</code>, and
84
 * <code>InfoByPoint</code>.</p>
85
 *
86
 * @see FLyrDefault
87
 */
88
public class FLayers extends FLyrDefault implements LayerCollection,
89
        InfoByPoint, List<FLayer> {
90

    
91
    /**
92
     * List with all listeners registered for this kind of node.
93
     */
94
    protected ArrayList layerCollectionListeners = null;
95

    
96
    /**
97
     * A synchronized list with the layers.
98
     */
99
    protected List<FLayer> layers = null;
100

    
101
    protected MapContext fmap;
102

    
103
    /**
104
     * Useful for debug the problems during the implementation.
105
     */
106
    private static final Logger logger = LoggerFactory.getLogger(FLayers.class);
107

    
108
    public FLayers() {
109
        super();
110
        layerCollectionListeners = new ArrayList();
111
        layers = Collections.synchronizedList(new ArrayList());
112
    }
113
    /*
114
     * (non-Javadoc)
115
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#addLayerCollectionListener(com.iver.cit.gvsig.fmap.layers.LayerCollectionListener)
116
     */
117

    
118
    @Override
119
    public void addLayerCollectionListener(LayerCollectionListener listener) {
120
        if (!layerCollectionListeners.contains(listener)) {
121
            layerCollectionListeners.add(listener);
122
        }
123
    }
124

    
125
    /*
126
     * (non-Javadoc)
127
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#setAllVisibles(boolean)
128
     */
129
    @Override
130
    public void setAllVisibles(boolean visible) {
131
        FLayer lyr;
132

    
133
        for (int i = 0; i < layers.size(); i++) {
134
            lyr = ((FLayer) layers.get(i));
135
            lyr.setVisible(visible);
136

    
137
            if (lyr instanceof LayerCollection) {
138
                ((LayerCollection) lyr).setAllVisibles(visible);
139
            }
140
        }
141
    }
142

    
143
    /*
144
     * (non-Javadoc)
145
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#removeLayerCollectionListener(com.iver.cit.gvsig.fmap.layers.LayerCollectionListener)
146
     */
147
    @Override
148
    public void removeLayerCollectionListener(LayerCollectionListener listener) {
149
        layerCollectionListeners.remove(listener);
150
    }
151

    
152
    /**
153
     * Adds a layer on an specified position in this node.
154
     *
155
     * @param pos position in the inner list where the layer will be added
156
     * @param layer a layer
157
     */
158
    private void doAddLayer(int pos, FLayer layer) {
159
        layers.add(pos, layer);
160
        ToolsLocator.getDisposableManager().bind(layer);
161
        layer.setParentLayer(this);
162
        IProjection layerProj = layer.getProjection();
163
        if (layerProj != null && fmap != null) {
164
            IProjection mapContextProj = fmap.getProjection();
165
            // TODO REVISAR ESTO !!!!
166
            // Esta condici?n puede que no fuese exacta para todos los casos
167
            if (!layerProj.getAbrev().equals(mapContextProj.getAbrev())) {
168
                ICoordTrans ct = layerProj.getCT(mapContextProj);
169
                layer.setCoordTrans(ct);
170
            } else {
171
                layer.setCoordTrans(null);
172
            }
173
        }
174
        this.updateDrawVersion();
175
    }
176

    
177
    /*
178
     * (non-Javadoc)
179
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#addLayer(com.iver.cit.gvsig.fmap.layers.FLayer)
180
     */
181
    @Override
182
    public void addLayer(FLayer layer) {
183

    
184
        MapContext mco = this.getMapContext();
185

    
186
        if (mco != null) {
187
            /*
188
             * Get order manager from map context
189
             */
190
            int position = mco.getOrderManager().getPosition(this, layer);
191
            addLayer(position, layer);
192
        } else {
193
            /*
194
             * This happens when FLayers object is not in a
195
             * map context, so no order manager is available.
196
             */
197
            addLayer(layers.size(), layer);
198
        }
199
    }
200

    
201
    /**
202
     * Adds a layer in an specified position in this node.
203
     *
204
     * @param pos
205
     * @param layer a layer
206
     */
207
    public void addLayer(int pos, FLayer layer) {
208
        try {
209
            //Notificamos a la capa que va a ser a?adida
210
            if (layer instanceof FLyrDefault) {
211
                ((FLyrDefault) layer).wakeUp();
212
            }
213

    
214
            if (layer instanceof FLayers) {
215
                FLayers layers = (FLayers) layer;
216
                fmap.addAsCollectionListener(layers);
217
            }
218
            callLayerAdding(LayerCollectionEvent.createLayerAddingEvent(layer));
219

    
220
            doAddLayer(pos, layer);
221

    
222
            callLayerAdded(LayerCollectionEvent.createLayerAddedEvent(layer));
223
        } catch (CancelationException e) {
224
            logger.warn(e.getMessage());
225
        } catch (LoadLayerException e) {
226
            layer.setAvailable(false);
227
            layer.addError(e);
228
        }
229
    }
230

    
231
    /*
232
     * (non-Javadoc)
233
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#moveTo(int, int)
234
     */
235
    @Override
236
    public void moveTo(int from, int to) throws CancelationException {
237
        int newfrom = layers.size() - from - 1;
238
        int newto = layers.size() - to - 1;
239
        if (newfrom < 0 || newfrom >= layers.size() || newto < 0 || newto >= layers.size()) {
240
            return;
241
        }
242
        FLayer aux = (FLayer) layers.get(newfrom);
243
        callLayerMoving(LayerPositionEvent.createLayerMovingEvent(aux, newfrom, newto));
244
        layers.remove(newfrom);
245
        layers.add(newto, aux);
246
        this.updateDrawVersion();
247
        callLayerMoved(LayerPositionEvent.createLayerMovedEvent(aux, newfrom, newto));
248
    }
249

    
250
    /**
251
     * Removes an inner layer.
252
     *
253
     * @param lyr a layer
254
     */
255
    private void doRemoveLayer(FLayer lyr) {
256
        layers.remove(lyr);
257
        lyr.dispose();
258
        this.updateDrawVersion();
259
    }
260

    
261
    /*
262
     * (non-Javadoc)
263
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#removeLayer(com.iver.cit.gvsig.fmap.layers.FLayer)
264
     */
265
    @Override
266
    public void removeLayer(FLayer lyr) throws CancelationException {
267
        callLayerRemoving(LayerCollectionEvent.createLayerRemovingEvent(lyr));
268
        doRemoveLayer(lyr);
269
        callLayerRemoved(LayerCollectionEvent.createLayerRemovedEvent(lyr));
270
    }
271

    
272
    /*
273
     * (non-Javadoc)
274
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#removeLayer(int)
275
     */
276
    @Override
277
    public void removeLayer(int idLayer) {
278
        FLayer lyr = (FLayer) layers.get(idLayer);
279
        callLayerRemoving(LayerCollectionEvent.createLayerRemovingEvent(lyr));
280
        this.doRemoveLayer(lyr);
281
//                layers.remove(idLayer);
282
//                this.updateDrawVersion();
283
        callLayerRemoved(LayerCollectionEvent.createLayerRemovedEvent(lyr));
284
    }
285

    
286
    /*
287
     * (non-Javadoc)
288
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#removeLayer(java.lang.String)
289
     */
290
    @Override
291
    public void removeLayer(String layerName) {
292
        FLayer lyr;
293

    
294
        for (int i = 0; i < layers.size(); i++) {
295
            lyr = ((FLayer) layers.get(i));
296

    
297
            if (lyr.getName().compareToIgnoreCase(layerName) == 0) {
298
                removeLayer(i);
299

    
300
                break;
301
            }
302
        }
303
    }
304

    
305
    /**
306
     * Replace a layer identified by its name, by another.
307
     *
308
     * @param layerName the name of the layer to be replaced
309
     * @param layer the new layer
310
     * @throws org.gvsig.fmap.mapcontext.exceptions.LoadLayerException
311
     * @deprecated use {@link FLayers#replaceLayer(FLayer, FLayer)}
312
     */
313
    public void replaceLayer(String layerName, FLayer layer) throws LoadLayerException {
314
        replaceLayer(getLayer(layerName), layer);
315
    }
316

    
317
    /**
318
     * Replace a layer by another layer. It search recursively by all the
319
     * ILayerCollection nodes
320
     *
321
     * @param layer the layer to be replaced
322
     * @param newLayer the new layer
323
     * @throws org.gvsig.fmap.mapcontext.exceptions.LoadLayerException
324
     */
325
    public void replaceLayer(FLayer layer, FLayer newLayer) throws LoadLayerException {
326
        replaceLayer(this, layer, newLayer);
327
    }
328

    
329
    /**
330
     * Replace a layer by other layer. It search recursively by all the
331
     * ILayerCollection nodes
332
     *
333
     * @param parentLayer the parent layer
334
     * @param layer the layer to be replaced
335
     * @param newLayer the new layer
336
     * @throws LoadLayerException
337
     */
338
    private void replaceLayer(FLayers parentLayer, FLayer layer, FLayer newLayer) throws LoadLayerException {
339
        FLayer lyr;
340
        for (int i = 0; i < parentLayer.getLayersCount(); i++) {
341
            lyr = ((FLayer) parentLayer.getLayer(i));
342
            if (lyr.equals(layer)) {
343
                parentLayer.removeLayer(i);
344
                parentLayer.addLayer(i, newLayer);
345
                break;
346
            }
347
            if (lyr instanceof LayerCollection) {
348
                replaceLayer((FLayers) lyr, layer, newLayer);
349
            }
350
        }
351
    }
352

    
353
    /*
354
     * (non-Javadoc)
355
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#getVisibles()
356
     */
357
    @Override
358
    public FLayer[] getVisibles() {
359
        ArrayList array = new ArrayList();
360
        LayersIterator iter = new LayersIterator(this) {
361
            @Override
362
            public boolean evaluate(FLayer layer) {
363
                return layer.isVisible();
364
            }
365

    
366
        };
367

    
368
        while (iter.hasNext()) {
369
            array.add(iter.nextLayer());
370
        }
371

    
372
        return (FLayer[]) array.toArray(new FLayer[0]);
373
    }
374

    
375
    /*
376
     * (non-Javadoc)
377
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#getLayer(int)
378
     */
379
    @Override
380
    public FLayer getLayer(int index) {
381
        return (FLayer) layers.get(index);
382
    }
383

    
384
    /*
385
     * (non-Javadoc)
386
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#getLayer(java.lang.String)
387
     */
388
    @Override
389
    public FLayer getLayer(String layerName) {
390
        for (int i = 0; i < layers.size(); i++) {
391
            FLayer lyr = ((FLayer) layers.get(i));
392
            if (lyr.getName().compareToIgnoreCase(layerName) == 0) {
393
                return lyr;
394
            }
395
            if (lyr instanceof FLayers) {
396
                List layerList = toPlainList(lyr);
397
                for (int j = 0; j < layerList.size(); j++) {
398
                    FLayer lyr2 = ((FLayer) layerList.get(j));
399
                    if (lyr2.getName().compareToIgnoreCase(layerName) == 0) {
400
                        return lyr2;
401
                    }
402
                }
403
            }
404
        }
405
        return null;
406
    }
407

    
408
    private List toPlainList(FLayer layer) {
409
        return toPlainList(layer, new ArrayList());
410
    }
411

    
412
    private List toPlainList(FLayer layer, List result) {
413
        if (layer instanceof FLayers) {
414
            FLayers layerGroup = (FLayers) layer;
415
            for (int i = 0; i < layerGroup.getLayersCount(); i++) {
416
                toPlainList(layerGroup.getLayer(i), result);
417
            }
418
        } else {
419
            result.add(layer);
420
        }
421
        return result;
422
    }
423

    
424
    /*
425
     * (non-Javadoc)
426
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#getLayersCount()
427
     */
428
    @Override
429
    public int getLayersCount() {
430
        return layers.size();
431
    }
432

    
433
    /*
434
     * (non-Javadoc)
435
     * @see com.iver.cit.gvsig.fmap.layers.FLayer#draw(java.awt.image.BufferedImage, java.awt.Graphics2D, com.iver.cit.gvsig.fmap.ViewPort, com.iver.utiles.swing.threads.Cancellable, double)
436
     */
437
    @Override
438
    public void draw(BufferedImage image, Graphics2D g, ViewPort viewPort,
439
            Cancellable cancel, double scale) throws ReadException {
440
        // FIXME Arreglar este error
441
        throw new RuntimeException("Esto no deberia de llamarse");
442
    }
443

    
444
    /*
445
     * (non-Javadoc)
446
     *
447
     * @see com.iver.cit.gvsig.fmap.layers.FLayer#print(java.awt.Graphics2D,
448
     * com.iver.cit.gvsig.fmap.ViewPort,
449
     * com.iver.utiles.swing.threads.Cancellable, double,
450
     * javax.print.attribute.PrintAttributes)
451
     */
452
    @Override
453
    public void print(Graphics2D g, ViewPort viewPort, Cancellable cancel,
454
            double scale, PrintAttributes properties)
455
            throws ReadException {
456
        throw new RuntimeException("No deberia pasar por aqui");
457
    }
458

    
459
    public void print_old(Graphics2D g, ViewPort viewPort, Cancellable cancel,
460
            double scale, PrintAttributes properties)
461
            throws ReadException {
462
        this.print_old(g, viewPort, cancel, scale, properties, null);
463
    }
464

    
465
    /**
466
     * <p>
467
     * Checks all layers (each one as a sub-node of this node <i>collection of
468
     * layers</i>) of this collection and draws their requested properties. If a
469
     * node is a group of layers (<code>ComposedLayer</code>), executes it's
470
     * drawn.</p>
471
     *
472
     * <p>
473
     * All nodes which could group with the composed layer <code>group</code>,
474
     * will be drawn together. And once the <code>
475
     * group</code> is drawn, will be set to <code>null</code> if hasn't a
476
     * parent layer.</p>
477
     *
478
     * <p>
479
     * The particular implementation depends on the kind of each layer and
480
     * composed layer. And this process can be cancelled at any time by the
481
     * shared object <code>cancel</code>.</p>
482
     *
483
     * <p>
484
     * According the print quality, labels will be printed in different
485
     * resolution:
486
     * <ul>
487
     * <li><b>PrintQuality.DRAFT</b>: 72 dpi (dots per inch).</li>
488
     * <li><b>PrintQuality.NORMAL</b>: 300 dpi (dots per inch).</li>
489
     * <li><b>PrintQuality.HIGH</b>: 600 dpi (dots per inch).</li>
490
     * </ul>
491
     * </p>
492
     *
493
     * @param g for rendering 2-dimensional shapes, text and images on the
494
     * Java(tm) platform
495
     * @param viewPort the information for drawing the layers
496
     * @param cancel shared object that determines if this layer can continue
497
     * being drawn
498
     * @param scale the scale of the view. Must be between
499
     * {@linkplain FLayer#getMinScale()} and {@linkplain FLayer#getMaxScale()}.
500
     * @param properties properties that will be print
501
     * @param group a composed layer pending to paint; if this parameter is
502
     * <code>null</code>, the composed layer
503
     *
504
     * @return <code>null</code> if the layers in <code>group</code> had been
505
     * drawn or were <code>null</code>; otherwise, the <code>group</code>
506
     * @throws org.gvsig.fmap.dal.exception.ReadException
507
     *
508
     * @see FLayer#print(Graphics2D, ViewPort, Cancellable, double,
509
     * PrintAttributes)
510
     */
511
    public ComposedLayer print_old(Graphics2D g, ViewPort viewPort, Cancellable cancel, double scale, PrintAttributes properties, ComposedLayer group)
512
            throws ReadException {
513
        double dpi = 72;
514

    
515
        int resolution = properties.getPrintQuality();
516
        if (resolution == PrintAttributes.PRINT_QUALITY_NORMAL) {
517
            dpi = 300;
518
        } else if (resolution == PrintAttributes.PRINT_QUALITY_HIGH) {
519
            dpi = 600;
520
        } else if (resolution == PrintAttributes.PRINT_QUALITY_DRAFT) {
521
            dpi = 72;
522
        }
523

    
524
        // TODO: A la hora de imprimir, isWithinScale falla, porque est?
525
        // calculando la escala en pantalla, no para el layout.
526
        // Revisar esto.
527
        // TODO: We have to check when we have to call the drawLabels method when exists a ComposedLayer group.
528
        for (int i = 0; i < layers.size(); i++) {
529
            FLayer lyr = (FLayer) layers.get(i);
530
            if (!lyr.isVisible() || !lyr.isWithinScale(scale)) {
531
                continue;
532
            }
533

    
534
            try {
535

    
536
                // Checks for draw group (ComposedLayer)
537
                if (group != null) {
538
                    if (lyr instanceof FLayers) {
539
                        group = ((FLayers) lyr).print_old(g, viewPort, cancel, scale, properties, group);
540
                    } else {
541
                        // If layer can be added to the group, does it
542
                        if (lyr instanceof ILabelable
543
                                && ((ILabelable) lyr).isLabeled()
544
                                && ((ILabelable) lyr).getLabelingStrategy() != null
545
                                && ((ILabelable) lyr).getLabelingStrategy().shouldDrawLabels(scale)) {
546
                            group.add(lyr);
547
                        } else {
548
                            // draw the 'pending to draw' layer group
549
                            group.print(g, viewPort, cancel, scale, properties);
550

    
551
                            // gets a new group instance
552
                            if (lyr instanceof ILabelable
553
                                    && ((ILabelable) lyr).isLabeled()
554
                                    && ((ILabelable) lyr).getLabelingStrategy() != null
555
                                    && ((ILabelable) lyr).getLabelingStrategy().shouldDrawLabels(scale)) {
556
                                group = lyr.newComposedLayer();
557
                            } else {
558
                                group = null;
559
                            }
560
                            // if layer hasn't group, draws it inmediately
561
                            if (group == null) {
562
                                if (lyr instanceof FLayers) {
563
                                    group = ((FLayers) lyr).print_old(g, viewPort, cancel, scale, properties, group);
564
                                } else {
565
                                    lyr.print(g, viewPort, cancel, scale, properties);
566
                                    if (lyr instanceof ILabelable
567
                                            && ((ILabelable) lyr).isLabeled()
568
                                            && ((ILabelable) lyr).getLabelingStrategy() != null
569
                                            && ((ILabelable) lyr).getLabelingStrategy().shouldDrawLabels(scale)) {
570
                                        ILabelable lLayer = (ILabelable) lyr;
571
                                        lLayer.drawLabels(null, g, viewPort, cancel, scale, dpi);
572
                                    }
573
                                }
574
                            } else {
575
                                // add the layer to the group
576
                                group.setMapContext(fmap);
577
                                group.add(lyr);
578

    
579
                            }
580

    
581
                        }
582
                    }
583
                } else {
584
                    // gets a new group instance
585
                    group = lyr.newComposedLayer();
586
                    // if layer hasn't group, draws it inmediately
587
                    if (group == null) {
588
                        if (lyr instanceof FLayers) {
589
                            group = ((FLayers) lyr).print_old(g, viewPort, cancel, scale, properties, group);
590
                        } else {
591
                            lyr.print(g, viewPort, cancel, scale, properties);
592
                            if (lyr instanceof ILabelable && ((ILabelable) lyr).isLabeled()) {
593
                                ILabelable lLayer = (ILabelable) lyr;
594

    
595
                                lLayer.drawLabels(null, g, viewPort, cancel, scale, dpi);
596
                            }
597
                        }
598
                    } else {
599
                        // add the layer to the group
600
                        group.setMapContext(fmap);
601
                        group.add(lyr);
602

    
603
                    }
604
                }
605

    
606
            } catch (Exception e) {
607
                String mesg = Messages.getString("error_printing_layer") + " " + lyr.getName() + ": " + e.getMessage();
608
                fmap.addLayerError(mesg);
609
                logger.error(mesg, e);
610
            }
611

    
612
        }
613

    
614
        if (group != null && this.getParentLayer() == null) {
615
            //si tenemos un grupo pendiente de pintar, pintamos
616
            group.print(g, viewPort, cancel, scale, properties);
617
            group = null;
618

    
619
        }
620
        return group;
621
    }
622

    
623
    /*
624
     * (non-Javadoc)
625
     * @see com.iver.cit.gvsig.fmap.layers.FLayer#getFullExtent()
626
     */
627
    @Override
628
    public Envelope getFullEnvelope() {
629
        Envelope rAux = null;
630
        boolean first = true;
631

    
632
        for (Iterator iter = layers.iterator(); iter.hasNext();) {
633
            FLayer capa = (FLayer) iter.next();
634
            try {
635
                if (first) {
636
                    rAux = (Envelope) capa.getFullEnvelope().clone();
637
                    first = false;
638
                } else {
639
                    rAux.add(capa.getFullEnvelope());
640
                }
641
            } catch (Exception e) {
642
                logger.warn("Can't calculate the envelope of the layer group '"+this.getName()+"'.",e);
643
            }
644
        }
645

    
646
        return rAux;
647
    }
648

    
649
    /**
650
     * Notifies all listeners associated to this collection of layers, that
651
     * another layer is going to be added or replaced in the internal list of
652
     * layers.
653
     *
654
     * @param event a layer collection event with the new layer
655
     */
656
    protected void callLayerAdding(LayerCollectionEvent event)
657
            throws CancelationException {
658
        ArrayList aux = (ArrayList) layerCollectionListeners.clone();
659
        for (Iterator iter = aux.iterator(); iter.hasNext();) {
660
            ((LayerCollectionListener) iter.next()).layerAdding(event);
661
        }
662
    }
663

    
664
    /**
665
     * Notifies all listeners associated to this collection of layers, that a
666
     * layer is going to be removed from the internal list of layers.
667
     *
668
     * @param event a layer collection event with the layer being removed
669
     *
670
     * @throws CancelationException any exception produced during the
671
     * cancellation of the driver.
672
     */
673
    protected void callLayerRemoving(LayerCollectionEvent event)
674
            throws CancelationException {
675
        ArrayList aux = (ArrayList) layerCollectionListeners.clone();
676
        for (Iterator iter = aux.iterator(); iter.hasNext();) {
677
            ((LayerCollectionListener) iter.next()).layerRemoving(event);
678
        }
679
    }
680

    
681
    /**
682
     * Notifies all listeners associated to this collection of layers, that a
683
     * layer is going to be moved in the internal list of layers.
684
     *
685
     * @param event a layer collection event with the layer being moved, and the
686
     * initial and final positions
687
     *
688
     * @throws CancelationException any exception produced during the
689
     * cancellation of the driver.
690
     */
691
    protected void callLayerMoving(LayerPositionEvent event)
692
            throws CancelationException {
693
        ArrayList aux = (ArrayList) layerCollectionListeners.clone();
694
        for (Iterator iter = aux.iterator(); iter.hasNext();) {
695
            ((LayerCollectionListener) iter.next()).layerMoving(event);
696
        }
697
    }
698

    
699
    /**
700
     * Notifies all listeners associated to this collection of layers, that
701
     * another layer has been added or replaced in the internal list of layers.
702
     *
703
     * @param event a layer collection event with the new layer
704
     */
705
    protected void callLayerAdded(LayerCollectionEvent event) {
706
        ArrayList aux = (ArrayList) layerCollectionListeners.clone();
707
        for (Iterator iter = aux.iterator(); iter.hasNext();) {
708
            ((LayerCollectionListener) iter.next()).layerAdded(event);
709
        }
710
    }
711

    
712
    /**
713
     * Notifies all listeners associated to this collection of layers, that
714
     * another layer has been removed from the internal list of layers.
715
     *
716
     * @param event a layer collection event with the layer removed
717
     */
718
    protected void callLayerRemoved(LayerCollectionEvent event) {
719
        ArrayList aux = (ArrayList) layerCollectionListeners.clone();
720
        for (Iterator iter = aux.iterator(); iter.hasNext();) {
721
            ((LayerCollectionListener) iter.next()).layerRemoved(event);
722
        }
723
    }
724

    
725
    /**
726
     * Notifies all listeners associated to this collection of layers, that
727
     * another layer has been moved in the internal list of layers.
728
     *
729
     * @param event a layer collection event with the layer moved, and the initial
730
     * and final positions
731
     */
732
    protected void callLayerMoved(LayerPositionEvent event) {
733
        ArrayList aux = (ArrayList) layerCollectionListeners.clone();
734
        for (Iterator iter = aux.iterator(); iter.hasNext();) {
735
            ((LayerCollectionListener) iter.next()).layerMoved(event);
736
        }
737
    }
738

    
739
    @Override
740
    public void saveToState(PersistentState state) throws PersistenceException {
741

    
742
        super.saveToState(state);
743

    
744
        state.set("mapContext", fmap);
745

    
746
        List layersToSave = new ArrayList();
747
        Iterator iter = this.layers.iterator();
748
        while (iter.hasNext()) {
749
            FLayer layer = (FLayer) iter.next();
750
            if (!layer.isTemporary()) {
751
                layersToSave.add(layer);
752
            }
753
        }
754
        state.set("layers", layersToSave);
755
    }
756

    
757
    @Override
758
    public void loadFromState(PersistentState state) throws PersistenceException {
759

    
760
        super.loadFromState(state);
761

    
762
        setMapContext((MapContext) state.get("mapContext"));
763
        Iterator iter = state.getIterator("layers");
764
        while (iter.hasNext()) {
765
            FLayer item = (FLayer) iter.next();
766
            layers.add(item);
767
        }
768
    }
769

    
770
    /*
771
     * (non-Javadoc)
772
     * @see com.iver.cit.gvsig.fmap.layers.FLyrDefault#getMapContext()
773
     */
774
    @Override
775
    public MapContext getMapContext() {
776
        return fmap;
777
    }
778

    
779
    /*
780
     * (non-Javadoc)
781
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#setAllActives(boolean)
782
     */
783
    @Override
784
    public void setAllActives(boolean active) {
785
        FLayer lyr;
786

    
787
        for (int i = 0; i < layers.size(); i++) {
788
            lyr = ((FLayer) layers.get(i));
789
            lyr.setActive(active);
790

    
791
            if (lyr instanceof LayerCollection) {
792
                ((LayerCollection) lyr).setAllActives(active);
793
            }
794
        }
795
    }
796

    
797
    /*
798
     * (non-Javadoc)
799
     * @see com.iver.cit.gvsig.fmap.layers.layerOperations.LayerCollection#getActives()
800
     */
801
    @Override
802
    public FLayer[] getActives() {
803
        List ret = new ArrayList();
804
        LayersIterator it = new LayersIterator(this) {
805

    
806
            @Override
807
            public boolean evaluate(FLayer layer) {
808
                return layer.isActive();
809
            }
810

    
811
        };
812

    
813
        while (it.hasNext()) {
814
            ret.add(it.next());
815
        }
816
        return (FLayer[]) ret.toArray(new FLayer[0]);
817
    }
818

    
819
    /*
820
     * (non-Javadoc)
821
     * @see com.iver.cit.gvsig.fmap.layers.FLyrDefault#getMinScale()
822
     */
823
    @Override
824
    public double getMinScale() {
825
        return -1; // La visibilidad o no la controla cada capa
826
        // dentro de una colecci?n
827
    }
828
    /*
829
     * (non-Javadoc)
830
     * @see com.iver.cit.gvsig.fmap.layers.FLyrDefault#getMaxScale()
831
     */
832

    
833
    @Override
834
    public double getMaxScale() {
835
        return -1;
836
    }
837
    /*
838
     * (non-Javadoc)
839
     * @see com.iver.cit.gvsig.fmap.layers.FLyrDefault#setMinScale(double)
840
     */
841

    
842
    @Override
843
    public void setMinScale(double minScale) {
844
        for (Iterator iter = layers.iterator(); iter.hasNext();) {
845
            FLayer lyr = (FLayer) iter.next();
846
            lyr.setMinScale(minScale);
847
        }
848
    }
849
    /*
850
     * (non-Javadoc)
851
     * @see com.iver.cit.gvsig.fmap.layers.FLyrDefault#setMaxScale(double)
852
     */
853

    
854
    @Override
855
    public void setMaxScale(double maxScale) {
856
        for (Iterator iter = layers.iterator(); iter.hasNext();) {
857
            FLayer lyr = (FLayer) iter.next();
858
            lyr.setMinScale(maxScale);
859
        }
860
    }
861
    /*
862
     * (non-Javadoc)
863
     * @see com.iver.cit.gvsig.fmap.layers.FLyrDefault#setActive(boolean)
864
     */
865

    
866
    @Override
867
    public void setActive(boolean b) {
868
        super.setActive(b);
869
        for (int i = 0; i < layers.size(); i++) {
870
            ((FLayer) layers.get(i)).setActive(b);
871
        }
872
    }
873
    /* (non-Javadoc)
874
     * @see com.iver.cit.gvsig.fmap.layers.FLyrDefault#addLayerListener(com.iver.cit.gvsig.fmap.layers.LayerListener)
875
     */
876

    
877
    @Override
878
    public boolean addLayerListener(LayerListener o) {
879
        for (int i = 0; i < layers.size(); i++) {
880
            ((FLayer) layers.get(i)).addLayerListener(o);
881
        }
882
        return true;
883
    }
884

    
885
    @Override
886
    public DynObjectSet getInfo(Point p, double tolerance,
887
            Cancellable cancel) throws LoadLayerException, DataException {
888
        return getInfo(this.getMapContext().getViewPort().convertToMapPoint(p), tolerance);
889
    }
890

    
891
    @Override
892
    public DynObjectSet getInfo(Point p, double tolerance, Cancellable cancel,
893
            boolean fast) throws LoadLayerException, DataException {
894
        return getInfo(this.getMapContext().getViewPort().convertToMapPoint(p), tolerance);
895
    }
896

    
897
    @Override
898
    public DynObjectSet getInfo(org.gvsig.fmap.geom.primitive.Point p,
899
            double tolerance) throws LoadLayerException, DataException {
900
        int i;
901
        FLayer layer;
902
        List res = new ArrayList();
903
        for (i = 0; i < this.layers.size(); i++) {
904
            layer = (FLayer) layers.get(i);
905
            if (layer instanceof InfoByPoint) {
906
                InfoByPoint queryable_layer = (InfoByPoint) layer;
907
                res.add(queryable_layer.getInfo(p, tolerance));
908
            }
909
        }
910
        DynObjectSet[] innerSets
911
                = (DynObjectSet[]) res.toArray(new DynObjectSet[res.size()]);
912
        return new MultiDynObjectSet(innerSets);
913
    }
914

    
915
    /*
916
     * (non-Javadoc)
917
     * @see com.iver.cit.gvsig.fmap.layers.FLyrDefault#getTocImageIcon()
918
     */
919
    @Override
920
    public String getTocImageIcon() {
921
        return "layer-icon-group";
922
    }
923

    
924
    /**
925
     * <p>
926
     * Sets the <code>MapContext</code> that contains this layer node.</p>
927
     *
928
     * @param mapContext the <code>MapContext</code> that contains this layer
929
     * node
930
     */
931
    public void setMapContext(MapContext mapContext) {
932
        this.fmap = mapContext;
933
    }
934

    
935
    @Override
936
    public void accept(Visitor visitor) throws BaseException {
937
        for (int i = 0; i < this.getLayersCount(); i++) {
938
            FLayer layer = this.getLayer(i);
939
            try {
940
                if (layer instanceof LayersVisitable) {
941
                    ((LayersVisitable) layer).accept(visitor);
942
                } else {
943
                    visitor.visit(layer);
944
                }
945
            } catch (VisitCanceledException ex) {
946
                break;
947
            }
948
        }
949
    }
950

    
951
    @Override
952
    public void accept(LayersVisitor visitor) throws BaseException {
953
        for (int i = 0; i < this.getLayersCount(); i++) {
954
            FLayer layer = this.getLayer(i);
955
            if (layer instanceof LayersVisitable) {
956
                ((LayersVisitable) layer).accept(visitor);
957
            } else {
958
                visitor.visit(layer);
959
            }
960
        }
961
    }
962

    
963
    /*
964
     * (non-Javadoc)
965
     *
966
     * @see org.gvsig.metadata.Metadata#getMetadataID()
967
     */
968
    @Override
969
    public Object getMetadataID() throws MetadataException {
970
        StringBuilder strb = new StringBuilder();
971
        strb.append("Layers(");
972
        strb.append(this.getName());
973
        strb.append("):{");
974
        Iterator iter = this.layers.iterator();
975
        while (iter.hasNext()) {
976
            strb.append(((FLayer) iter.next()).getMetadataID());
977
            strb.append(",");
978
        }
979
        strb.append("}");
980
        return strb.toString();
981

    
982
    }
983

    
984
    /*
985
     * (non-Javadoc)
986
     *
987
     * @see org.gvsig.metadata.Metadata#getMetadataChildren()
988
     */
989
    @Override
990
    public Set getMetadataChildren() {
991
        Set ret = new TreeSet();
992
        Iterator iter = this.layers.iterator();
993
        while (iter.hasNext()) {
994
            ret.add(iter.next());
995
        }
996
        return ret;
997
    }
998

    
999
    /*
1000
     * (non-Javadoc)
1001
     *
1002
     * @see org.gvsig.metadata.Metadata#getMetadataName()
1003
     */
1004
    @Override
1005
    public String getMetadataName() throws MetadataException {
1006
        StringBuilder strb = new StringBuilder();
1007
        strb.append("Layer Group '");
1008
        strb.append(this.getName());
1009
        strb.append("': {");
1010
        Iterator iter = this.layers.iterator();
1011
        while (iter.hasNext()) {
1012
            strb.append(((FLayer) iter.next()).getMetadataName());
1013
            strb.append(",");
1014
        }
1015
        strb.append("}");
1016
        return strb.toString();
1017
    }
1018

    
1019
    @Override
1020
    public void beginDraw(Graphics2D g, ViewPort viewPort) {
1021
        LayerDrawEvent beforeEvent = new LayerDrawEvent(this, g, viewPort, LayerDrawEvent.LAYER_BEFORE_DRAW);
1022
        fmap.fireLayerDrawingEvent(beforeEvent);
1023
    }
1024

    
1025
    @Override
1026
    public void endDraw(Graphics2D g, ViewPort viewPort) {
1027
        LayerDrawEvent afterEvent = new LayerDrawEvent(this, g, viewPort, LayerDrawEvent.LAYER_AFTER_DRAW);
1028
        fmap.fireLayerDrawingEvent(afterEvent);
1029
    }
1030

    
1031
    public static class RegisterPersistence implements Callable {
1032

    
1033
        @Override
1034
        public Object call() {
1035

    
1036
            PersistenceManager manager = ToolsLocator.getPersistenceManager();
1037
            DynStruct definition = manager.addDefinition(
1038
                    FLayers.class,
1039
                    "FLayers",
1040
                    "FLayers Persistence definition",
1041
                    null,
1042
                    null
1043
            );
1044
            definition.extend(PersistenceManager.PERSISTENCE_NAMESPACE, "FLyrDefault");
1045

    
1046
            definition.addDynFieldObject("mapContext").setClassOfValue(MapContext.class).setMandatory(true);
1047
            definition.addDynFieldList("layers").setClassOfItems(FLayer.class).setMandatory(true);
1048

    
1049
            return Boolean.TRUE;
1050
        }
1051
    }
1052

    
1053
    @Override
1054
    protected void doDispose() throws BaseException {
1055
        if (layers != null) {
1056
            for (int i = 0; i < layers.size(); i++) {
1057
                dispose((Disposable) layers.get(i));
1058
            }
1059
        }
1060
    }
1061

    
1062
    @Override
1063
    public void move(FLayer layer, LayerCollection group, int where, FLayer adjoiningLayer) throws LayerNotFoundInCollectionException {
1064

    
1065
        callLayerRemoving(LayerCollectionEvent.createLayerRemovingEvent(layer));
1066
        group.addLayer(layer, where, adjoiningLayer);
1067
        removeLayer(layer);
1068
        this.updateDrawVersion();
1069
        callLayerRemoved(LayerCollectionEvent.createLayerRemovedEvent(layer));
1070

    
1071
    }
1072

    
1073
    public void join(FLayer layer, LayerCollection group) {
1074
        try {
1075
            layers.remove(layer);
1076
            group.addLayer(layer, END, null);
1077
            this.updateDrawVersion();
1078
        } catch (LayerNotFoundInCollectionException e) {
1079
            throw new MapContextRuntimeException(e);
1080
        }
1081
    }
1082

    
1083
    @Override
1084
    public void move(FLayer layer, LayerCollection group) {
1085
        try {
1086
            move(layer, group, END, null);
1087
        } catch (LayerNotFoundInCollectionException e) {
1088
            throw new MapContextRuntimeException(e);
1089
        }
1090
    }
1091

    
1092
    @Override
1093
    public void addLayer(FLayer layer, int where, FLayer adjoiningLayer)
1094
            throws LayerNotFoundInCollectionException {
1095

    
1096
        switch (where) {
1097
            case BEGIN:
1098
                addLayer(0, layer);
1099
                break;
1100
            case BEFORE:
1101
                if (adjoiningLayer != null) {
1102
                    if (this.layers.contains(adjoiningLayer)) {
1103
                        for (int i = 0; i < this.getLayersCount(); i++) {
1104
                            if (adjoiningLayer == this.getLayer(i)) {
1105
                                addLayer(i, layer);
1106
                                break;
1107
                            }
1108
                        }
1109
                    } else {
1110
                        throw new LayerNotFoundInCollectionException(adjoiningLayer, this);
1111
                    }
1112
                } else {
1113
                    addLayer(0, layer);
1114
                }
1115
                break;
1116
            case AFTER:
1117
                if (adjoiningLayer != null) {
1118
                    if (this.layers.contains(adjoiningLayer)) {
1119
                        for (int i = 0; i < this.getLayersCount(); i++) {
1120
                            if (adjoiningLayer == this.getLayer(i)) {
1121
                                addLayer(i + 1, layer);
1122
                                break;
1123
                            }
1124
                        }
1125
                    } else {
1126
                        throw new LayerNotFoundInCollectionException(adjoiningLayer, this);
1127
                    }
1128
                } else {
1129
                    this.addLayer(layer);
1130
                }
1131
                break;
1132
            default: // By default add layer an the end of the collection
1133
                this.addLayer(layer);
1134
                break;
1135
        }
1136

    
1137
    }
1138

    
1139
    public FLayer getFirstActiveLayer() {
1140
        LayersIterator it = new LayersIterator(this) {
1141
            @Override
1142
            public boolean evaluate(FLayer layer) {
1143
                return layer.isActive();
1144
            }
1145
        };
1146
        if( it.hasNext() ) {
1147
            return (FLayer) it.next();
1148
        }
1149
        return null;
1150
    }
1151

    
1152
    public FLyrVect getFirstActiveVectorLayer() {
1153
        LayersIterator it = new LayersIterator(this) {
1154
            @Override
1155
            public boolean evaluate(FLayer layer) {
1156
                return layer.isActive() && layer instanceof FLyrVect;
1157
            }
1158
        };
1159
        if( it.hasNext() ) {
1160
            return (FLyrVect) it.next();
1161
        }
1162
        return null;
1163
    }
1164
    
1165
    @Override
1166
    public Iterator iterator() {
1167
        return this.layers.iterator();
1168
    }
1169

    
1170
    public Iterator deepiterator() {
1171
        List layers = toPlainList(this);
1172
        return layers.iterator();
1173
    }
1174
    
1175
    @Override
1176
    public int size() {
1177
        return this.layers.size();
1178
    }
1179
    
1180
    @Override
1181
    public FLayer get(int index) {
1182
        return this.layers.get(index);
1183
    }
1184
    
1185
    @Override
1186
    public boolean isEmpty() {
1187
        return this.layers.isEmpty();
1188
    }
1189

    
1190
    @Override
1191
    public boolean contains(Object o) {
1192
        return this.layers.contains(o);
1193
    }
1194

    
1195
    @Override
1196
    public Object[] toArray() {
1197
        return this.layers.toArray();
1198
    }
1199

    
1200
    @Override
1201
    public Object[] toArray(Object[] ts) {
1202
        return this.layers.toArray(ts);
1203
    }
1204

    
1205
    @Override
1206
    public boolean add(FLayer e) {
1207
        this.addLayer(e);
1208
        return true;
1209
    }
1210

    
1211
    @Override
1212
    public boolean remove(Object o) {
1213
        this.removeLayer((FLayer) o);
1214
        return true;
1215
    }
1216

    
1217
    @Override
1218
    public boolean containsAll(Collection clctn) {
1219
        return this.layers.containsAll(clctn);
1220
    }
1221

    
1222
    @Override
1223
    public void add(int i, FLayer e) {
1224
        this.addLayer(i, (FLayer) e);
1225
    }
1226

    
1227
    @Override
1228
    public FLayer remove(int i) {
1229
        FLayer o = this.getLayer(i);
1230
        this.removeLayer(i);
1231
        return o;
1232
    }
1233

    
1234
    @Override
1235
    public int indexOf(Object o) {
1236
        return this.layers.indexOf(o);
1237
    }
1238

    
1239
    @Override
1240
    public int lastIndexOf(Object o) {
1241
        return this.layers.lastIndexOf(o);
1242
    }
1243

    
1244
    @Override
1245
    public ListIterator listIterator() {
1246
        return this.layers.listIterator();
1247
    }
1248

    
1249
    @Override
1250
    public ListIterator listIterator(int i) {
1251
        return this.layers.listIterator(i);
1252
    }
1253

    
1254
    @Override
1255
    public List subList(int i, int i1) {
1256
        return this.layers.subList(i, i1);
1257
    }
1258
    
1259
    @Override
1260
    public boolean addAll(Collection clctn) {
1261
        Iterator it = clctn.iterator();
1262
        while( it.hasNext() ) {
1263
            this.add((FLayer) it.next());
1264
        }
1265
        return true;
1266
    }
1267

    
1268
    @Override
1269
    public boolean addAll(int i, Collection clctn) {
1270
        Iterator it = clctn.iterator();
1271
        while( it.hasNext() ) {
1272
            this.add(i, (FLayer) it.next());
1273
        }
1274
        return true;
1275
    }
1276

    
1277
    @Override
1278
    public boolean removeAll(Collection clctn) {
1279
        Iterator it = clctn.iterator();
1280
        while( it.hasNext() ) {
1281
            this.remove((FLayer) it.next());
1282
        }
1283
        return true;
1284
    }
1285

    
1286
    @Override
1287
    public boolean retainAll(Collection clctn) {
1288
        Iterator it = this.layers.iterator();
1289
        while( it.hasNext() ) {
1290
            Object o = it.next();
1291
            if( !clctn.contains(o) ) {
1292
                this.remove((FLayer) o);
1293
            }
1294
        }
1295
        return true;
1296
    }
1297

    
1298
    @Override
1299
    public FLayer set(int i, FLayer e) {
1300
        throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
1301
    }
1302

    
1303
}