Statistics
| Revision:

gvsig-vectorediting / org.gvsig.vectorediting / trunk / org.gvsig.vectorediting / org.gvsig.vectorediting.lib / org.gvsig.vectorediting.lib.prov / org.gvsig.vectorediting.lib.prov.regularpolygon / src / main / java / org / gvsig / vectorediting / lib / prov / regularpolygon / RegularPolygonEditingProvider.java @ 575

History | View | Annotate | Download (19.2 KB)

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

    
25
package org.gvsig.vectorediting.lib.prov.regularpolygon;
26

    
27
import java.util.ArrayList;
28
import java.util.HashMap;
29
import java.util.LinkedHashMap;
30
import java.util.List;
31
import java.util.Map;
32

    
33
import org.gvsig.fmap.dal.feature.FeatureStore;
34
import org.gvsig.fmap.geom.Geometry;
35
import org.gvsig.fmap.geom.Geometry.SUBTYPES;
36
import org.gvsig.fmap.geom.Geometry.TYPES;
37
import org.gvsig.fmap.geom.GeometryLocator;
38
import org.gvsig.fmap.geom.GeometryManager;
39
import org.gvsig.fmap.geom.aggregate.MultiCurve;
40
import org.gvsig.fmap.geom.primitive.Circumference;
41
import org.gvsig.fmap.geom.primitive.Curve;
42
import org.gvsig.fmap.geom.primitive.OrientablePrimitive;
43
import org.gvsig.fmap.geom.primitive.Point;
44
import org.gvsig.fmap.geom.primitive.Surface;
45
import org.gvsig.fmap.geom.type.GeometryType;
46
import org.gvsig.fmap.mapcontext.rendering.symbols.ISymbol;
47
import org.gvsig.tools.ToolsLocator;
48
import org.gvsig.tools.dynobject.DynObject;
49
import org.gvsig.tools.exception.BaseException;
50
import org.gvsig.tools.i18n.I18nManager;
51
import org.gvsig.tools.service.spi.ProviderServices;
52
import org.gvsig.vectorediting.lib.api.DrawingStatus;
53
import org.gvsig.vectorediting.lib.api.EditingServiceParameter;
54
import org.gvsig.vectorediting.lib.api.exceptions.DrawServiceException;
55
import org.gvsig.vectorediting.lib.api.exceptions.FinishServiceException;
56
import org.gvsig.vectorediting.lib.api.exceptions.InvalidEntryException;
57
import org.gvsig.vectorediting.lib.api.exceptions.StartServiceException;
58
import org.gvsig.vectorediting.lib.api.exceptions.StopServiceException;
59
import org.gvsig.vectorediting.lib.spi.AbstractEditingProvider;
60
import org.gvsig.vectorediting.lib.spi.DefaultDrawingStatus;
61
import org.gvsig.vectorediting.lib.spi.DefaultEditingServiceParameter;
62
import org.gvsig.vectorediting.lib.spi.EditingProvider;
63
import org.gvsig.vectorediting.lib.spi.EditingProviderFactory;
64
import org.gvsig.vectorediting.lib.spi.EditingProviderLocator;
65
import org.gvsig.vectorediting.lib.spi.EditingProviderManager;
66
import org.gvsig.vectorediting.lib.spi.EditingProviderServices;
67

    
68
/**
69
 *
70
 * @author llmarques
71
 *
72
 */
73
public class RegularPolygonEditingProvider extends AbstractEditingProvider
74
    implements EditingProvider {
75

    
76
    protected final Integer DEFAULT_VALUE_SIDES = 5;
77
    protected final String DEFAULT_VALUE_MODE = "key_inscribed";
78
    protected final String SIDES_KEY= "sides";
79
    protected final String MODE_KEY = "mode";
80

    
81
    protected EditingServiceParameter polygonSides;
82

    
83
    protected EditingServiceParameter center;
84

    
85
    protected EditingServiceParameter mode;
86

    
87
    protected EditingServiceParameter pointOfCircle;
88

    
89
    protected Map<EditingServiceParameter, Object> values;
90

    
91
    protected Map<String, String> defaultValues = null;
92

    
93
    protected Map<String, String> options;
94

    
95
    protected FeatureStore featureStore;
96

    
97
    public RegularPolygonEditingProvider(ProviderServices providerServices,
98
        DynObject parameters) {
99
        super(providerServices);
100
        this.featureStore =
101
            (FeatureStore) parameters
102
                .getDynValue(EditingProviderFactory.FEATURE_STORE_FIELD);
103

    
104
        Integer sideDefaultValue = Integer.valueOf((String)getDefaultValues().get(SIDES_KEY));
105
        if (sideDefaultValue == null){
106
            sideDefaultValue = DEFAULT_VALUE_SIDES;
107
        }
108

    
109
        polygonSides =
110
            new DefaultEditingServiceParameter(SIDES_KEY, sideDefaultValue,
111
                "sides_of_regular_polygon", EditingServiceParameter.TYPE.VALUE);
112

    
113
        I18nManager i18nManager = ToolsLocator.getI18nManager();
114
        options = new LinkedHashMap<String, String>();
115
        options.put(i18nManager.getTranslation("key_inscribed"), "inscribed");
116
        options.put(i18nManager.getTranslation("key_circumscribed"),
117
            "circumscribed");
118

    
119
        EditingProviderServices editingProviderServices =
120
            (EditingProviderServices) getProviderServices();
121

    
122
        String consoleMsg =
123
            editingProviderServices.makeConsoleMessage(null, options);
124

    
125
        String modeDefaultValue = (String)getDefaultValues().get(MODE_KEY);
126
        if (modeDefaultValue == null){
127
            modeDefaultValue = DEFAULT_VALUE_MODE;
128
        }
129

    
130
        mode =
131
            new DefaultEditingServiceParameter(MODE_KEY, consoleMsg, options,
132
                modeDefaultValue,
133
                EditingServiceParameter.TYPE.OPTION);
134

    
135
        center =
136
            new DefaultEditingServiceParameter("center",
137
                "center_of_regular_polygon",
138
                EditingServiceParameter.TYPE.POSITION);
139

    
140
        pointOfCircle =
141
            new DefaultEditingServiceParameter("point_of_circle",
142
                "point_of_circle", EditingServiceParameter.TYPE.POSITION);
143
    }
144

    
145
    private Map<String, String> getDefaultValues() {
146
        if (this.defaultValues == null){
147
            EditingProviderManager editingProviderManager =
148
                EditingProviderLocator.getProviderManager();
149
            this.defaultValues = editingProviderManager.getDefaultValues("regularpolygon");
150
        }
151
        return this.defaultValues;
152
    }
153

    
154
    public EditingServiceParameter next() {
155
        if (values.get(polygonSides) == null) {
156
            return polygonSides;
157
        } else if (values.get(mode) == null) {
158
            return mode;
159
        } else if (values.get(center) == null) {
160
            return center;
161
        } else if (values.get(pointOfCircle) == null) {
162
            return pointOfCircle;
163
        }
164
        return null;
165
    }
166

    
167
    public DrawingStatus getDrawingStatus(Point mousePosition)
168
        throws DrawServiceException {
169
        DefaultDrawingStatus drawingStatus = new DefaultDrawingStatus();
170
        EditingProviderManager editingProviderManager =
171
            EditingProviderLocator.getProviderManager();
172
        ISymbol lineSymbolEditing = editingProviderManager.getSymbol("line-symbol-editing");
173
        ISymbol auxiliaryLineSymbolEditing = editingProviderManager.getSymbol("auxiliary-line-symbol-editing");
174
        ISymbol auxiliaryPointSymbolEditing = editingProviderManager.getSymbol("auxiliary-point-symbol-editing");
175
        GeometryManager geometryManager = GeometryLocator.getGeometryManager();
176

    
177
        if (values != null) {
178

    
179
            Integer sidesValue = (Integer) values.get(polygonSides);
180
            String modeValue = (String) values.get(mode);
181
            Point centerValue = (Point) values.get(center);
182

    
183
            if (centerValue != null) {
184

    
185
                try {
186

    
187
                    EditingProviderServices editingProviderServices =
188
                        (EditingProviderServices) getProviderServices();
189
                    int subtype =
190
                        editingProviderServices.getSubType(featureStore);
191

    
192
                    if (modeValue == null) {
193
                        modeValue = (String)getDefaultValues().get(MODE_KEY);
194
                    }
195

    
196
                    Curve polygon =
197
                        calculateRegularPolygon(modeValue, centerValue,
198
                            mousePosition, sidesValue);
199
                    Curve line =
200
                        editingProviderServices.createLine(centerValue,
201
                            mousePosition, subtype);
202

    
203
                    double radius = mousePosition.distance(centerValue);
204
                    Circumference circumference = (Circumference) geometryManager.create(TYPES.CIRCUMFERENCE, SUBTYPES.GEOM2D);
205
                    circumference.setPoints(centerValue, radius);
206

    
207
                    drawingStatus.addStatus(circumference, auxiliaryLineSymbolEditing, "");
208
                    drawingStatus.addStatus(polygon, lineSymbolEditing, "");
209
                    drawingStatus.addStatus(centerValue, auxiliaryPointSymbolEditing, "");
210
                    drawingStatus.addStatus(line, auxiliaryLineSymbolEditing, "");
211

    
212
                } catch (BaseException e) {
213
                    throw new DrawServiceException(e);
214
                }
215
            }
216
        }
217
        return drawingStatus;
218
    }
219

    
220
    /**
221
     * Calculates polygon as of mode (inscribed or circumscribed), center of
222
     * polygon, point of circle and number of sides.
223
     *
224
     * @param modeValue
225
     *            of polygon (inscribed or circumscribed).
226
     * @param centerValue
227
     *            of polygon
228
     * @param pointOfCircleValue
229
     *            of circle that inscribe or circumscribe polygon
230
     * @param sidesValue
231
     *            number of sides.
232
     */
233
    protected Curve calculateRegularPolygon(String modeValue,
234
        Point centerValue, Point pointOfCircle, int sidesValue)
235
        throws BaseException {
236

    
237
        GeometryManager geometryManager = GeometryLocator.getGeometryManager();
238

    
239
        EditingProviderServices editingProviderServices =
240
            (EditingProviderServices) getProviderServices();
241
        int subtype = editingProviderServices.getSubType(featureStore);
242

    
243
        Curve polygon =
244
            (Curve) geometryManager.create(Geometry.TYPES.CURVE, subtype);
245

    
246
        double initAngle =
247
            editingProviderServices.getAngle(centerValue, pointOfCircle);
248
        double radiusValue = pointOfCircle.distance(centerValue);
249
        double theta = -(2 * Math.PI) / sidesValue;
250

    
251
        if (modeValue.equalsIgnoreCase("key_inscribed")) {
252

    
253
            for (int i = 0; i < sidesValue; ++i) {
254

    
255
                double x =
256
                    centerValue.getX()
257
                        + (radiusValue * Math.cos((theta * i) + initAngle));
258
                double y =
259
                    centerValue.getY()
260
                        + (radiusValue * Math.sin((theta * i) + initAngle));
261

    
262
                Point createPoint =
263
                    editingProviderServices.createPoint(x, y, subtype);
264

    
265
                polygon.addVertex(createPoint);
266
            }
267

    
268
        } else if (modeValue.equalsIgnoreCase("key_circumscribed")) {
269

    
270
            for (int i = 0; i < sidesValue; ++i) {
271

    
272
                double x =
273
                    centerValue.getX()
274
                        + (radiusValue * Math.cos((theta * i) + initAngle));
275
                double y =
276
                    centerValue.getY()
277
                        + (radiusValue * Math.sin((theta * i) + initAngle));
278

    
279
                Point apothemPoint =
280
                    editingProviderServices.createPoint(x, y, subtype);
281

    
282
                Double[] apothemParams =
283
                    editingProviderServices.getLineParams(centerValue,
284
                        apothemPoint);
285
                Point[] perpendicular =
286
                    editingProviderServices.getPerpendicular(apothemParams[0],
287
                        apothemParams[1], apothemPoint, subtype);
288

    
289
                double x1 =
290
                    centerValue.getX()
291
                        + (radiusValue * Math
292
                            .cos((theta * (i + 1)) + initAngle));
293
                double y1 =
294
                    centerValue.getY()
295
                        + (radiusValue * Math
296
                            .sin((theta * (i + 1)) + initAngle));
297

    
298
                Point nextApothemPoint =
299
                    editingProviderServices.createPoint(x1, y1, subtype);
300

    
301
                Double[] nextApothemParams =
302
                    editingProviderServices.getLineParams(centerValue,
303
                        nextApothemPoint);
304
                Point[] nextPerpendicular =
305
                    editingProviderServices.getPerpendicular(
306
                        nextApothemParams[0], nextApothemParams[1],
307
                        nextApothemPoint, subtype);
308

    
309
                Point vertex =
310
                    editingProviderServices.getIntersection(perpendicular,
311
                        nextPerpendicular, subtype);
312

    
313
                polygon.addVertex(vertex);
314
            }
315
        }
316
        polygon = (Curve) closeGeometry(polygon);
317
        return polygon;
318
    }
319

    
320
    /**
321
     * Close geometry if it is necessary.
322
     *
323
     * @param geometry
324
     *            to be closed.
325
     */
326
    protected Geometry closeGeometry(Geometry geometry) {
327

    
328
        if (!isClose(geometry) && (geometry != null)) {
329

    
330
            if (geometry instanceof Surface) {
331
                Surface surface = (Surface) geometry;
332
                Point firstp = surface.getVertex(0);
333
                firstp = (Point) firstp.cloneGeometry();
334
                surface.addVertex(firstp);
335
                return surface;
336
            } else if (geometry instanceof Curve) {
337
                Curve line = (Curve) geometry;
338
                Point firstp = line.getVertex(0);
339
                firstp = (Point) firstp.cloneGeometry();
340
                line.addVertex(firstp);
341
                return line;
342
            }
343
        }
344
        return geometry;
345
    }
346

    
347
    /**
348
     * Gets if the geometry received by parameter is closed or not.
349
     *
350
     * @param geometry
351
     * @return True if geometry is closed, otherwise false.
352
     */
353
    private boolean isClose(Geometry geometry) {
354

    
355
        if (geometry != null) {
356

    
357
            if (geometry instanceof OrientablePrimitive) {
358
                OrientablePrimitive orientablePrimitive =
359
                    (OrientablePrimitive) geometry;
360
                Point firstPoint = orientablePrimitive.getVertex(0);
361
                Point lastPoint =
362
                    orientablePrimitive.getVertex(orientablePrimitive
363
                        .getNumVertices() - 1);
364
                if (firstPoint.equals(lastPoint)) {
365
                    return true;
366
                }
367
            }
368
        }
369
        return false;
370
    }
371

    
372
    public void stop() throws StopServiceException {
373
        if (values != null) {
374
            values.clear();
375
        }
376
    }
377

    
378
    public List<EditingServiceParameter> getParameters() {
379
        List<EditingServiceParameter> parameters =
380
            new ArrayList<EditingServiceParameter>();
381
        parameters.add(polygonSides);
382
        parameters.add(mode);
383
        parameters.add(center);
384
        parameters.add(pointOfCircle);
385

    
386
        return parameters;
387
    }
388

    
389
    public void setValue(Object value) throws InvalidEntryException {
390
        EditingServiceParameter parameter = next();
391
        validateAndInsertValue(parameter, value);
392
    }
393

    
394
    private void validateAndInsertValue(EditingServiceParameter param,
395
        Object value) throws InvalidEntryException {
396

    
397
        if (param == polygonSides) {
398
            if (value instanceof Double) {
399

    
400
                Integer sides = ((Double) value).intValue();
401

    
402
                if (sides > 2) {
403
                    values.put(param, sides);
404
                    getDefaultValues().put(SIDES_KEY, sides.toString());
405
                    polygonSides.setDefaultValue(sides.toString());
406
                    return;
407
                } else {
408
                    throw new InvalidEntryException(null);
409
                }
410
            } else if (value instanceof Point) {
411

    
412
                values.put(param, Integer.valueOf((String)getDefaultValues().get(SIDES_KEY)));
413
                values.put(center, (Point) value);
414
                return;
415
            }
416
        } else if (param == mode) {
417

    
418
            if (value instanceof String) {
419

    
420
                I18nManager i18nManager = ToolsLocator.getI18nManager();
421
                String option = (String) value;
422

    
423
                if (option.equalsIgnoreCase(i18nManager
424
                    .getTranslation("key_inscribed"))) {
425

    
426
                    values.put(param, "key_inscribed");
427
                    getDefaultValues().put(MODE_KEY, "key_inscribed");
428
                    mode.setDefaultValue("key_inscribed");
429
                    return;
430

    
431
                } else if (option.equalsIgnoreCase(i18nManager
432
                    .getTranslation("key_circumscribed"))) {
433

    
434
                    values.put(param, "key_circumscribed");
435
                    getDefaultValues().put(MODE_KEY, "key_circumscribed");
436
                    mode.setDefaultValue("key_circumscribed");
437
                    return;
438

    
439
                } else {
440
                    throw new InvalidEntryException(null);
441
                }
442
            } else if (value instanceof Point) {
443

    
444
                values.put(param, getDefaultValues().get(MODE_KEY));
445
                if(values.get(center) != null){
446
                    values.put(pointOfCircle, (Point) value);
447
                } else {
448
                    values.put(center, (Point) value);
449
                }
450
                return;
451
            }
452

    
453
        } else if (param == center) {
454

    
455
            if (value instanceof Point) {
456
                values.put(param, value);
457
                return;
458
            }
459

    
460
        } else if (param == pointOfCircle) {
461
            if (value instanceof Point) {
462
                values.put(param, value);
463
                return;
464
            }
465

    
466
        }
467
        throw new InvalidEntryException(null);
468
    }
469

    
470
    public Geometry finish() throws FinishServiceException {
471

    
472
        if (values != null) {
473

    
474
            Integer sidesValue = (Integer) values.get(polygonSides);
475
            String modeValue = (String) values.get(mode);
476
            Point centerValue = (Point) values.get(center);
477
            Point pointOfCircleValue = (Point) values.get(pointOfCircle);
478

    
479
            try {
480
                Curve polygon =
481
                    calculateRegularPolygon(modeValue, centerValue,
482
                        pointOfCircleValue, sidesValue);
483

    
484
                EditingProviderServices editingProviderServices =
485
                    (EditingProviderServices) getProviderServices();
486
                GeometryType geomType =
487
                    editingProviderServices.getGeomType(featureStore);
488

    
489
                if (geomType.isTypeOf(Geometry.TYPES.MULTICURVE)) {
490
//                    return polygon.toLines();
491

    
492
                    MultiCurve multiCurve =
493
                        GeometryLocator.getGeometryManager().createMultiCurve(
494
                            geomType.getSubType());
495
                    multiCurve.addCurve(polygon);
496
                    return multiCurve;
497
                }
498

    
499
                return polygon;
500
            } catch (Exception e) {
501
                throw new FinishServiceException(e);
502
            }
503
        }
504
        return null;
505
    }
506

    
507
    public void finishAndStore() throws FinishServiceException {
508
        try {
509
            Geometry geometry = finish();
510
            if (geometry != null) {
511
                EditingProviderServices editingProviderServices =
512
                    (EditingProviderServices) getProviderServices();
513
                editingProviderServices.insertGeometryIntoFeatureStore(
514
                    geometry, featureStore);
515
            }
516
        } catch (Exception e) {
517
            throw new FinishServiceException(e);
518
        }
519

    
520
    }
521

    
522
    public void start() throws StartServiceException {
523
        values = new HashMap<EditingServiceParameter, Object>();
524
    }
525

    
526
    public String getName() {
527
        return RegularPolygonEditingProviderFactory.PROVIDER_NAME;
528
    }
529

    
530
}