Statistics
| Revision:

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

History | View | Annotate | Download (11.5 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 31297 2009-10-15 13:28:14Z fpenarrubia $
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
                                AffineTransform auxT = AffineTransform.getScaleInstance(scale, scale);
182
                                Point2D auxP = new Point2D.Double();
183
                                auxT.transform(layers[i].getOffset(), auxP);
184
                                layers[i].setOffset(auxP);
185
                        }
186
                }
187
        }
188

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

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

    
206
        public int getOnePointRgb() {
207
                // will paint only the last layer pixel
208
                return layers[layers.length-1].getOnePointRgb();
209
        }
210

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

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

    
233
        }
234

    
235

    
236
        public XMLEntity getXMLEntity() {
237
                XMLEntity xml = new XMLEntity();
238

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

    
251
        public boolean isSuitableFor(IGeometry geom) {
252
                return (geom.getGeometryType()%FShape.Z) == FShape.POINT;
253
        }
254

    
255

    
256
        public String getClassName() {
257
                return getClass().getName();
258
        }
259

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

    
271

    
272
        public void print(Graphics2D g, AffineTransform at, FShape shape, PrintRequestAttributeSet properties) {
273
                for (int i = 0; i < layers.length; i++) {
274
                        layers[i].print(g, at, shape, properties);
275
                }
276

    
277
        }
278

    
279
        public void setLayer(int index, ISymbol layer) throws IndexOutOfBoundsException {
280
                layers[index] = (IMarkerSymbol) layer;
281
        }
282

    
283
        public void swapLayers(int index1, int index2) {
284
                ISymbol aux1 = getLayer(index1), aux2 = getLayer(index2);
285
                layers[index2] = (IMarkerSymbol) aux1;
286
                layers[index1] = (IMarkerSymbol) aux2;
287
        }
288

    
289
        public ISymbol getLayer(int layerIndex) {
290
//                try{
291
                        return layers[layerIndex];
292
//                } catch (Exception e) {
293
//                        return null;
294
//                }
295
        }
296

    
297
        public int getLayerCount() {
298
                return layers.length;
299
        }
300

    
301
        public void addLayer(ISymbol newLayer) {
302
                addLayer(newLayer, layers.length);
303
        }
304

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

    
308
                IMarkerSymbol newMarker = (IMarkerSymbol) newLayer;
309
                if (getLayerCount() == 0) {
310
                        // apply the new layer properties to this multilayer
311

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

    
341
        public boolean removeLayer(ISymbol layer) {
342

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

    
354
        @Override
355
        public void setUnit(int unit) {
356
                super.setUnit(unit);
357
                for (int i = 0; i < layers.length; i++) {
358
                        layers[i].setUnit(unit);
359
                }
360
        }
361

    
362
        @Override
363
        public void setReferenceSystem(int system) {
364
                super.setReferenceSystem(system);
365
                for (int i = 0; i < layers.length; i++) {
366
                        layers[i].setReferenceSystem(system);
367
                }
368
        }
369

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

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

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

    
402
        }
403
}