Statistics
| Revision:

root / trunk / libraries / libFMap / src / com / iver / cit / gvsig / fmap / core / symbols / MultiLayerMarkerSymbol.java @ 28367

History | View | Annotate | Download (11.3 KB)

1
/* gvSIG. Sistema de Informaci?n Geogr?fica de la Generalitat Valenciana
2
 *
3
 * Copyright (C) 2005 IVER T.I. and Generalitat Valenciana.
4
 *
5
 * This program is free software; you can redistribute it and/or
6
 * modify it under the terms of the GNU General Public License
7
 * as published by the Free Software Foundation; either version 2
8
 * of the License, or (at your option) any later version.
9
 *
10
 * This program is distributed in the hope that it will be useful,
11
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13
 * GNU General Public License for more details.
14
 *
15
 * You should have received a copy of the GNU General Public License
16
 * along with this program; if not, write to the Free Software
17
 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307,USA.
18
 *
19
 * For more information, contact:
20
 *
21
 *  Generalitat Valenciana
22
 *   Conselleria d'Infraestructures i Transport
23
 *   Av. Blasco Ib??ez, 50
24
 *   46010 VALENCIA
25
 *   SPAIN
26
 *
27
 *      +34 963862235
28
 *   gvsig@gva.es
29
 *      www.gvsig.gva.es
30
 *
31
 *    or
32
 *
33
 *   IVER T.I. S.A
34
 *   Salamanca 50
35
 *   46005 Valencia
36
 *   Spain
37
 *
38
 *   +34 963163400
39
 *   dac@iver.es
40
 */
41

    
42
/* CVS MESSAGES:
43
*
44
* $Id: MultiLayerMarkerSymbol.java 28367 2009-05-04 15:30:10Z vcaballero $
45
* $Log$
46
* Revision 1.17  2007-09-21 12:25:32  jaume
47
* cancellation support extended down to the IGeometry and ISymbol level
48
*
49
* Revision 1.16  2007/09/19 16:21:32  jaume
50
* removed unnecessary imports
51
*
52
* Revision 1.13  2007/09/17 14:16:11  jaume
53
* multilayer symbols sizing bug fixed
54
*
55
* Revision 1.12  2007/09/17 11:37:55  jaume
56
* fixed multilayermarkersymbol unit symbol
57
*
58
* Revision 1.11  2007/08/09 07:57:51  jvidal
59
* javadoc
60
*
61
* Revision 1.10  2007/07/18 06:54:34  jaume
62
* continuing with cartographic support
63
*
64
* Revision 1.9  2007/07/03 10:58:29  jaume
65
* first refactor on CartographicSupport
66
*
67
* Revision 1.8  2007/06/29 13:07:01  jaume
68
* +PictureLineSymbol
69
*
70
* Revision 1.7  2007/06/07 06:50:40  jaume
71
* *** empty log message ***
72
*
73
* Revision 1.6  2007/05/08 08:47:40  jaume
74
* *** empty log message ***
75
*
76
* Revision 1.5  2007/04/17 07:01:53  bsanchez
77
* - Corregido fallo de Double.MIN_VALUE por Double.NEGATIVE_INFINITY comentado por Victor Olaya.
78
*
79
* Revision 1.4  2007/03/29 16:02:01  jaume
80
* *** empty log message ***
81
*
82
* Revision 1.3  2007/03/26 14:26:02  jaume
83
* implemented Print
84
*
85
* Revision 1.2  2007/03/09 11:20:57  jaume
86
* Advanced symbology (start committing)
87
*
88
* Revision 1.1.2.2  2007/02/21 07:34:08  jaume
89
* labeling starts working
90
*
91
* Revision 1.1.2.1  2007/02/16 10:54:12  jaume
92
* multilayer splitted to multilayerline, multilayermarker,and  multilayerfill
93
*
94
*
95
*/
96
package com.iver.cit.gvsig.fmap.core.symbols;
97

    
98
import java.awt.Color;
99
import java.awt.Graphics2D;
100
import java.awt.Rectangle;
101
import java.awt.geom.AffineTransform;
102
import java.awt.geom.Point2D;
103
import java.util.ArrayList;
104

    
105
import javax.print.attribute.PrintRequestAttributeSet;
106

    
107
import com.iver.cit.gvsig.fmap.ViewPort;
108
import com.iver.cit.gvsig.fmap.core.FPoint2D;
109
import com.iver.cit.gvsig.fmap.core.FShape;
110
import com.iver.cit.gvsig.fmap.core.IGeometry;
111
import com.iver.cit.gvsig.fmap.core.SymbologyFactory;
112
import com.iver.utiles.XMLEntity;
113
import com.iver.utiles.swing.threads.Cancellable;
114
/**
115
 * MultiLayerMarkerSymbol allows to group several marker symbols (xxxMarkerSymbol
116
 * implementing IMarkerSymbol)in one and treat it as an only one symbol.
117
 *
118
 * @author jaume dominguez faus - jaume.dominguez@iver.es
119
 */
120
public class MultiLayerMarkerSymbol extends AbstractMarkerSymbol implements IMarkerSymbol, IMultiLayerSymbol {
121
        private IMarkerSymbol[] layers = new IMarkerSymbol[0];
122
        private MultiLayerMarkerSymbol selectionSymbol;
123
        private Point2D offset = new Point2D.Double();
124
        private double rotation;
125
        private double markerSize;
126
        public Color getColor() {
127
                /*
128
                 * a multilayer symbol does not define any color, the color
129
                 * of each layer is defined by the layer itself
130
                 */
131
                return null;
132
        }
133

    
134
        public Point2D getOffset() {
135
                return offset;
136
        }
137

    
138
        public double getRotation() {
139
                return rotation;
140
        }
141

    
142

    
143
        public void setColor(Color color) {
144
                /*
145
                 * will apply the color to each layer
146
                 */
147
                for (int i = 0; i < layers.length; i++) {
148
                        layers[i].setColor(color);
149
                }
150
        }
151

    
152
        public void setOffset(Point2D offset) {
153
                this.offset = offset;
154
        }
155

    
156
        public void setRotation(double rotation) {
157
                this.rotation = rotation;
158
        }
159

    
160

    
161
        public double getSize() {
162
                double myMarkerSize = 0;
163

    
164
                for (int i = 0; i < getLayerCount(); i++) {
165
                        myMarkerSize = Math.max(myMarkerSize, ((IMarkerSymbol) getLayer(i)).getSize());
166
                }
167

    
168
                if (markerSize != myMarkerSize)
169
                        markerSize = myMarkerSize;
170
                return markerSize;
171
        }
172

    
173
        public void setSize(double size) {
174
                if (size > 0 && size != getSize()) {
175

    
176
                        double scale = size / getSize();
177
                        this.markerSize = size;
178
                        for (int i = 0; i < layers.length; i++) {
179
                                double lSize = layers[i].getSize();
180
                                layers[i].setSize(lSize*scale);
181
                        }
182
                }
183
        }
184

    
185
        public void draw(Graphics2D g, AffineTransform affineTransform, FShape shp, Cancellable cancel) {
186
                FPoint2D p = (FPoint2D) shp;
187
                g.rotate(rotation, p.getX(), p.getY());
188
                for (int i = 0; (cancel==null || !cancel.isCanceled()) && i < layers.length; i++) {
189
                        layers[i].draw(g, affineTransform, shp, cancel);
190
                }
191
                g.rotate(-rotation, p.getX(), p.getY());
192
        }
193

    
194
        public void drawInsideRectangle(Graphics2D g, AffineTransform scaleInstance, Rectangle r, PrintRequestAttributeSet properties) throws SymbolDrawingException {
195
                g.rotate(rotation, r.getCenterX(), r.getCenterY());
196
                for (int i = 0; i < layers.length; i++) {
197
                        layers[i].drawInsideRectangle(g, scaleInstance, r, properties);
198
                }
199
                g.rotate(-rotation, r.getCenterX(), r.getCenterY());
200
        }
201

    
202
        public int getOnePointRgb() {
203
                // will paint only the last layer pixel
204
                return layers[layers.length-1].getOnePointRgb();
205
        }
206

    
207
        public void getPixExtentPlus(FShape shp, float[] distances,
208
                        ViewPort viewPort, int dpi) {
209
                float[] myDistances = new float[] {0,0};
210
                distances[0] = 0;
211
                distances[1] = 0;
212
                for (int i = 0; i < layers.length; i++) {
213
                        layers[i].getPixExtentPlus(shp, myDistances, viewPort, dpi);
214
                        distances[0] = Math.max(myDistances[0], distances[0]);
215
                        distances[1] = Math.max(myDistances[1], distances[1]);
216
                }
217
        }
218

    
219
        public ISymbol getSymbolForSelection() {
220
                if (selectionSymbol == null) {
221
                        selectionSymbol = new MultiLayerMarkerSymbol();
222
                        selectionSymbol.setDescription(getDescription());
223
                        for (int i = 0; i < layers.length; i++) {
224
                                selectionSymbol.addLayer(layers[i].getSymbolForSelection());
225
                        }
226
                }
227
                return selectionSymbol;
228

    
229
        }
230

    
231

    
232
        public XMLEntity getXMLEntity() {
233
                XMLEntity xml = new XMLEntity();
234

    
235
                xml.putProperty("className", getClass().getName());
236
                xml.putProperty("isShapeVisible", isShapeVisible());
237
                xml.putProperty("desc", getDescription());
238
                xml.putProperty("size",        getSize());
239
                xml.putProperty("unit", getUnit());
240
                xml.putProperty("referenceSystem", getReferenceSystem());
241
                for (int i = 0; i < layers.length; i++) {
242
                        xml.addChild(layers[i].getXMLEntity());
243
                }
244
                return xml;
245
        }
246

    
247
        public boolean isSuitableFor(IGeometry geom) {
248
                return geom.getGeometryType() == FShape.POINT;
249
        }
250

    
251

    
252
        public String getClassName() {
253
                return getClass().getName();
254
        }
255

    
256
        public void setXMLEntity(XMLEntity xml) {
257
                setIsShapeVisible(xml.getBooleanProperty("isShapeVisible"));
258
                setDescription(xml.getStringProperty("desc"));
259
                setSize(xml.getDoubleProperty("size"));
260
                setUnit(xml.getIntProperty("unit"));
261
                setReferenceSystem(xml.getIntProperty("referenceSystem"));
262
                for (int i = 0; i < xml.getChildrenCount(); i++) {
263
                        addLayer((IMarkerSymbol) SymbologyFactory.createSymbolFromXML(xml.getChild(i), "layer" + i));
264
                }
265
        }
266

    
267

    
268
        public void print(Graphics2D g, AffineTransform at, FShape shape, PrintRequestAttributeSet properties) {
269
                for (int i = 0; i < layers.length; i++) {
270
                        layers[i].print(g, at, shape, properties);
271
                }
272

    
273
        }
274

    
275
        public void setLayer(int index, ISymbol layer) throws IndexOutOfBoundsException {
276
                layers[index] = (IMarkerSymbol) layer;
277
        }
278

    
279
        public void swapLayers(int index1, int index2) {
280
                ISymbol aux1 = getLayer(index1), aux2 = getLayer(index2);
281
                layers[index2] = (IMarkerSymbol) aux1;
282
                layers[index1] = (IMarkerSymbol) aux2;
283
        }
284

    
285
        public ISymbol getLayer(int layerIndex) {
286
//                try{
287
                        return layers[layerIndex];
288
//                } catch (Exception e) {
289
//                        return null;
290
//                }
291
        }
292

    
293
        public int getLayerCount() {
294
                return layers.length;
295
        }
296

    
297
        public void addLayer(ISymbol newLayer) {
298
                addLayer(newLayer, layers.length);
299
        }
300

    
301
        public void addLayer(ISymbol newLayer, int layerIndex) throws IndexOutOfBoundsException {
302
                if (newLayer == null )/*|| newLayer instanceof ILabelStyle)*/ return; // null or symbols that are styles are not allowed
303

    
304
                IMarkerSymbol newMarker = (IMarkerSymbol) newLayer;
305
                if (getLayerCount() == 0) {
306
                        // apply the new layer properties to this multilayer
307

    
308
                        setReferenceSystem(newMarker.getReferenceSystem());
309
                        markerSize = newMarker.getSize();
310
                        //setSize(newMarker.getSize());
311
                        setUnit(newMarker.getUnit());
312
                } else {
313
                        if (newMarker.getSize() > getSize()) {
314
                                //setSize(newMarker.getSize());
315
                                markerSize = newMarker.getSize();
316
                        }
317
                        newMarker.setReferenceSystem(getReferenceSystem());
318
                        newMarker.setUnit(getUnit());
319
                }
320
                selectionSymbol = null; /* forces the selection symbol to be re-created
321
                                                                  * next time it is required
322
                                                                  */
323
                if (layerIndex < 0 || layers.length < layerIndex)
324
                        throw new IndexOutOfBoundsException(layerIndex+" < 0 or "+layerIndex+" > "+layers.length);
325
                ArrayList<ISymbol> newLayers = new ArrayList<ISymbol>();
326
                for (int i = 0; i < layers.length; i++) {
327
                        newLayers.add(layers[i]);
328
                }
329
                try {
330
                        newLayers.add(layerIndex, newLayer);
331
                        layers = (IMarkerSymbol[]) newLayers.toArray(new IMarkerSymbol[0]);
332
                } catch (ArrayStoreException asEx) {
333
                        throw new ClassCastException(newLayer.getClassName()+" is not an IMarkerSymbol");
334
                }
335
        }
336

    
337
        public boolean removeLayer(ISymbol layer) {
338

    
339
                int capacity = 0;
340
                capacity = layers.length;
341
                ArrayList lst = new ArrayList(capacity);
342
                for (int i = 0; i < capacity; i++) {
343
                        lst.add(layers[i]);
344
                }
345
                boolean contains = lst.remove(layer);
346
                layers = (IMarkerSymbol[])lst.toArray(new IMarkerSymbol[0]);
347
                return contains;
348
        }
349

    
350
        @Override
351
        public void setUnit(int unit) {
352
                super.setUnit(unit);
353
                for (int i = 0; i < layers.length; i++) {
354
                        layers[i].setUnit(unit);
355
                }
356
        }
357

    
358
        @Override
359
        public void setReferenceSystem(int system) {
360
                super.setReferenceSystem(system);
361
                for (int i = 0; i < layers.length; i++) {
362
                        layers[i].setReferenceSystem(system);
363
                }
364
        }
365

    
366
        public void setAlpha(int alpha) {
367
                // first, get the biggest alpha in the layers and the index if such layer
368
                int maxAlpha = Integer.MIN_VALUE;
369
                int maxAlphaLayerIndex = 0;
370
                for (int i = 0; i < layers.length; i++) {
371
                        if (layers[i].getColor().getAlpha() > maxAlpha) {
372
                                maxAlpha = layers[i].getColor().getAlpha();
373
                                maxAlphaLayerIndex = i;
374
                        }
375
                }
376

    
377
                // now, max alpha takes the value of the desired alpha and the rest
378
                // will take a scaled (to biggest alpha) alpha value
379
                for (int i = 0; i < layers.length; i++) {
380
                        int r = layers[i].getColor().getRed();
381
                        int g = layers[i].getColor().getGreen();
382
                        int b = layers[i].getColor().getBlue();
383

    
384
                        if (i!=maxAlphaLayerIndex) {
385
                                double scaledAlpha = (double) layers[i].getColor().getAlpha()/maxAlpha;
386
                                int myAlpha = (int) (alpha*scaledAlpha);
387
                                if (myAlpha == 0)
388
                                        myAlpha = 1;
389
                                layers[i].setColor(new Color(r, g, b, myAlpha));
390
                        } else {
391
                                int myAlpha = (int) alpha;
392
                                if (myAlpha == 0)
393
                                        myAlpha = 1;
394
                                layers[i].setColor(new Color(r, g, b, myAlpha));
395
                        }
396
                }
397

    
398
        }
399
}