Statistics
| Revision:

svn-gvsig-desktop / trunk / libraries / libFMap / src / com / iver / cit / gvsig / fmap / core / symbols / MultiLayerMarkerSymbol.java @ 20024

History | View | Annotate | Download (10.9 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 20024 2008-04-08 07:50:29Z 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.hardcode.gdbms.driver.exceptions.ReadDriverException;
108
import com.iver.cit.gvsig.fmap.ViewPort;
109
import com.iver.cit.gvsig.fmap.core.FPoint2D;
110
import com.iver.cit.gvsig.fmap.core.FShape;
111
import com.iver.cit.gvsig.fmap.core.IGeometry;
112
import com.iver.cit.gvsig.fmap.core.SymbologyFactory;
113
import com.iver.utiles.XMLEntity;
114
import com.iver.utiles.swing.threads.Cancellable;
115
/**
116
 * MultiLayerMarkerSymbol allows to group several marker symbols (xxxMarkerSymbol
117
 * implementing IMarkerSymbol)in one and treat it as an only one symbol.
118
 *
119
 * @author jaume dominguez faus - jaume.dominguez@iver.es
120
 */
121
public class MultiLayerMarkerSymbol extends AbstractMarkerSymbol implements IMarkerSymbol, IMultiLayerSymbol {
122
        private IMarkerSymbol[] layers = new IMarkerSymbol[0];
123
        private MultiLayerMarkerSymbol selectionSymbol;
124
        private Point2D offset = new Point2D.Double();
125
        private double rotation;
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
        public void setSize(double size) {
161
                if (size > 0 && size != getSize()) {
162

    
163
                        double scale = size / getSize();
164
                        super.setSize(size);
165
                        for (int i = 0; i < layers.length; i++) {
166
                                double lSize = layers[i].getSize();
167
                                layers[i].setSize(lSize*scale);
168
                        }
169
                }
170
        }
171

    
172
        public void draw(Graphics2D g, AffineTransform affineTransform, FShape shp, Cancellable cancel) {
173
                FPoint2D p = (FPoint2D) shp;
174
                g.rotate(rotation, p.getX(), p.getY());
175
                for (int i = 0; (cancel==null || !cancel.isCanceled()) && i < layers.length; i++) {
176
                        layers[i].draw(g, affineTransform, shp, cancel);
177
                }
178
                g.rotate(-rotation, p.getX(), p.getY());
179
        }
180

    
181
        public void drawInsideRectangle(Graphics2D g, AffineTransform scaleInstance, Rectangle r) throws SymbolDrawingException {
182
                g.rotate(rotation, r.getCenterX(), r.getCenterY());
183
                for (int i = 0; i < layers.length; i++) {
184
                        layers[i].drawInsideRectangle(g, scaleInstance, r);
185
                }
186
                g.rotate(-rotation, r.getCenterX(), r.getCenterY());
187
        }
188

    
189
        public int getOnePointRgb() {
190
                // will paint only the last layer pixel
191
                return layers[layers.length-1].getOnePointRgb();
192
        }
193

    
194
        public void getPixExtentPlus(FShape shp, float[] distances,
195
                        ViewPort viewPort, int dpi) {
196
                float[] myDistances = new float[] {0,0};
197
                distances[0] = 0;
198
                distances[1] = 0;
199
                for (int i = 0; i < layers.length; i++) {
200
                        layers[i].getPixExtentPlus(shp, myDistances, viewPort, dpi);
201
                        distances[0] = Math.max(myDistances[0], distances[0]);
202
                        distances[1] = Math.max(myDistances[1], distances[1]);
203
                }
204
        }
205

    
206
        public ISymbol getSymbolForSelection() {
207
                if (selectionSymbol == null) {
208
                        selectionSymbol = new MultiLayerMarkerSymbol();
209
                        selectionSymbol.setDescription(getDescription());
210
                        for (int i = 0; i < layers.length; i++) {
211
                                selectionSymbol.addLayer(layers[i].getSymbolForSelection());
212
                        }
213
                }
214
                return selectionSymbol;
215

    
216
        }
217

    
218

    
219
        public int getSymbolType() {
220
                return FShape.POINT;
221
        }
222

    
223
        public XMLEntity getXMLEntity() {
224
                XMLEntity xml = new XMLEntity();
225

    
226
                xml.putProperty("className", getClass().getName());
227
                xml.putProperty("isShapeVisible", isShapeVisible());
228
                xml.putProperty("desc", getDescription());
229
                xml.putProperty("size",        getSize());
230
                xml.putProperty("unit", getUnit());
231
                xml.putProperty("referenceSystem", getReferenceSystem());
232
                for (int i = 0; i < layers.length; i++) {
233
                        xml.addChild(layers[i].getXMLEntity());
234
                }
235
                return xml;
236
        }
237

    
238
        public boolean isSuitableFor(IGeometry geom) {
239
                return geom.getGeometryType() == FShape.POINT;
240
        }
241

    
242

    
243
        public String getClassName() {
244
                return getClass().getName();
245
        }
246

    
247
        public void setXMLEntity(XMLEntity xml) {
248
                setIsShapeVisible(xml.getBooleanProperty("isShapeVisible"));
249
                setDescription(xml.getStringProperty("desc"));
250
                setSize(xml.getDoubleProperty("size"));
251
                setUnit(xml.getIntProperty("unit"));
252
                setReferenceSystem(xml.getIntProperty("referenceSystem"));
253
                for (int i = 0; i < xml.getChildrenCount(); i++) {
254
                        addLayer((IMarkerSymbol) SymbologyFactory.createSymbolFromXML(xml.getChild(i), "layer" + i));
255
                }
256
        }
257

    
258

    
259
        public void print(Graphics2D g, AffineTransform at, FShape shape, PrintRequestAttributeSet properties)
260
                        throws ReadDriverException {
261
                for (int i = 0; i < layers.length; i++) {
262
                        layers[i].print(g, at, shape, properties);
263
                }
264

    
265
        }
266

    
267
        public void setLayer(int index, ISymbol layer) throws IndexOutOfBoundsException {
268
                layers[index] = (IMarkerSymbol) layer;
269
        }
270

    
271
        public void swapLayers(int index1, int index2) {
272
                ISymbol aux1 = getLayer(index1), aux2 = getLayer(index2);
273
                layers[index2] = (IMarkerSymbol) aux1;
274
                layers[index1] = (IMarkerSymbol) aux2;
275
        }
276

    
277
        public ISymbol getLayer(int layerIndex) {
278
//                try{
279
                        return layers[layerIndex];
280
//                } catch (Exception e) {
281
//                        return null;
282
//                }
283
        }
284

    
285
        public int getLayerCount() {
286
                return layers.length;
287
        }
288

    
289
        public void addLayer(ISymbol newLayer) {
290
                addLayer(newLayer, layers.length);
291
        }
292

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

    
296
                IMarkerSymbol newMarker = (IMarkerSymbol) newLayer;
297
                if (getLayerCount() == 0) {
298
                        // apply the new layer properties to this multilayer
299

    
300
                        setReferenceSystem(newMarker.getReferenceSystem());
301
                        setSize(newMarker.getSize());
302
                        setUnit(newMarker.getUnit());
303
                } else {
304
                        if (newMarker.getSize() > getSize()) {
305
                                setSize(newMarker.getSize());
306
                        }
307
                        newMarker.setReferenceSystem(getReferenceSystem());
308
                        newMarker.setUnit(getUnit());
309
                }
310
                selectionSymbol = null; /* forces the selection symbol to be re-created
311
                                                                  * next time it is required
312
                                                                  */
313
                if (layerIndex < 0 || layers.length < layerIndex)
314
                        throw new IndexOutOfBoundsException(layerIndex+" < 0 or "+layerIndex+" > "+layers.length);
315
                ArrayList<ISymbol> newLayers = new ArrayList<ISymbol>();
316
                for (int i = 0; i < layers.length; i++) {
317
                        newLayers.add(layers[i]);
318
                }
319
                newLayers.add(layerIndex, newLayer);
320
                layers = (IMarkerSymbol[]) newLayers.toArray(new IMarkerSymbol[0]);
321
        }
322

    
323
        public boolean removeLayer(ISymbol layer) {
324

    
325
                int capacity = 0;
326
                capacity = layers.length;
327
                ArrayList lst = new ArrayList(capacity);
328
                for (int i = 0; i < capacity; i++) {
329
                        lst.add(layers[i]);
330
                }
331
                boolean contains = lst.remove(layer);
332
                layers = (IMarkerSymbol[])lst.toArray(new IMarkerSymbol[0]);
333
                return contains;
334
        }
335

    
336
        @Override
337
        public void setUnit(int unit) {
338
                super.setUnit(unit);
339
                for (int i = 0; i < layers.length; i++) {
340
                        layers[i].setUnit(unit);
341
                }
342
        }
343

    
344
        @Override
345
        public void setReferenceSystem(int system) {
346
                super.setReferenceSystem(system);
347
                for (int i = 0; i < layers.length; i++) {
348
                        layers[i].setReferenceSystem(system);
349
                }
350
        }
351

    
352
        public void setAlpha(int alpha) {
353
                // first, get the biggest alpha in the layers and the index if such layer
354
                int maxAlpha = Integer.MIN_VALUE;
355
                int maxAlphaLayerIndex = 0;
356
                for (int i = 0; i < layers.length; i++) {
357
                        if (layers[i].getColor().getAlpha() > maxAlpha) {
358
                                maxAlpha = layers[i].getColor().getAlpha();
359
                                maxAlphaLayerIndex = i;
360
                        }
361
                }
362

    
363
                // now, max alpha takes the value of the desired alpha and the rest
364
                // will take a scaled (to biggest alpha) alpha value
365
                for (int i = 0; i < layers.length; i++) {
366
                        int r = layers[i].getColor().getRed();
367
                        int g = layers[i].getColor().getGreen();
368
                        int b = layers[i].getColor().getBlue();
369

    
370
                        if (i!=maxAlphaLayerIndex) {
371
                                double scaledAlpha = (double) layers[i].getColor().getAlpha()/maxAlpha;
372
                                int myAlpha = (int) (alpha*scaledAlpha);
373
                                if (myAlpha == 0)
374
                                        myAlpha = 1;
375
                                layers[i].setColor(new Color(r, g, b, myAlpha));
376
                        } else {
377
                                int myAlpha = (int) alpha;
378
                                if (myAlpha == 0)
379
                                        myAlpha = 1;
380
                                layers[i].setColor(new Color(r, g, b, myAlpha));
381
                        }
382
                }
383

    
384
        }
385
}