Changeset 23189 in osm for applications/editors/josm/plugins/graphview
- Timestamp:
- 2010-09-15T18:53:09+02:00 (14 years ago)
- Location:
- applications/editors/josm/plugins/graphview
- Files:
-
- 83 edited
Legend:
- Unmodified
- Added
- Removed
-
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/access/AccessEvaluator.java
r16520 r23189 19 19 public interface AccessEvaluator<N, W> { 20 20 21 22 23 24 25 26 27 28 29 30 21 /** 22 * checks whether a way may be accessed in the given direction 23 * 24 * @param way object to be checked; != null 25 * @param segmentProperties map from road property types to their values for this way's 26 * segments; each value must be a valid value for its property type; 27 * != null 28 */ 29 public boolean wayUsable(W way, boolean forward, 30 Map<RoadPropertyType<?>, Object> roadPropertyValues); 31 31 32 33 34 35 36 37 38 39 40 32 /** 33 * checks whether a node may be accessed/passed 34 * 35 * @param node object to be checked; != null 36 * @param segmentProperties map from road property types to their values for SegmentNodes 37 * based on this node, each value must be a valid value for its 38 * property type; != null 39 */ 40 public boolean nodeUsable(N node, Map<RoadPropertyType<?>, Object> roadPropertyValues); 41 41 42 42 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/access/AccessParameters.java
r16520 r23189 11 11 public interface AccessParameters { 12 12 13 13 public String getAccessClass(); 14 14 15 16 17 18 19 15 /** 16 * returns true iff a road with a given access type can be used 17 * @param accessType access type to check usablitly for; != null 18 */ 19 public boolean getAccessTypeUsable(AccessType accessType); 20 20 21 22 23 24 25 26 21 /** 22 * returns all {@link VehiclePropertyType}s a value is avaliable for. 23 * The value can be accessed using {@link #getVehiclePropertyValue(VehiclePropertyType)} 24 * @return collection of property types; != null 25 */ 26 public Collection<VehiclePropertyType<?>> getAvailableVehicleProperties(); 27 27 28 29 30 31 32 33 34 35 36 37 28 /** 29 * returns the value for a vehicle property. 30 * 31 * @param <V> type of property value 32 * @param vehicleProperty property to get value for; != null 33 * @return value for vehicleProperty, null if no value is available. 34 * Guaranteed to be valid according to vehicleProperty's 35 * {@link VehiclePropertyType#isValidValue(Object)} method. 36 */ 37 public <V> V getVehiclePropertyValue(VehiclePropertyType<V> vehicleProperty); 38 38 39 39 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/access/AccessRuleset.java
r16520 r23189 12 12 public interface AccessRuleset { 13 13 14 15 16 17 18 19 20 21 22 23 14 /** 15 * for a mode of transport, returns all transport categories it is a subset of. 16 * For example, the returned collection for "motorcycle" might include "motor_vehicle" and "vehicle". 17 * 18 * @param transportMode mode of transport to get "supertypes" for; != null 19 * @return parameters superset categories, including the parameter itself, 20 * in the order of decreasing specificness 21 * empty if the parameter was no known mode of transport; != null 22 */ 23 public List<String> getAccessHierarchyAncestors(String transportMode); 24 24 25 26 27 28 29 30 25 /** 26 * returns all base tags. 27 * Base tags are tags that make an object "eligible" for access evaluation 28 * (commonly things like highway=* or barrier=*) 29 */ 30 public Collection<Tag> getBaseTags(); 31 31 32 33 34 35 36 32 /** 33 * returns ruleset-specific implications 34 * @return list of implications in the order they are expected to be applied; != null 35 */ 36 public List<Implication> getImplications(); 37 37 38 38 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/access/AccessRulesetReader.java
r16836 r23189 20 20 public class AccessRulesetReader { 21 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 22 public static class AccessRulesetSyntaxException extends IOException { 23 private static final long serialVersionUID = 1L; 24 public AccessRulesetSyntaxException(String message) { 25 super(message); 26 }; 27 public AccessRulesetSyntaxException(Throwable t) { 28 super(t.toString()); 29 } 30 } 31 32 /** private constructor to prevent instantiation */ 33 private AccessRulesetReader() { } 34 35 public static AccessRuleset readAccessRuleset(InputStream inputStream) 36 throws AccessRulesetSyntaxException, IOException { 37 38 RulesetHandler rulesetHandler = new RulesetHandler(); 39 40 try { 41 XMLReader reader = XMLReaderFactory.createXMLReader(); 42 InputSource input = new InputSource(inputStream); 43 reader.setContentHandler(rulesetHandler); 44 reader.setErrorHandler(null); 45 reader.parse(input); 46 } catch (SAXException e) { 47 throw new AccessRulesetSyntaxException(e); 48 } 49 50 return rulesetHandler.getAccessRuleset(); 51 } 52 53 private static class RulesetHandler extends DefaultHandler { 54 55 private static class AccessClass { 56 final String name; 57 final AccessClass parent; 58 public AccessClass(String name, AccessClass parent) { 59 this.name = name; 60 this.parent = parent; 61 } 62 List<String> getAncestorHierarchy() { 63 List<String> names; 64 if (parent == null) { 65 names = new LinkedList<String>(); 66 } else { 67 names = parent.getAncestorHierarchy(); 68 } 69 names.add(0, name); 70 return names; 71 } 72 } 73 74 private final Collection<AccessClass> accessClasses = new LinkedList<AccessClass>(); 75 private final Collection<Tag> baseTags = new LinkedList<Tag>(); 76 77 private static enum Section {NONE, CLASSES, BASETAGS, IMPLICATIONS}; 78 private Section currentSection = Section.NONE; 79 80 private AccessClass currentAccessClass = null; 81 82 private ImplicationXMLReader implicationReader = null; 83 private final List<Implication> implications = new LinkedList<Implication>(); 84 85 /** returns the AccessRuleset that was read */ 86 AccessRuleset getAccessRuleset() { 87 88 return new AccessRuleset() { 89 90 public List<String> getAccessHierarchyAncestors(String transportMode) { 91 for (AccessClass accessClass : accessClasses) { 92 if (accessClass.name.equals(transportMode)) { 93 return accessClass.getAncestorHierarchy(); 94 } 95 } 96 return new LinkedList<String>(); 97 } 98 99 public Collection<Tag> getBaseTags() { 100 return baseTags; 101 } 102 103 public List<Implication> getImplications() { 104 return implications; 105 } 106 107 }; 108 } 109 110 @Override 111 public void startElement(String uri, String localName, String name, 112 Attributes attributes) throws SAXException { 113 114 if (implicationReader != null) { 115 implicationReader.startElement(uri, localName, name, attributes); 116 return; 117 } 118 119 if ("classes".equals(name)) { 120 121 if (currentSection != Section.NONE) { 122 throw new SAXException("classes element below root child level"); 123 } 124 125 currentSection = Section.CLASSES; 126 127 } else if ("class".equals(name)) { 128 129 String className = attributes.getValue("name"); 130 131 if (currentSection != Section.CLASSES) { 132 throw new SAXException("class element (" + className + ") outside classes element"); 133 } else if (className == null) { 134 throw new SAXException("class element without name"); 135 } 136 137 AccessClass newAccessClass = new AccessClass(className, currentAccessClass); 138 139 accessClasses.add(newAccessClass); 140 141 currentAccessClass = newAccessClass; 142 143 } else if ("basetags".equals(name)) { 144 145 if (currentSection != Section.NONE) { 146 throw new SAXException("classes element below root child level"); 147 } 148 149 currentSection = Section.BASETAGS; 150 151 } else if ("tag".equals(name)) { 152 153 if (currentSection == Section.BASETAGS) { 154 baseTags.add(readTag(attributes)); 155 } else { 156 throw new SAXException("tag element outside basetag and implication elements"); 157 } 158 159 } else if ("implications".equals(name)) { 160 161 if (currentSection != Section.NONE) { 162 throw new SAXException("implications element below root child level"); 163 } 164 165 implicationReader = new ImplicationXMLReader(); 166 currentSection = Section.IMPLICATIONS; 167 168 } 169 170 } 171 172 private static Tag readTag(Attributes attributes) throws SAXException { 173 174 String key = attributes.getValue("k"); 175 String value = attributes.getValue("v"); 176 177 if (key == null) { 178 throw new SAXException("tag without key"); 179 } else if (value == null) { 180 throw new SAXException("tag without value (key is " + key + ")"); 181 } 182 183 return new Tag(key, value); 184 } 185 186 @Override 187 public void endElement(String uri, String localName, String name) 188 throws SAXException { 189 190 if (implicationReader != null && !"implications".equals(name)) { 191 implicationReader.endElement(uri, localName, name); 192 } 193 194 if ("classes".equals(name)) { 195 196 if (currentSection != Section.CLASSES) { 197 throw new SAXException("closed classes while it wasn't open"); 198 } else if (currentAccessClass != null) { 199 throw new SAXException("closed classes element before all class elements were closed"); 200 } 201 202 currentSection = Section.NONE; 203 204 } else if ("class".equals(name)) { 205 206 if (currentAccessClass == null) { 207 throw new SAXException("closed class element while none was open"); 208 } 209 210 currentAccessClass = currentAccessClass.parent; 211 212 } else if ("basetags".equals(name)) { 213 214 if (currentSection != Section.BASETAGS) { 215 throw new SAXException("closed basetags while it wasn't open"); 216 } 217 218 currentSection = Section.NONE; 219 220 } else if ("implications".equals(name)) { 221 222 if (currentSection != Section.IMPLICATIONS) { 223 throw new SAXException("closed implications while it wasn't open"); 224 } 225 226 implications.addAll(implicationReader.getImplications()); 227 implicationReader = null; 228 currentSection = Section.NONE; 229 230 } 231 232 } 233 234 }; 235 235 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/access/AccessType.java
r16520 r23189 3 3 public enum AccessType { 4 4 5 6 7 8 9 10 11 12 13 14 5 YES("yes"), 6 PERMISSIVE("permissive"), 7 DESIGNATED("designated"), 8 DESTINATION("destination"), 9 AGRICULTURAL("agricultural"), 10 FORESTRY("forestry"), 11 DELIVERY("delivery"), 12 PRIVATE("private"), 13 NO("no"), 14 UNDEFINED(); 15 15 16 17 18 19 16 private String[] valueStrings; 17 private AccessType(String... valueStrings) { 18 this.valueStrings = valueStrings; 19 } 20 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 21 /** 22 * returns the AccessType that fits for a tag's value 23 * 24 * @param valueString a tag's value; != null 25 * @return AccessType for the value; != null, will be UNDEFINED for unknown values 26 */ 27 public static AccessType getAccessType(String valueString) { 28 for (AccessType accessType : AccessType.values()) { 29 for (String typeValueString : accessType.valueStrings) { 30 if (typeValueString.equals(valueString)) { 31 return accessType; 32 } 33 } 34 } 35 return UNDEFINED; 36 } 37 37 38 38 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/access/Implication.java
r16520 r23189 15 15 public final class Implication { 16 16 17 18 17 private final TagCondition condition; 18 private final Collection<Tag> impliedTags; 19 19 20 21 22 23 20 public Implication(TagCondition condition, Collection<Tag> impliedTags) { 21 this.condition = condition; 22 this.impliedTags = impliedTags; 23 } 24 24 25 26 27 28 29 30 31 32 25 /** 26 * applies this implication to a tag group. 27 * The resulting tag group will contain all tags from the original group 28 * and all implied tags with a key that didn't occur in the original group. 29 * 30 * @param tags tag group to apply implications to; != null 31 */ 32 public TagGroup apply(TagGroup tags) { 33 33 34 34 if (condition.matches(tags)) { 35 35 36 36 Map<String, String> newTagMap = new HashMap<String, String>(); 37 37 38 39 40 38 for (Tag tag : tags) { 39 newTagMap.put(tag.key, tag.value); 40 } 41 41 42 43 44 45 46 42 for (Tag impliedTag : impliedTags) { 43 if (!newTagMap.containsKey(impliedTag.key)) { 44 newTagMap.put(impliedTag.key, impliedTag.value); 45 } 46 } 47 47 48 48 return new MapBasedTagGroup(newTagMap); 49 49 50 51 52 50 } else { 51 return tags; 52 } 53 53 54 54 } 55 55 56 57 58 59 56 @Override 57 public String toString() { 58 return condition.toString() + " => " + impliedTags.toString(); 59 } 60 60 61 61 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/access/ImplicationXMLReader.java
r16520 r23189 17 17 public class ImplicationXMLReader { 18 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 19 private final List<Implication> implications = new LinkedList<Implication>(); 20 21 private static enum State {BEFORE_IMPLICATION, BEFORE_CONDITION, CONDITION, BEFORE_IMPLIES, IMPLIES, AFTER_IMPLIES}; 22 private State state = State.BEFORE_IMPLICATION; 23 24 private ConditionReader currentConditionReader; 25 private TagCondition currentCondition; 26 private Collection<Tag> currentImpliedTags; 27 28 boolean tagOpen = false; 29 30 public void startElement(String uri, String localName, String name, Attributes attributes) 31 throws SAXException { 32 33 switch (state) { 34 35 case BEFORE_IMPLICATION: 36 37 if ("implication".equals(name)) { 38 state = State.BEFORE_CONDITION; 39 return; 40 } 41 break; 42 43 case BEFORE_CONDITION: 44 45 if ("condition".equals(name)) { 46 currentConditionReader = new ConditionReader(); 47 state = State.CONDITION; 48 return; 49 } 50 break; 51 52 case CONDITION: 53 currentConditionReader.startElement(uri, localName, name, attributes); 54 return; 55 56 case BEFORE_IMPLIES: 57 58 if ("implies".equals(name)) { 59 currentImpliedTags = new LinkedList<Tag>(); 60 state = State.IMPLIES; 61 return; 62 } 63 break; 64 65 case IMPLIES: 66 67 if ("tag".equals(name)) { 68 if (tagOpen) { 69 throw new SAXException("tag element inside other tag element"); 70 } 71 currentImpliedTags.add(readTag(attributes)); 72 tagOpen = true; 73 return; 74 } 75 break; 76 77 } 78 79 //all vaild paths end with return; reaching this indicates an invalid tag 80 throw new SAXException("invalid opening xml tag <" + name + "> in state " + state); 81 82 } 83 84 public void endElement(String uri, String localName, String name) 85 throws SAXException { 86 87 switch (state) { 88 89 case CONDITION: 90 91 if (name.equals("condition")) { 92 if (!currentConditionReader.isFinished()) { 93 throw new SAXException("condition isn't finished at </condition> tag"); 94 } else { 95 currentCondition = currentConditionReader.getCondition(); 96 currentConditionReader = null; 97 state = State.BEFORE_IMPLIES; 98 return; 99 } 100 } else { 101 currentConditionReader.endElement(uri, localName, name); 102 return; 103 } 104 105 case IMPLIES: 106 107 if (name.equals("implies")) { 108 state = State.AFTER_IMPLIES; 109 return; 110 } else if (name.equals("tag")) { 111 if (!tagOpen) { 112 throw new SAXException("closing tag element that was not open"); 113 } 114 tagOpen = false; 115 return; 116 } 117 break; 118 119 case AFTER_IMPLIES: 120 121 if (name.equals("implication")) { 122 implications.add(new Implication(currentCondition, currentImpliedTags)); 123 currentCondition = null; 124 currentImpliedTags = null; 125 state = State.BEFORE_IMPLICATION; 126 return; 127 } 128 break; 129 130 } 131 132 //all vaild paths end with return; reaching this indicates an invalid tag 133 throw new SAXException("invalid closing xml tag </" + name + "> in state " + state); 134 135 } 136 137 public List<Implication> getImplications() throws SAXException { 138 139 if (state != State.BEFORE_IMPLICATION) { 140 throw new SAXException("some tags not been closed; now in state " + state); 141 } else { 142 return new ArrayList<Implication>(implications); 143 } 144 } 145 146 private static Tag readTag(Attributes attributes) throws SAXException { 147 148 String key = attributes.getValue("k"); 149 String value = attributes.getValue("v"); 150 151 if (key == null) { 152 throw new SAXException("tag without key"); 153 } else if (value == null) { 154 throw new SAXException("tag without value (key is " + key + ")"); 155 } 156 157 return new Tag(key, value); 158 } 159 160 private static String readKey(Attributes attributes) throws SAXException { 161 162 String key = attributes.getValue("k"); 163 164 if (key == null) { 165 throw new SAXException("key element without attribute k"); 166 } 167 168 return key; 169 } 170 171 /** 172 * class to be used for reading tag condition sections of xml files 173 */ 174 private static class ConditionReader { 175 176 String openingName; 177 TagCondition condition; 178 boolean finished; 179 180 private final List<ConditionReader> childReaders = new LinkedList<ConditionReader>(); 181 private ConditionReader currentChildReader = null; 182 183 public void startElement(String uri, String localName, String name, Attributes attributes) 184 throws SAXException { 185 186 if (finished) { 187 throw new SAXException("condition is already finished at <" + name + ">"); 188 } 189 190 if (currentChildReader != null) { 191 currentChildReader.startElement(uri, localName, name, attributes); 192 return; 193 } 194 195 //first tag is start tag of this condition 196 if (openingName == null) { 197 198 openingName = name; 199 200 if ("tag".equals(name)) { 201 condition = TagConditionLogic.tag(readTag(attributes)); 202 } else if ("key".equals(name)) { 203 condition = TagConditionLogic.key(readKey(attributes)); 204 } else if (!("or".equals(name)) && !("and".equals(name)) && !("not".equals(name))) { 205 throw new SAXException("unknown tag for condition: " + name); 206 } 207 208 //all tags after the first are start tags of child conditions 209 } else { 210 211 if ("tag".equals(openingName) || "key".equals(openingName)) { 212 throw new SAXException("element must not have childs: " + openingName); 213 } 214 215 currentChildReader = new ConditionReader(); 216 currentChildReader.startElement(uri, localName, name, attributes); 217 218 } 219 220 } 221 222 public void endElement(String uri, String localName, String name) 223 throws SAXException { 224 225 if (finished) { 226 throw new SAXException("condition is already finished at </" + name + ">"); 227 } 228 229 /* if active child reader exists, pass parameter to it. */ 230 if (currentChildReader != null) { 231 232 currentChildReader.endElement(uri, localName, name); 233 234 if (currentChildReader.isFinished()) { 235 childReaders.add(currentChildReader); 236 currentChildReader = null; 237 } 238 239 } else { 240 241 if (openingName.equals(name)) { 242 243 List<TagCondition> childConditions = new ArrayList<TagCondition>(); 244 for (ConditionReader childReader : childReaders) { 245 childConditions.add(childReader.getCondition()); 246 } 247 248 if ("and".equals(openingName)) { 249 if (childConditions.size() > 0) { 250 condition = TagConditionLogic.and(childConditions); 251 } else { 252 throw new SAXException("<and> needs at least one child"); 253 } 254 } else if ("or".equals(openingName)) { 255 if (childConditions.size() > 0) { 256 condition = TagConditionLogic.or(childConditions); 257 } else { 258 throw new SAXException("<or> needs at least one child"); 259 } 260 } else if ("not".equals(openingName)) { 261 if (childConditions.size() == 1) { 262 condition = TagConditionLogic.not(childConditions.get(0)); 263 } else { 264 throw new SAXException("<not> needs at least one child"); 265 } 266 } 267 268 finished = true; 269 270 } else { 271 throw new SAXException("wrong closing tag " + name + 272 " (</" + openingName + "> expected"); 273 } 274 275 } 276 277 } 278 279 public boolean isFinished() { 280 return finished; 281 } 282 283 public TagCondition getCondition() { 284 if (!finished) { 285 throw new IllegalStateException("condition " + openingName + " not yet finished"); 286 } else { 287 assert condition != null; 288 return condition; 289 } 290 } 291 292 } 293 293 294 294 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/access/RulesetAccessEvaluator.java
r19216 r23189 18 18 public class RulesetAccessEvaluator<N, W, R, M> implements AccessEvaluator<N, W> { 19 19 20 21 22 20 private final DataSource<N, W, R, M> dataSource; 21 private final AccessRuleset ruleset; 22 private final AccessParameters parameters; 23 23 24 25 26 27 28 29 30 31 24 /** 25 * @param dataSource object that allows access to data objects and tags/members; != null 26 * @param ruleset ruleset that is used for evaluation; != null 27 * @param parameters parameters object that describes the vehicle 28 * and situation to evaluate access for; != null 29 */ 30 public RulesetAccessEvaluator(DataSource<N, W, R, M> dataSource, AccessRuleset ruleset, AccessParameters parameters) { 31 assert dataSource != null && ruleset != null && parameters != null; 32 32 33 34 35 33 this.dataSource = dataSource; 34 this.ruleset = ruleset; 35 this.parameters = parameters; 36 36 37 37 } 38 38 39 40 39 public boolean wayUsable(W way, boolean forward, 40 Map<RoadPropertyType<?>, Object> segmentPropertyValues) { 41 41 42 42 TagGroup wayTags = dataSource.getTagsW(way); 43 43 44 45 46 47 44 TagGroup wayTagsWithImplications = new MapBasedTagGroup(wayTags); 45 for (Implication implication : ruleset.getImplications()) { 46 wayTagsWithImplications = implication.apply(wayTagsWithImplications); 47 } 48 48 49 49 /* check base tagging */ 50 50 51 52 53 54 55 56 57 51 boolean usableWay = false; 52 for (Tag tag : ruleset.getBaseTags()) { 53 if (wayTags.contains(tag)) { 54 usableWay = true; 55 break; 56 } 57 } 58 58 59 60 61 59 if (!usableWay) { 60 return false; 61 } 62 62 63 63 /* evaluate one-way tagging */ 64 64 65 65 String onewayValue = wayTagsWithImplications.getValue("oneway"); 66 66 67 68 69 70 67 if (forward && "-1".equals(onewayValue) 68 && !"foot".equals(parameters.getAccessClass())) { 69 return false; 70 } 71 71 72 73 74 75 76 72 if (!forward 73 && ("1".equals(onewayValue) || "yes".equals(onewayValue) || "true".equals(onewayValue)) 74 && !"foot".equals(parameters.getAccessClass())) { 75 return false; 76 } 77 77 78 78 /* evaluate properties and access tagging */ 79 79 80 81 80 return objectUsable(segmentPropertyValues, wayTags); 81 } 82 82 83 83 public boolean nodeUsable(N node, Map<RoadPropertyType<?>,Object> roadPropertyValues) { 84 84 85 85 TagGroup nodeTags = dataSource.getTagsN(node); 86 86 87 88 87 return objectUsable(roadPropertyValues, nodeTags); 88 }; 89 89 90 91 90 private boolean objectUsable(Map<RoadPropertyType<?>, Object> roadPropertyValues, 91 TagGroup tags) { 92 92 93 93 /* evaluate road properties */ 94 94 95 96 97 98 99 95 for (RoadPropertyType<?> property : roadPropertyValues.keySet()) { 96 if (!property.isUsable(roadPropertyValues.get(property), parameters)) { 97 return false; 98 } 99 } 100 100 101 101 /* evaluate access type */ 102 102 103 103 AccessType accessType = UNDEFINED; 104 104 105 105 if (tags.size() > 0) { 106 106 107 108 107 Map<String, AccessType> accessTypePerClass = 108 createAccessTypePerClassMap(tags, ruleset.getAccessHierarchyAncestors(parameters.getAccessClass())); 109 109 110 111 112 113 110 for (String accessClass : ruleset.getAccessHierarchyAncestors(parameters.getAccessClass())) { 111 accessType = accessTypePerClass.get(accessClass); 112 if (accessType != UNDEFINED) { break; } 113 } 114 114 115 115 } 116 116 117 118 117 return parameters.getAccessTypeUsable(accessType); 118 } 119 119 120 121 120 private Map<String, AccessType> createAccessTypePerClassMap( 121 TagGroup wayTags, Collection<String> accessClasses) { 122 122 123 124 125 126 123 /* 124 * create map and fill with UNDEFINED values 125 * (this also allows to use keySet instead of accessClasses later) 126 */ 127 127 128 128 Map<String, AccessType> accessTypePerClass = new HashMap<String, AccessType>(); 129 129 130 131 132 130 for (String accessClass : accessClasses) { 131 accessTypePerClass.put(accessClass, AccessType.UNDEFINED); 132 } 133 133 134 134 /* evaluate implied tagging of base tag */ 135 135 136 137 138 139 140 141 142 136 Tag baseTag = null; 137 for (Tag tag : wayTags) { 138 if (ruleset.getBaseTags().contains(tag)) { 139 baseTag = tag; 140 break; 141 } 142 } 143 143 144 144 if (baseTag != null) { 145 145 146 147 148 149 146 TagGroup tagsWithBaseImplications = new MapBasedTagGroup(baseTag); 147 for (Implication implication : ruleset.getImplications()) { 148 tagsWithBaseImplications = implication.apply(tagsWithBaseImplications); 149 } 150 150 151 151 setAccessTypesFromTags(accessTypePerClass, tagsWithBaseImplications); 152 152 153 153 } 154 154 155 155 /* evaluate implied tagging of other tags */ 156 156 157 158 159 160 161 162 157 Map<String, String> tagMap = new HashMap<String, String>(); 158 for (Tag tag : wayTags) { 159 if (!tag.equals(baseTag)) { 160 tagMap.put(tag.key, tag.value); 161 } 162 } 163 163 164 165 166 167 164 TagGroup tagsWithOtherImplications = new MapBasedTagGroup(tagMap); 165 for (Implication implication : ruleset.getImplications()) { 166 tagsWithOtherImplications = implication.apply(tagsWithOtherImplications); 167 } 168 168 169 169 setAccessTypesFromTags(accessTypePerClass, tagsWithOtherImplications); 170 170 171 171 /* evaluate explicit access tagging */ 172 172 173 174 175 176 177 178 179 173 for (String key : ruleset.getAccessHierarchyAncestors(parameters.getAccessClass())) { 174 String value = wayTags.getValue(key); 175 if (value != null) { 176 AccessType accessType = AccessType.getAccessType(value); 177 accessTypePerClass.put(key, accessType); 178 } 179 } 180 180 181 182 181 return accessTypePerClass; 182 } 183 183 184 185 186 187 188 189 190 191 192 193 194 195 196 184 /** 185 * adds all access information from a collection of tags to a [access class -> access type] map. 186 * Existing entries will be replaced. 187 */ 188 private void setAccessTypesFromTags(Map<String, AccessType> accessTypePerClass, TagGroup tags) { 189 for (String accessClass : accessTypePerClass.keySet()) { 190 String value = tags.getValue(accessClass); 191 if (value != null) { 192 AccessType accessType = AccessType.getAccessType(value); 193 accessTypePerClass.put(accessClass, accessType); 194 } 195 } 196 } 197 197 198 198 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/data/DataSource.java
r19216 r23189 9 9 */ 10 10 public interface DataSource<N, W, R, M> { 11 12 /** returns all nodes */13 public Iterable<N> getNodes();14 11 15 ways */16 W> getWays();12 /** returns all nodes */ 13 public Iterable<N> getNodes(); 17 14 18 relations */19 R> getRelations();15 /** returns all ways */ 16 public Iterable<W> getWays(); 20 17 21 node's latitude*/22 double getLat(N node);18 /** returns all relations */ 19 public Iterable<R> getRelations(); 23 20 24 ongitude */25 on(N node);21 /** returns a node's latitude */ 22 public double getLat(N node); 26 23 27 way's nodes*/28 Iterable<N> getNodes(W way);24 /** returns a node's longitude */ 25 public double getLon(N node); 29 26 30 relation's members */31 M> getMembers(R relation);27 /** returns a way's nodes */ 28 public Iterable<N> getNodes(W way); 32 29 33 node's tags */34 TagGroup getTagsN(N node);30 /** returns a relation's members */ 31 public Iterable<M> getMembers(R relation); 35 32 36 way's tags */37 W(W way);33 /** returns a node's tags */ 34 public TagGroup getTagsN(N node); 38 35 39 relation's tags */40 R(R relation);36 /** returns a way's tags */ 37 public TagGroup getTagsW(W way); 41 38 42 member's role*/43 String getRole(M member);39 /** returns a relation's tags */ 40 public TagGroup getTagsR(R relation); 44 41 45 /** returns a relation member's member object */ 46 public Object getMember(M member); 47 48 /** returns whether a relation member is a node */ 49 public boolean isNMember(M member); 50 51 /** returns whether a relation member is a way */ 52 public boolean isWMember(M member); 53 54 /** returns whether a relation member is a relation */ 55 public boolean isRMember(M member); 56 57 /** 58 * adds an observer. 59 * Does nothing if the parameter is already an observer of this DataSource. 60 * 61 * @param observer observer object, != null 62 */ 63 public void addObserver(DataSourceObserver observer); 42 /** returns a relation member's role */ 43 public String getRole(M member); 64 44 65 /** 66 * deletes an observer that has been added using {@link #addObserver(DataSourceObserver)}. 67 * Does nothing if the parameter isn't currently an observer of this DataSource. 68 * 69 * @param observer observer object, != null 70 */ 71 public void deleteObserver(DataSourceObserver observer); 45 /** returns a relation member's member object */ 46 public Object getMember(M member); 47 48 /** returns whether a relation member is a node */ 49 public boolean isNMember(M member); 50 51 /** returns whether a relation member is a way */ 52 public boolean isWMember(M member); 53 54 /** returns whether a relation member is a relation */ 55 public boolean isRMember(M member); 56 57 /** 58 * adds an observer. 59 * Does nothing if the parameter is already an observer of this DataSource. 60 * 61 * @param observer observer object, != null 62 */ 63 public void addObserver(DataSourceObserver observer); 64 65 /** 66 * deletes an observer that has been added using {@link #addObserver(DataSourceObserver)}. 67 * Does nothing if the parameter isn't currently an observer of this DataSource. 68 * 69 * @param observer observer object, != null 70 */ 71 public void deleteObserver(DataSourceObserver observer); 72 72 73 73 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/data/DataSourceObserver.java
r19216 r23189 8 8 public interface DataSourceObserver { 9 9 10 11 12 13 14 10 /** 11 * informs this observer about changes in an observed data source 12 * @param dataSource observed data source that has changed; != null 13 */ 14 public void update(DataSource<?, ?, ?, ?> dataSource); 15 15 16 16 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/data/MapBasedTagGroup.java
r16520 r23189 13 13 public class MapBasedTagGroup implements TagGroup { 14 14 15 15 private final Map<String, String> tagMap; 16 16 17 18 19 20 21 22 23 24 17 /** 18 * @param tagMap map from keys to values; != null; 19 * must not be modified after being used as parameter 20 */ 21 public MapBasedTagGroup(Map<String, String> tagMap) { 22 if (tagMap == null) { 23 throw new IllegalArgumentException(); 24 } 25 25 26 27 26 this.tagMap = tagMap; 27 } 28 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 29 /** 30 * @param tags tags to add to the group; != null, each != null 31 */ 32 public MapBasedTagGroup(Iterable<Tag> tags) { 33 if (tags == null) { 34 throw new IllegalArgumentException(); 35 } 36 this.tagMap = new HashMap<String, String>(); 37 for (Tag tag : tags) { 38 if (tag == null) { 39 throw new IllegalArgumentException(); 40 } else { 41 this.tagMap.put(tag.key, tag.value); 42 } 43 } 44 } 45 45 46 47 48 49 50 51 52 53 54 55 56 57 58 46 /** 47 * @param tags tags to add to the group; each != null 48 */ 49 public MapBasedTagGroup(Tag... tags) { 50 this.tagMap = new HashMap<String, String>(tags.length); 51 for (Tag tag : tags) { 52 if (tag == null) { 53 throw new IllegalArgumentException(); 54 } else { 55 this.tagMap.put(tag.key, tag.value); 56 } 57 } 58 } 59 59 60 61 62 63 60 public String getValue(String key) { 61 assert key != null; 62 return tagMap.get(key); 63 } 64 64 65 66 67 68 65 public boolean containsKey(String key) { 66 assert key != null; 67 return tagMap.containsKey(key); 68 } 69 69 70 71 72 73 70 public boolean containsValue(String value) { 71 assert value != null; 72 return tagMap.containsValue(value); 73 } 74 74 75 76 77 78 75 public boolean contains(Tag tag) { 76 assert tag != null; 77 return tag.value.equals(tagMap.get(tag.key)); 78 } 79 79 80 81 82 80 public int size() { 81 return tagMap.size(); 82 } 83 83 84 85 86 87 88 84 /** 85 * returns an Iterator providing access to all Tags. 86 * The Iterator does not support the {@link Iterator#remove()} method. 87 */ 88 public Iterator<Tag> iterator() { 89 89 90 90 Collection<Tag> tagCollection = new LinkedList<Tag>(); 91 91 92 93 94 92 for (String key : tagMap.keySet()) { 93 tagCollection.add(new Tag(key, tagMap.get(key))); 94 } 95 95 96 96 return Collections.unmodifiableCollection(tagCollection).iterator(); 97 97 98 98 } 99 99 100 101 102 103 100 @Override 101 public String toString() { 102 return tagMap.toString(); 103 } 104 104 105 105 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/data/Tag.java
r16520 r23189 6 6 public class Tag { 7 7 8 9 8 /** key of the tag; != null */ 9 public final String key; 10 10 11 12 11 /** value of the tag; != null */ 12 public final String value; 13 13 14 15 16 17 18 14 public Tag(String key, String value) { 15 assert key != null && value != null; 16 this.key = key; 17 this.value = value; 18 } 19 19 20 21 22 23 24 25 26 27 28 20 @Override 21 public boolean equals(Object obj) { 22 if (!(obj instanceof Tag)) { 23 return false; 24 } else { 25 Tag otherTag = (Tag)obj; 26 return key.equals(otherTag.key) && value.equals(otherTag.value); 27 } 28 } 29 29 30 31 32 33 30 @Override 31 public int hashCode() { 32 return key.hashCode() + value.hashCode(); 33 } 34 34 35 36 37 38 35 @Override 36 public String toString() { 37 return key + "=" + value; 38 } 39 39 40 40 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/data/TagGroup.java
r16520 r23189 8 8 public interface TagGroup extends Iterable<Tag> { 9 9 10 11 12 13 14 10 /** 11 * returns the value for the given key or null if no tag in this group uses that key 12 * @param key key whose value will be returned; != null 13 */ 14 public String getValue(String key); 15 15 16 17 18 19 20 16 /** 17 * returns true if this tag group contains a tag with the given key 18 * @param key key to check for; != null 19 */ 20 public boolean containsKey(String key); 21 21 22 23 24 25 26 22 /** 23 * returns true if this tag group contains at least one tag with the given value 24 * @param value value to check for; != null 25 */ 26 public boolean containsValue(String value); 27 27 28 29 30 31 32 28 /** 29 * returns true if this tag group contains the given tag 30 * @param tag tag to check for; != null 31 */ 32 public boolean contains(Tag tag); 33 33 34 35 36 37 34 /** 35 * returns the number of tags in this group 36 */ 37 public int size(); 38 38 39 39 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/graph/ConnectorEvaluationGroup.java
r18130 r23189 16 16 public class ConnectorEvaluationGroup extends EvaluationGroup { 17 17 18 19 18 private final Set<Segment> segments; 19 private final List<SegmentNode> borderNodes; 20 20 21 22 23 24 25 26 27 21 /** 22 * @param segments set of Segments, must not be modified 23 * after being used as constructor parameter; != null 24 * @param borderNodes nodes that are used as starting/target nodes for sequences; != null 25 */ 26 public ConnectorEvaluationGroup(Set<Segment> segments, Set<SegmentNode> borderNodes) { 27 assert segments != null && borderNodes != null; 28 28 29 30 31 29 this.segments = segments; 30 this.borderNodes = new ArrayList<SegmentNode>(borderNodes); 31 } 32 32 33 34 35 36 37 38 39 33 /** 34 * returns all nodes that can be used as start/target nodes 35 * @return border nodes; != null 36 */ 37 public Collection<SegmentNode> getBorderNodes() { 38 return borderNodes; 39 } 40 40 41 42 43 44 45 46 47 41 /** 42 * returns all segments in the group 43 * @return segment set; != null 44 */ 45 public Set<Segment> getSegments() { 46 return segments; 47 } 48 48 49 50 51 52 53 54 55 56 57 58 59 60 49 /** 50 * returns a segment sequence that runs from an inbound to an outbound 51 * segment or null if no connection is possible. 52 * {@link EvaluationGroup#evaluate(Collection)} needs be called before this method. 53 * 54 * @param startNode start of the potential sequence; must be border node; != null 55 * @param targetNode target of the potential sequence; must be border node; != null 56 * @return sequence of segments or null 57 */ 58 public List<Segment> getSegmentSequence(SegmentNode startNode, SegmentNode targetNode) { 59 assert startNode != null && borderNodes.contains(startNode); 60 assert targetNode != null && borderNodes.contains(targetNode); 61 61 62 62 if (!evaluated) { throw new IllegalStateException("group not yet evaluated"); } 63 63 64 65 64 int inboundIndex = borderNodes.indexOf(startNode); 65 int outboundIndex = borderNodes.indexOf(targetNode); 66 66 67 68 67 return segmentSequences[inboundIndex][outboundIndex]; 68 } 69 69 70 71 70 @Override 71 protected void evaluateImpl(Collection<Restriction> restrictions) { 72 72 73 73 /* find segment sequences from inbound to outbound segments */ 74 74 75 76 75 @SuppressWarnings("unchecked") //cannot create generic array without cast 76 List<Segment>[][] sequenceArray = new List[borderNodes.size()][borderNodes.size()]; 77 77 78 79 78 for (int startIndex = 0; startIndex < borderNodes.size(); startIndex ++) { 79 for (int targetIndex = 0; targetIndex < borderNodes.size(); targetIndex ++) { 80 80 81 82 83 81 List<Segment> sequence = 82 findSegmentSequence(borderNodes.get(startIndex), 83 borderNodes.get(targetIndex), restrictions); 84 84 85 85 sequenceArray[startIndex][targetIndex] = sequence; 86 86 87 88 87 } 88 } 89 89 90 91 90 segmentSequences = sequenceArray; 91 } 92 92 93 94 95 96 97 93 @Override 94 protected boolean isUsableNode(SegmentNode node) { 95 return shareElement(segments, node.getInboundSegments()) 96 || shareElement(segments, node.getOutboundSegments()); 97 } 98 98 99 100 101 102 99 @Override 100 protected boolean isUsableSegment(Segment segment) { 101 return segments.contains(segment); 102 } 103 103 104 105 106 107 104 @Override 105 public String toString() { 106 return "ConnectorEG " + segments; 107 } 108 108 109 109 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/graph/EvaluationGroup.java
r16520 r23189 19 19 abstract class EvaluationGroup { 20 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 21 protected boolean evaluated = false; 22 23 /** 24 * array of sequences. 25 * First index is inbound segment/start node index, 26 * second index is outbound segment/target node index. 27 * Will contain the segment sequence after evaluation or null if none exists. 28 */ 29 protected List<Segment>[][] segmentSequences; 30 31 private static final List<Segment> EMPTY_SEGMENT_LIST = 32 Collections.unmodifiableList(new ArrayList<Segment>(0)); 33 34 private static final List<Restriction> EMPTY_RESTRICTION_LIST = 35 Collections.unmodifiableList(new ArrayList<Restriction>(0)); 36 37 private static class State { 38 SegmentNode currentNode; 39 Set<SegmentNode> visitedNodes; 40 Collection<Restriction> activeRestrictions; 41 List<Segment> segmentHistory; 42 } 43 44 45 /** 46 * tries to find a legal sequence of segments between two segment nodes. 47 * 48 * @return list of segments if connection is possible, null otherwise. 49 */ 50 protected List<Segment> findSegmentSequence( 51 SegmentNode firstNode, SegmentNode lastNode, 52 Collection<Restriction> restrictions) { 53 54 return findSegmentSequence(firstNode, lastNode, restrictions, 55 EMPTY_RESTRICTION_LIST, EMPTY_RESTRICTION_LIST); 56 57 } 58 59 /** 60 * tries to find a legal sequence of segments between two segments. 61 * 62 * @return list of segments if connection is possible, null otherwise. 63 * The list does NOT include firstSegment and lastSegment, 64 * but they are considered for restrictions. 65 */ 66 protected List<Segment> findSegmentSequence( 67 Segment firstSegment, Segment lastSegment, 68 Collection<Restriction> restrictions) { 69 70 if (firstSegment == lastSegment) { 71 72 return EMPTY_SEGMENT_LIST; 73 74 } else { 75 76 Collection<Restriction> initiallyActiveRestrictions = 77 activeRestrictionsAfterSegment(firstSegment, EMPTY_RESTRICTION_LIST, restrictions); 78 79 Collection<Restriction> restrictionsForbiddenAtLastNode = new HashSet<Restriction>(); 80 for (Restriction restriction : restrictions) { 81 if (restriction.getTos().contains(lastSegment)) { 82 restrictionsForbiddenAtLastNode.add(restriction); 83 } 84 } 85 86 return findSegmentSequence( 87 firstSegment.getNode2(), lastSegment.getNode1(), restrictions, 88 initiallyActiveRestrictions, restrictionsForbiddenAtLastNode); 89 } 90 91 } 92 93 /** 94 * tries to find a legal sequence of segments between two segment nodes. 95 * 96 * @param restrictions all restrictions that have to be taken into account 97 * @param initiallyActiveRestrictions restrictions that are already active at firstNode 98 * @param restrictionsForbiddenAtLastNode restrictions that must NOT be active at lastNode 99 * @return list of segments if connection is possible, null otherwise. 100 */ 101 private List<Segment> findSegmentSequence( 102 SegmentNode firstNode, SegmentNode lastNode, 103 Collection<Restriction> restrictions, 104 Collection<Restriction> initiallyActiveRestrictions, 105 Collection<Restriction> restrictionsForbiddenAtLastNode) { 106 107 if (firstNode == lastNode 108 && !shareElement(initiallyActiveRestrictions, restrictionsForbiddenAtLastNode)) { 109 return EMPTY_SEGMENT_LIST; 110 } 111 112 Queue<State> stateQueue = new LinkedList<State>(); 113 stateQueue.add(createStartingState(firstNode, initiallyActiveRestrictions)); 114 115 /* search for a possible segment sequence */ 116 117 while (stateQueue.size() > 0) { 118 119 State state = stateQueue.poll(); 120 121 Collection<State> subsequentStates = createSubsequentStates(state, restrictions); 122 123 for (State subsequentState : subsequentStates) { 124 if (subsequentState.currentNode == lastNode 125 && !shareElement(subsequentState.activeRestrictions, 126 restrictionsForbiddenAtLastNode)) { 127 return subsequentState.segmentHistory; 128 } 129 } 130 131 stateQueue.addAll(subsequentStates); 132 133 } 134 135 return null; 136 } 137 138 private static State createStartingState(SegmentNode firstNode, 139 Collection<Restriction> initiallyActiveRestrictions) { 140 141 State startingState = new State(); 142 startingState.currentNode = firstNode; 143 startingState.activeRestrictions = initiallyActiveRestrictions; 144 startingState.segmentHistory = EMPTY_SEGMENT_LIST; 145 startingState.visitedNodes = new HashSet<SegmentNode>(); 146 startingState.visitedNodes.add(firstNode); 147 148 return startingState; 149 } 150 151 private List<State> createSubsequentStates(State state, Collection<Restriction> allRestrictions) { 152 153 List<State> subsequentStates = new ArrayList<State>(); 154 155 for (Segment segment : state.currentNode.getOutboundSegments()) { 156 157 if (isUsableSegment(segment) && 158 isLegalSegment(segment, state.activeRestrictions)) { 159 160 State newState = new State(); 161 162 newState.activeRestrictions = activeRestrictionsAfterSegment( 163 segment, state.activeRestrictions, allRestrictions); 164 165 newState.segmentHistory = new ArrayList<Segment>(state.segmentHistory.size() + 1); 166 newState.segmentHistory.addAll(state.segmentHistory); 167 newState.segmentHistory.add(segment); 168 169 newState.currentNode = segment.getNode2(); 170 171 newState.visitedNodes = new HashSet<SegmentNode>(state.visitedNodes); 172 newState.visitedNodes.add(newState.currentNode); 173 174 /* add state to queue, 175 * but avoid cycles as well as leaving the node set 176 */ 177 178 if (!state.visitedNodes.contains(newState.currentNode) 179 && isUsableNode(newState.currentNode)) { 180 181 subsequentStates.add(newState); 182 183 } 184 } 185 } 186 187 return subsequentStates; 188 } 189 190 /** 191 * returns all restrictions from a collection that have a segment as from member 192 * @return segment list; != null; must not be modified. 193 * May throw an exception when modifying is attempted. 194 */ 195 private static List<Restriction> getRestrictionsStartedBySegment( 196 Collection<Restriction> restrictions, Segment segment) { 197 198 List<Restriction> result = EMPTY_RESTRICTION_LIST; 199 for (Restriction restriction : restrictions) { 200 if (restriction.getFrom() == segment) { 201 if (result == EMPTY_RESTRICTION_LIST) { 202 result = new ArrayList<Restriction>(restrictions.size()); 203 } 204 result.add(restriction); 205 } 206 } 207 208 return result; 209 } 210 211 private static Collection<Restriction> activeRestrictionsAfterSegment(Segment segment, 212 Collection<Restriction> activeRestrictionsBeforeSegment, 213 Collection<Restriction> allRestrictions) { 214 215 Collection<Restriction> result = EMPTY_RESTRICTION_LIST; 216 217 for (Restriction restriction : activeRestrictionsBeforeSegment) { 218 if (restriction.getVias().contains(segment)) { 219 if (result == EMPTY_RESTRICTION_LIST) { 220 result = new ArrayList<Restriction>(allRestrictions.size()); 221 } 222 result.add(restriction); 223 } 224 } 225 226 Collection<Restriction> newRestrictions = 227 getRestrictionsStartedBySegment(allRestrictions, segment); 228 229 if (newRestrictions.size() > 0) { 230 if (result == EMPTY_RESTRICTION_LIST) { 231 result = newRestrictions; 232 } else { 233 result.addAll(newRestrictions); 234 } 235 } 236 237 return result; 238 } 239 240 private static boolean isLegalSegment( 241 Segment segment, Collection<Restriction> activeRestrictions) { 242 243 for (Restriction restriction : activeRestrictions) { 244 if (restriction.getTos().contains(segment)) { 245 return false; 246 } 247 } 248 249 return true; 250 } 251 252 /** returns true iff at least one element is contained in both collections */ 253 protected static boolean shareElement( 254 Collection<?> collection1, Collection<?> collection2) { 255 for (Object element : collection1) { 256 if (collection2.contains(element)) { 257 return true; 258 } 259 } 260 return false; 261 } 262 263 public final void evaluate(Collection<Restriction> restrictions) { 264 265 if (evaluated) { return; } 266 267 evaluateImpl(restrictions); 268 269 evaluated = true; 270 } 271 272 /** 273 * finds in- and outbound segments (if necessary) and segment sequences. 274 * After calling this method, the group must be correctly evaluated 275 * (see {@link #isCorrectlyEvaluated()}). 276 * 277 * @param restrictions restrictions that are used when determining possible connections, 278 * will not be modified; != null 279 */ 280 abstract protected void evaluateImpl(Collection<Restriction> restrictions); 281 282 /** 283 * returns whether a node can be used while finding a segment sequence 284 * @param node node to check; != null 285 */ 286 abstract protected boolean isUsableNode(SegmentNode node); 287 288 /** 289 * returns whether a segment can be used while finding a segment sequence 290 * @param segment segment to check; != null 291 */ 292 abstract protected boolean isUsableSegment(Segment segment); 293 293 294 294 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/graph/GraphEdge.java
r18130 r23189 12 12 public interface GraphEdge { 13 13 14 15 14 /** returns the node this edge starts at; != null */ 15 GraphNode getStartNode(); 16 16 17 18 17 /** returns the node this edge leads to; != null */ 18 GraphNode getTargetNode(); 19 19 20 21 22 23 24 25 20 /** returns all property types for which property values are available */ 21 Collection<GraphEdgePropertyType<?>> getAvailableProperties(); 22 23 /** TODO */ 24 <V> V getPropertyValue(GraphEdgePropertyType<V> property); 25 26 26 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/graph/GraphNode.java
r16520 r23189 13 13 public interface GraphNode { 14 14 15 16 17 18 15 /** 16 * returns all edges that lead to this GraphNode; != null 17 */ 18 public Collection<GraphEdge> getInboundEdges(); 19 19 20 21 22 23 20 /** 21 * returns all edges that start at this GraphNode; != null 22 */ 23 public Collection<GraphEdge> getOutboundEdges(); 24 24 25 26 27 28 29 30 31 25 /** 26 * returns the SegmentNode this GraphNode is based on 27 * 28 * @return SegmentNode, must be one of the nodes of the Segment returned 29 * by {@link #getSegment()}; != null 30 */ 31 public SegmentNode getSegmentNode(); 32 32 33 34 35 36 37 38 33 /** 34 * returns the Segment this GraphNode is based on 35 * 36 * @return Segment; != null 37 */ 38 public Segment getSegment(); 39 39 40 40 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/graph/JunctionEvaluationGroup.java
r18130 r23189 15 15 public class JunctionEvaluationGroup extends EvaluationGroup { 16 16 17 17 private final Set<SegmentNode> segmentNodes; 18 18 19 20 19 protected List<Segment> inboundSegments; 20 protected List<Segment> outboundSegments; 21 21 22 23 24 25 26 27 28 29 22 /** 23 * @param segmentNodes set of SegmentNodes, must not be modified 24 * after being used as constructor parameter; != null 25 */ 26 public JunctionEvaluationGroup(Set<SegmentNode> segmentNodes) { 27 assert segmentNodes != null; 28 this.segmentNodes = segmentNodes; 29 } 30 30 31 32 33 34 35 36 37 38 39 40 31 /** 32 * returns all segments that can be used to enter this group. 33 * {@link #evaluate(Iterable)} needs be called before this method. 34 * 35 * @return segment collection; != null 36 */ 37 public Collection<Segment> getInboundSegments() { 38 if (!evaluated) { throw new IllegalStateException("group not yet evaluated"); } 39 return inboundSegments; 40 } 41 41 42 43 44 45 46 47 48 49 50 51 42 /** 43 * returns all segments that can be used to leave this group. 44 * {@link #evaluate(Iterable)} needs be called before this method. 45 * 46 * @return segment collection; != null 47 */ 48 public Collection<Segment> getOutboundSegments() { 49 if (!evaluated) { throw new IllegalStateException("group not yet evaluated"); } 50 return outboundSegments; 51 } 52 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 53 /** 54 * returns a segment sequence that runs from an inbound to an outbound 55 * segment or null if no connection is possible. 56 * {@link EvaluationGroup#evaluate(Collection)} needs be called before this method. 57 * 58 * @param inboundSegment start of the potential sequence; 59 * must be inbound segment; != null 60 * @param outboundSegment target of the potential sequence; 61 * must be outbound segment; != null 62 * @return sequence of segments or null 63 */ 64 public List<Segment> getSegmentSequence(Segment inboundSegment, Segment outboundSegment) { 65 assert inboundSegment != null && inboundSegments.contains(inboundSegment); 66 assert outboundSegment != null && outboundSegments.contains(outboundSegment); 67 67 68 68 if (!evaluated) { throw new IllegalStateException("group not yet evaluated"); } 69 69 70 71 70 int inboundIndex = inboundSegments.indexOf(inboundSegment); 71 int outboundIndex = outboundSegments.indexOf(outboundSegment); 72 72 73 74 73 return segmentSequences[inboundIndex][outboundIndex]; 74 } 75 75 76 77 76 @Override 77 protected void evaluateImpl(final Collection<Restriction> restrictions) { 78 78 79 79 assert restrictions != null; 80 80 81 82 81 /* find inbound and outbound segments. An inbound segment is a segment whose target 82 * is in the set and whose start node isn't (analogous for outbound segments) */ 83 83 84 85 84 inboundSegments = new ArrayList<Segment>(); 85 outboundSegments = new ArrayList<Segment>(); 86 86 87 88 89 90 91 92 93 94 95 96 97 98 87 for (SegmentNode segmentNode : segmentNodes) { 88 for (Segment segment : segmentNode.getInboundSegments()) { 89 if (!segmentNodes.contains(segment.getNode1())) { 90 inboundSegments.add(segment); 91 } 92 } 93 for (Segment segment : segmentNode.getOutboundSegments()) { 94 if (!segmentNodes.contains(segment.getNode2())) { 95 outboundSegments.add(segment); 96 } 97 } 98 } 99 99 100 100 /* find segment sequences from inbound to outbound segments */ 101 101 102 103 102 @SuppressWarnings("unchecked") //cannot create generic array without cast 103 List<Segment>[][] sequenceArray = new List[inboundSegments.size()][outboundSegments.size()]; 104 104 105 106 105 for (int inboundIndex = 0; inboundIndex < inboundSegments.size(); inboundIndex ++) { 106 for (int outboundIndex = 0; outboundIndex < outboundSegments.size(); outboundIndex ++) { 107 107 108 109 110 108 List<Segment> sequence = 109 findSegmentSequence(inboundSegments.get(inboundIndex), 110 outboundSegments.get(outboundIndex), restrictions); 111 111 112 112 sequenceArray[inboundIndex][outboundIndex] = sequence; 113 113 114 115 114 } 115 } 116 116 117 117 segmentSequences = sequenceArray; 118 118 119 119 } 120 120 121 122 123 124 121 @Override 122 protected boolean isUsableNode(SegmentNode node) { 123 return segmentNodes.contains(node); 124 } 125 125 126 127 128 129 130 126 @Override 127 protected boolean isUsableSegment(Segment segment) { 128 return segmentNodes.contains(segment.getNode1()) 129 && segmentNodes.contains(segment.getNode2()); 130 } 131 131 132 133 134 135 132 @Override 133 public String toString() { 134 return "JunctionEG " + segmentNodes; 135 } 136 136 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/graph/TSBasedWayGraph.java
r18130 r23189 23 23 public class TSBasedWayGraph implements WayGraph, TransitionStructureObserver { 24 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 25 private static final GraphEdgePropertyType<?>[] PROPERTY_TYPES = 26 {GraphEdgeSegments.PROPERTY}; 27 //TODO: -> parameter 28 29 private static class GraphNodeImpl implements GraphNode { 30 private final SegmentNode node; 31 private final Segment segment; 32 private final List<GraphEdge> incomingEdges = new ArrayList<GraphEdge>(); 33 private final List<GraphEdge> outgoingEdges = new ArrayList<GraphEdge>(); 34 public GraphNodeImpl(SegmentNode node, Segment segment) { 35 assert node != null && segment != null; 36 assert segment.getNode1() == node || segment.getNode2() == node; 37 this.node = node; 38 this.segment = segment; 39 } 40 public SegmentNode getSegmentNode() { 41 return node; 42 } 43 public Segment getSegment() { 44 return segment; 45 } 46 public void addIncomingEdge(GraphEdge edge) { 47 assert edge != null; 48 incomingEdges.add(edge); 49 } 50 public Collection<GraphEdge> getInboundEdges() { 51 return incomingEdges; 52 } 53 public void addOutgoingEdge(GraphEdge edge) { 54 assert edge != null; 55 outgoingEdges.add(edge); 56 } 57 public Collection<GraphEdge> getOutboundEdges() { 58 return outgoingEdges; 59 } 60 @Override 61 public String toString() { 62 return "(" + node + "; " + segment + ")"; 63 } 64 } 65 66 private static class GraphEdgeImpl implements GraphEdge { 67 68 private final GraphNode startNode; 69 private final GraphNode targetNode; 70 private final Map<GraphEdgePropertyType<?>, Object> properties; 71 72 public GraphEdgeImpl(GraphNode startNode, GraphNode targetNode, 73 Map<GraphEdgePropertyType<?>, Object> properties) { 74 assert startNode != null && targetNode != null && properties != null; 75 this.startNode = startNode; 76 this.targetNode = targetNode; 77 this.properties = properties; 78 } 79 80 public GraphNode getStartNode() { 81 return startNode; 82 } 83 public GraphNode getTargetNode() { 84 return targetNode; 85 } 86 87 public Collection<GraphEdgePropertyType<?>> getAvailableProperties() { 88 return properties.keySet(); 89 } 90 public <V> V getPropertyValue(GraphEdgePropertyType<V> property) { 91 V result = (V) properties.get(property); 92 return result; 93 } 94 95 @Override 96 public String toString() { 97 return "(" + startNode + "-->" + targetNode + ")"; 98 } 99 100 }; 101 102 private final Set<WayGraphObserver> observers = new HashSet<WayGraphObserver>(); 103 104 private final TransitionStructure transitionStructure; 105 106 private Collection<GraphNode> nodes; 107 private List<GraphEdge> edges; 108 109 /** 110 * create a WayGraph based on a {@link TransitionStructure} 111 * @param transitionStructure transition structure this graph is to be based on; != null 112 */ 113 public TSBasedWayGraph(TransitionStructure transitionStructure) { 114 assert transitionStructure != null; 115 116 this.transitionStructure = transitionStructure; 117 transitionStructure.addObserver(this); 118 119 createNodesAndEdges(); 120 } 121 122 public Collection<GraphEdge> getEdges() { 123 return edges; 124 } 125 126 public Collection<GraphNode> getNodes() { 127 return nodes; 128 } 129 130 private void createNodesAndEdges() { 131 132 Collection<EvaluationGroup> evaluationGroups = 133 createEvaluationGroups(transitionStructure); 134 135 for (EvaluationGroup evaluationGroup : evaluationGroups) { 136 evaluationGroup.evaluate(transitionStructure.getRestrictions()); 137 } 138 139 createNodesAndEdgesFromEvaluationGroups(evaluationGroups); 140 141 evaluationGroups = null; 142 } 143 144 private static Collection<EvaluationGroup> createEvaluationGroups( 145 TransitionStructure transitionStructure) { 146 147 Map<SegmentNode, Set<SegmentNode>> nodeSetMap = 148 new HashMap<SegmentNode, Set<SegmentNode>>(); 149 150 /* first step: everything that is part of the same restriction goes into the same set */ 151 152 for (Restriction restriction : transitionStructure.getRestrictions()) { 153 154 /* group every node in via segments (which includes the 155 * last node of from and the first node of to) into a set */ 156 157 SegmentNode firstNode = restriction.getFrom().getNode2(); 158 createSetIfHasNone(firstNode, nodeSetMap); 159 160 for (Segment segment : restriction.getVias()) { 161 putInSameSet(segment.getNode1(), firstNode, nodeSetMap); 162 putInSameSet(segment.getNode2(), firstNode, nodeSetMap); 163 } 164 165 for (Segment segment : restriction.getTos()) { 166 putInSameSet(segment.getNode1(), firstNode, nodeSetMap); 167 } 168 169 } 170 171 /* second step: create own sets for each junction and end point 172 * (node connected with more than / less than two nodes). */ 173 174 for (SegmentNode node : transitionStructure.getNodes()) { 175 176 if (!nodeSetMap.containsKey(node) 177 && !isConnectedWithExactly2Nodes(node)) { 178 179 createSetIfHasNone(node, nodeSetMap); 180 181 } 182 183 } 184 185 /* third step: create segment sets for all segments that are not in one of the node sets 186 * (that is, at least one node is not part of a junction evaluation group 187 * or the nodes are part of different junction evaluation groups) */ 188 189 Map<Segment, Set<Segment>> segmentSetMap = 190 new HashMap<Segment, Set<Segment>>(); 191 192 for (Segment segment : transitionStructure.getSegments()) { 193 194 SegmentNode node1 = segment.getNode1(); 195 SegmentNode node2 = segment.getNode2(); 196 197 if (!nodeSetMap.containsKey(node1) || !nodeSetMap.containsKey(node2) 198 || nodeSetMap.get(node1) != nodeSetMap.get(node2)) { 199 200 createSetIfHasNone(segment, segmentSetMap); 201 202 for (Segment subsequentSegment : segment.getNode2().getOutboundSegments()) { 203 if (!nodeSetMap.containsKey(node2) 204 || subsequentSegment.getNode2() == node1) { 205 putInSameSet(subsequentSegment, segment, segmentSetMap); 206 } 207 } 208 //note that segments leading to this segment will share sets anyway, 209 //because this segment is a subsequent segment of them 210 211 212 } 213 214 } 215 216 /* create EvaluationGroup objects */ 217 218 Collection<EvaluationGroup> evaluationGroups = 219 new ArrayList<EvaluationGroup>(nodeSetMap.size() + segmentSetMap.size()); 220 221 Set<Set<SegmentNode>> nodeSets = new HashSet<Set<SegmentNode>>(nodeSetMap.values()); 222 for (Set<SegmentNode> nodeSet : nodeSets) { 223 evaluationGroups.add(new JunctionEvaluationGroup(nodeSet)); 224 } 225 226 HashSet<Set<Segment>> hashSets = new HashSet<Set<Segment>>(segmentSetMap.values()); 227 for (Set<Segment> segmentSet : hashSets) { 228 Set<SegmentNode> borderNodes = new HashSet<SegmentNode>(); 229 for (Segment segment : segmentSet) { 230 if (nodeSetMap.containsKey(segment.getNode1())) { 231 borderNodes.add(segment.getNode1()); 232 } 233 if (nodeSetMap.containsKey(segment.getNode2())) { 234 borderNodes.add(segment.getNode2()); 235 } 236 } 237 evaluationGroups.add(new ConnectorEvaluationGroup(segmentSet, borderNodes)); 238 } 239 240 return evaluationGroups; 241 } 242 243 private void createNodesAndEdgesFromEvaluationGroups( 244 Collection<EvaluationGroup> evaluationGroups) { 245 246 nodes = new LinkedList<GraphNode>(); 247 edges = new LinkedList<GraphEdge>(); 248 249 //map from Segments to GraphNodes; 250 //for those GraphNodes representing an "approaching node on segment" state 251 final Map<Segment, GraphNodeImpl> segment2GNMap_approaching = 252 new HashMap<Segment, GraphNodeImpl>(); 253 254 //map from Segments to GraphNodes; 255 //for those GraphNodes representing a "leaving node on segment" state 256 final Map<Segment, GraphNodeImpl> segment2GNMap_leaving = 257 new HashMap<Segment, GraphNodeImpl>(); 258 259 //map from SegmentNodes to GraphNode collections; 260 //for those GraphNodes representing an "approaching node on segment" state 261 final Map<SegmentNode, Collection<GraphNodeImpl>> segNode2GNMap_approaching = 262 new HashMap<SegmentNode, Collection<GraphNodeImpl>>(); 263 264 //map from SegmentNodes to GraphNodes collections; 265 //for those GraphNodes representing a "leaving node on segment" state 266 final Map<SegmentNode, Collection<GraphNodeImpl>> segNode2GNMap_leaving = 267 new HashMap<SegmentNode, Collection<GraphNodeImpl>>(); 268 269 270 271 /* create graph nodes and edges for junction evaluation groups */ 272 273 for (EvaluationGroup evaluationGroup : evaluationGroups) { 274 if (evaluationGroup instanceof JunctionEvaluationGroup) { 275 276 JunctionEvaluationGroup junctionEG = (JunctionEvaluationGroup) evaluationGroup; 277 278 //create graph nodes 279 for (Segment segment : junctionEG.getInboundSegments()) { 280 GraphNodeImpl graphNode = new GraphNodeImpl(segment.getNode2(), segment); 281 nodes.add(graphNode); 282 segment2GNMap_approaching.put(segment, graphNode); 283 addToCollectionMap(segNode2GNMap_approaching, segment.getNode2(), graphNode); 284 } 285 for (Segment segment : junctionEG.getOutboundSegments()) { 286 GraphNodeImpl graphNode = new GraphNodeImpl(segment.getNode1(), segment); 287 nodes.add(graphNode); 288 segment2GNMap_leaving.put(segment, graphNode); 289 addToCollectionMap(segNode2GNMap_leaving, segment.getNode1(), graphNode); 290 } 291 292 //create graph edges for all segment sequences between in- and outbound edges 293 for (Segment inboundSegment : junctionEG.getInboundSegments()) { 294 for (Segment outboundSegment : junctionEG.getOutboundSegments()) { 295 296 List<Segment> segmentSequence = 297 junctionEG.getSegmentSequence(inboundSegment, outboundSegment); 298 299 if (segmentSequence != null) { 300 301 createGraphEdge( 302 segment2GNMap_approaching.get(inboundSegment), 303 segment2GNMap_leaving.get(outboundSegment), 304 segmentSequence, 305 junctionEG); 306 307 } 308 } 309 } 310 311 } 312 } 313 314 /* create graph edges for connector evaluation groups. 315 * Because GraphNodes are created for pairs of SegmentNodes (from connector groups) 316 * and Segments (from junction groups), the GraphNodes already exist. 317 */ 318 319 for (EvaluationGroup evaluationGroup : evaluationGroups) { 320 if (evaluationGroup instanceof ConnectorEvaluationGroup) { 321 322 ConnectorEvaluationGroup connectorEG = (ConnectorEvaluationGroup) evaluationGroup; 323 324 for (SegmentNode startNode : connectorEG.getBorderNodes()) { 325 for (SegmentNode targetNode : connectorEG.getBorderNodes()) { 326 327 if (segNode2GNMap_leaving.containsKey(startNode) 328 && segNode2GNMap_approaching.containsKey(targetNode)) { 329 330 for (GraphNodeImpl startGraphNode : segNode2GNMap_leaving.get(startNode)) { 331 for (GraphNodeImpl targetGraphNode : segNode2GNMap_approaching.get(targetNode)) { 332 333 if (connectorEG.getSegments().contains(startGraphNode.getSegment()) 334 && connectorEG.getSegments().contains(targetGraphNode.getSegment())) { 335 336 List<Segment> segmentSequence = 337 connectorEG.getSegmentSequence(startNode, targetNode); 338 339 if (segmentSequence != null) { 340 createGraphEdge( 341 startGraphNode, 342 targetGraphNode, 343 segmentSequence, 344 connectorEG); 345 } 346 347 } 348 349 } 350 } 351 352 } 353 354 } 355 } 356 357 } 358 } 359 360 } 361 362 private void createGraphEdge( 363 GraphNodeImpl startNode, GraphNodeImpl targetNode, 364 List<Segment> segments, ConnectorEvaluationGroup evaluationGroup) { 365 366 Map<GraphEdgePropertyType<?>, Object> properties = 367 new HashMap<GraphEdgePropertyType<?>, Object>(); //TODO: replace HashMap with List-based solution 368 369 for (GraphEdgePropertyType<?> propertyType : PROPERTY_TYPES) { 370 Object value = propertyType.evaluate(evaluationGroup, segments, transitionStructure); 371 properties.put(propertyType, value); 372 } 373 374 createGraphEdge(startNode, targetNode, properties); 375 376 } 377 378 private void createGraphEdge( 379 GraphNodeImpl startNode, GraphNodeImpl targetNode, 380 List<Segment> segments, JunctionEvaluationGroup evaluationGroup) { 381 382 Map<GraphEdgePropertyType<?>, Object> properties = 383 new HashMap<GraphEdgePropertyType<?>, Object>(); //TODO: replace HashMap with List-based solution 384 385 for (GraphEdgePropertyType<?> propertyType : PROPERTY_TYPES) { 386 Object value = propertyType.evaluate(evaluationGroup, segments, transitionStructure); 387 properties.put(propertyType, value); 388 } 389 390 createGraphEdge(startNode, targetNode, properties); 391 392 } 393 394 /** 395 * creates a GraphEdge; 396 * adds it to its nodes' collections and {@link #edges} collection. 397 */ 398 private void createGraphEdge(GraphNodeImpl startNode, GraphNodeImpl targetNode, 399 Map<GraphEdgePropertyType<?>, Object> properties) { 400 401 GraphEdge newEdge = new GraphEdgeImpl(startNode, targetNode, properties); 402 403 startNode.addOutgoingEdge(newEdge); 404 targetNode.addIncomingEdge(newEdge); 405 406 edges.add(newEdge); 407 408 } 409 410 private static boolean isConnectedWithExactly2Nodes(SegmentNode node) { 411 412 Set<SegmentNode> connectedNodes = new HashSet<SegmentNode>(2); 413 414 for (Segment segment : node.getInboundSegments()) { 415 connectedNodes.add(segment.getNode1()); 416 } 417 for (Segment segment : node.getOutboundSegments()) { 418 connectedNodes.add(segment.getNode2()); 419 } 420 421 return connectedNodes.size() == 2; 422 } 423 424 /** 425 * creates a set for an object if none exists in a map. 426 * The set will contain the object and be added to the map with the object being its key. 427 */ 428 private static <T> void createSetIfHasNone(T object, Map<T, Set<T>> objectSetMap) { 429 430 if (!objectSetMap.containsKey(object)) { 431 @SuppressWarnings("unchecked") //no set with generic parameter can be created directly 432 Set<T> set = new HashSet(); 433 set.add(object); 434 objectSetMap.put(object, set); 435 } 436 437 } 438 439 /** 440 * puts an object in another object's set. 441 * If both nodes have sets already, these sets are merged. 442 * The objectSetMap is modified accordingly. 443 * 444 * @param object object that might or might not be in a set; != null 445 * @param objectInSet object that is guaranteed to be in a set; != null 446 * @param objectSetMap map from objects to the one set they are part of; != null 447 */ 448 private static <T> void putInSameSet(T object, T objectInSet, Map<T, Set<T>> objectSetMap) { 449 assert object != null && objectInSet != null && objectSetMap != null; 450 assert objectSetMap.containsKey(objectInSet); 451 452 Set<T> set = objectSetMap.get(objectInSet); 453 454 if (objectSetMap.containsKey(object)) { 455 456 /* merge the two sets */ 457 Set<T> oldSet = objectSetMap.get(object); 458 for (T objectFromOldSet : oldSet) { 459 set.add(objectFromOldSet); 460 objectSetMap.put(objectFromOldSet, set); 461 } 462 463 } else { 464 465 /* add object to objectInSet's set */ 466 set.add(object); 467 objectSetMap.put(object, set); 468 469 } 470 471 } 472 473 private static <K, E> void addToCollectionMap(final Map<K, Collection<E>> map, K key, E entry) { 474 if (!map.containsKey(key)) { 475 Collection<E> newCollection = new ArrayList<E>(); 476 map.put(key, newCollection); 477 } 478 map.get(key).add(entry); 479 } 480 481 public void update(TransitionStructure transitionStructure) { 482 createNodesAndEdges(); 483 notifyObservers(); 484 } 485 486 public void addObserver(WayGraphObserver observer) { 487 observers.add(observer); 488 } 489 490 public void deleteObserver(WayGraphObserver observer) { 491 observers.remove(observer); 492 } 493 494 private void notifyObservers() { 495 for (WayGraphObserver observer : observers) { 496 observer.update(this); 497 } 498 } 499 499 500 500 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/graph/WayGraph.java
r16520 r23189 8 8 public interface WayGraph { 9 9 10 11 10 Collection<GraphNode> getNodes(); 11 Collection<GraphEdge> getEdges(); 12 12 13 14 15 16 17 18 19 13 /** 14 * adds an observer. 15 * Does nothing if the parameter is already an observer of this WayGraph. 16 * 17 * @param observer observer object, != null 18 */ 19 public void addObserver(WayGraphObserver observer); 20 20 21 22 23 24 25 26 27 21 /** 22 * deletes an observer that has been added using {@link #addObserver(WayGraphObserver)}. 23 * Does nothing if the parameter isn't currently an observer of this WayGraph. 24 * 25 * @param observer observer object, != null 26 */ 27 public void deleteObserver(WayGraphObserver observer); 28 28 29 29 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/graph/WayGraphObserver.java
r16520 r23189 7 7 public interface WayGraphObserver { 8 8 9 10 11 12 13 9 /** 10 * informs this observer about changes in an observed graph 11 * @param wayGraph observed graph that has changed; != null 12 */ 13 public void update(WayGraph wayGraph); 14 14 15 15 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/GraphEdgePropertyType.java
r18130 r23189 17 17 */ 18 18 public interface GraphEdgePropertyType<V> { 19 20 /**21 * determines the property value for segments created from junction groups22 */23 public V evaluate(JunctionEvaluationGroup junctionGroup,24 List<Segment> segmentSequence,25 TransitionStructure transitionStructure);26 19 27 /** 28 * determines the property value for segments created from connector groups 29 */ 30 public V evaluate(ConnectorEvaluationGroup connectorGroup, 31 List<Segment> segmentSequence, 32 TransitionStructure transitionStructure); 33 20 /** 21 * determines the property value for segments created from junction groups 22 */ 23 public V evaluate(JunctionEvaluationGroup junctionGroup, 24 List<Segment> segmentSequence, 25 TransitionStructure transitionStructure); 26 27 /** 28 * determines the property value for segments created from connector groups 29 */ 30 public V evaluate(ConnectorEvaluationGroup connectorGroup, 31 List<Segment> segmentSequence, 32 TransitionStructure transitionStructure); 33 34 34 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/GraphEdgeSegments.java
r18130 r23189 12 12 * property for the graph that is being constructed will preserve information 13 13 * from the {@link TransitionStructure}. 14 * 14 * 15 15 * TODO: for some purposes, segments are not needed (only coordinate lists; 16 16 * without properties etc.) 17 17 */ 18 18 public final class GraphEdgeSegments implements GraphEdgePropertyType<List<Segment>> { 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 19 20 public static final GraphEdgeSegments PROPERTY = new GraphEdgeSegments(); 21 22 /** 23 * private constructor to make sure that {@link #INSTANCE} is the only instance 24 */ 25 private GraphEdgeSegments() { } 26 27 public List<Segment> evaluate(JunctionEvaluationGroup junctionGroup, 28 List<Segment> segmentSequence, TransitionStructure transitionStructure) { 29 return segmentSequence; 30 } 31 32 public List<Segment> evaluate(ConnectorEvaluationGroup connectorGroup, 33 List<Segment> segmentSequence, TransitionStructure transitionStructure) { 34 return segmentSequence; 35 } 36 37 37 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadIncline.java
r19216 r23189 8 8 public class RoadIncline implements RoadPropertyType<Float> { 9 9 10 11 12 13 10 public <N, W, R, M> Float evaluateN(N node, AccessParameters accessParameters, 11 DataSource<N,W,R,M> dataSource) { 12 return null; 13 }; 14 14 15 16 17 15 public <N, W, R, M> Float evaluateW(W way, boolean forward, AccessParameters accessParameters, 16 DataSource<N,W,R,M> dataSource) { 17 assert way != null && accessParameters != null && dataSource != null; 18 18 19 20 19 TagGroup tags = dataSource.getTagsW(way); 20 String inclineString = tags.getValue("incline"); 21 21 22 23 24 25 26 27 28 29 30 22 if (inclineString != null) { 23 Float incline = ValueStringParser.parseIncline(inclineString); 24 if (incline != null) { 25 if (!forward) { 26 incline = -incline; 27 } 28 return incline; 29 } 30 } 31 31 32 33 32 return null; 33 }; 34 34 35 36 35 public boolean isUsable(Object propertyValue, AccessParameters accessParameters) { 36 assert propertyValue instanceof Float; 37 37 38 38 float incline = (Float)propertyValue; 39 39 40 41 42 43 40 Float maxInclineUp = 41 accessParameters.getVehiclePropertyValue(VehiclePropertyTypes.MAX_INCLINE_UP); 42 Float maxInclineDown = 43 accessParameters.getVehiclePropertyValue(VehiclePropertyTypes.MAX_INCLINE_DOWN); 44 44 45 46 47 48 49 50 51 52 45 if (maxInclineUp != null && incline > maxInclineUp) { 46 return false; 47 } else if (maxInclineDown != null && -incline > maxInclineDown) { 48 return false; 49 } else { 50 return true; 51 } 52 } 53 53 54 54 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadMaxaxleload.java
r20243 r23189 6 6 7 7 public class RoadMaxaxleload extends RoadValueLimit { 8 9 10 11 12 13 14 8 public RoadMaxaxleload() { 9 super("maxaxleload", AXLELOAD, LimitType.MAXIMUM); 10 } 11 @Override 12 protected Float parse(String valueString) { 13 return ValueStringParser.parseWeight(valueString); 14 } 15 15 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadMaxheight.java
r20243 r23189 6 6 7 7 public class RoadMaxheight extends RoadValueLimit { 8 9 10 11 12 13 14 8 public RoadMaxheight() { 9 super("maxheight", HEIGHT, LimitType.MAXIMUM); 10 } 11 @Override 12 protected Float parse(String valueString) { 13 return ValueStringParser.parseMeasure(valueString); 14 } 15 15 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadMaxlength.java
r20243 r23189 6 6 7 7 public class RoadMaxlength extends RoadValueLimit { 8 9 10 11 12 13 14 8 public RoadMaxlength() { 9 super("maxlength", LENGTH, LimitType.MAXIMUM); 10 } 11 @Override 12 protected Float parse(String valueString) { 13 return ValueStringParser.parseMeasure(valueString); 14 } 15 15 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadMaxspeed.java
r19216 r23189 8 8 public class RoadMaxspeed implements RoadPropertyType<Float> { 9 9 10 10 private DataSource<?, ?, ?, ?> lastDataSource; 11 11 12 13 14 15 16 12 /** 13 * (re)creates information like boundaries if data source has changed 14 * since last call to {@link #evaluate(Object, boolean, AccessParameters, DataSource)} 15 */ 16 private <N, W, R, M> void initializeIfNecessary(DataSource<N, W, R, M> dataSource) { 17 17 18 18 if (dataSource != lastDataSource) { 19 19 20 21 22 23 24 25 26 20 /* 21 * 22 * currently no activities; 23 * place boundaries or similar features can be handled here 24 * once there is consensus on the topic of implicit maxspeeds, trafficzones etc. 25 * 26 */ 27 27 28 29 30 28 lastDataSource = dataSource; 29 } 30 } 31 31 32 33 34 32 public <N, W, R, M> Float evaluateN(N node, AccessParameters accessParameters, 33 DataSource<N, W, R, M> dataSource) { 34 assert node != null && accessParameters != null && dataSource != null; 35 35 36 36 initializeIfNecessary(dataSource); 37 37 38 39 38 return evaluateTags(dataSource.getTagsN(node)); 39 } 40 40 41 42 43 41 public <N, W, R, M> Float evaluateW(W way, boolean forward, AccessParameters accessParameters, 42 DataSource<N, W, R, M> dataSource) { 43 assert way != null && accessParameters != null && dataSource != null; 44 44 45 45 initializeIfNecessary(dataSource); 46 46 47 48 47 return evaluateTags(dataSource.getTagsW(way)); 48 } 49 49 50 51 50 private Float evaluateTags(TagGroup tags) { 51 String maxspeedString = tags.getValue("maxspeed"); 52 52 53 53 if (maxspeedString != null) { 54 54 55 56 57 58 55 Float maxspeed = ValueStringParser.parseSpeed(maxspeedString); 56 if (maxspeed != null) { 57 return maxspeed; 58 } 59 59 60 60 } 61 61 62 63 62 return null; 63 } 64 64 65 66 67 68 65 public boolean isUsable(Object propertyValue, AccessParameters accessParameters) { 66 assert propertyValue instanceof Float; 67 return true; 68 } 69 69 70 70 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadMaxweight.java
r20243 r23189 6 6 7 7 public class RoadMaxweight extends RoadValueLimit { 8 9 10 11 12 13 14 8 public RoadMaxweight() { 9 super("maxweight", WEIGHT, LimitType.MAXIMUM); 10 } 11 @Override 12 protected Float parse(String valueString) { 13 return ValueStringParser.parseWeight(valueString); 14 } 15 15 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadMaxwidth.java
r20243 r23189 6 6 7 7 public class RoadMaxwidth extends RoadValueLimit { 8 9 10 11 12 13 14 8 public RoadMaxwidth() { 9 super("maxwidth", WIDTH, LimitType.MAXIMUM); 10 } 11 @Override 12 protected Float parse(String valueString) { 13 return ValueStringParser.parseMeasure(valueString); 14 } 15 15 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadMinspeed.java
r20243 r23189 6 6 7 7 public class RoadMinspeed extends RoadValueLimit { 8 9 10 11 12 13 14 8 public RoadMinspeed() { 9 super("minspeed", SPEED, LimitType.MINIMUM); 10 } 11 @Override 12 protected Float parse(String valueString) { 13 return ValueStringParser.parseSpeed(valueString); 14 } 15 15 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadPropertyType.java
r19216 r23189 14 14 public interface RoadPropertyType<V> { 15 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 16 /** 17 * determines the property value for way-based segments. 18 * Uses the way the segment is created from. 19 * 20 * @param way way that is to be evaluated; != null 21 * @param forward chooses whether the property is evaluated 22 * in (true) or against (false) way direction 23 * @param accessParameters access parameters for properties that depend on vehicle/situation 24 * @param dataSource object providing access to all data; != null 25 * @return value of this property for the way; 26 * null if property cannot be determined / does not apply 27 */ 28 public <N, W, R, M> V evaluateW(W way, boolean forward, 29 AccessParameters accessParameters, DataSource<N, W, R, M> dataSource); 30 30 31 32 33 34 35 36 37 38 39 40 41 42 31 /** 32 * determines the property value for node-based segments. 33 * Uses the node the segment is created from. 34 * 35 * @param way node that is to be evaluated; != null 36 * @param accessParameters access parameters for properties that depend on vehicle/situation 37 * @param dataSource object providing access to all data; != null 38 * @return value of this property for the way; 39 * null if property cannot be determined / does not apply 40 */ 41 public <N, W, R, M> V evaluateN(N node, 42 AccessParameters accessParameters, DataSource<N, W, R, M> dataSource); 43 43 44 45 46 47 48 49 50 51 44 /** 45 * checks whether a segment with a given value for this property can be used by a vehicle 46 * with a certain set of access parameters 47 * 48 * @param object value of this property for the segment; 49 * MUST be of type V; != null 50 */ 51 public boolean isUsable(Object object, AccessParameters accessParameters); 52 52 53 53 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadSurface.java
r19216 r23189 9 9 public class RoadSurface implements RoadPropertyType<String> { 10 10 11 12 13 14 11 public <N, W, R, M> String evaluateN(N node, AccessParameters accessParameters, 12 DataSource<N,W,R,M> dataSource) { 13 return null; 14 }; 15 15 16 17 18 16 public <N, W, R, M> String evaluateW(W way, boolean forward, AccessParameters accessParameters, 17 DataSource<N,W,R,M> dataSource) { 18 assert way != null && accessParameters != null && dataSource != null; 19 19 20 21 20 TagGroup tags = dataSource.getTagsW(way); 21 return tags.getValue("surface"); 22 22 23 23 }; 24 24 25 26 25 public boolean isUsable(Object propertyValue, AccessParameters accessParameters) { 26 assert propertyValue instanceof String; 27 27 28 28 String surface = (String)propertyValue; 29 29 30 31 30 Collection<String> surfaceBlacklist = 31 accessParameters.getVehiclePropertyValue(VehiclePropertyTypes.SURFACE_BLACKLIST); 32 32 33 34 35 36 37 38 33 if (surfaceBlacklist != null && surfaceBlacklist.contains(surface)) { 34 return false; 35 } else { 36 return true; 37 } 38 } 39 39 40 40 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadTracktype.java
r19216 r23189 7 7 public class RoadTracktype implements RoadPropertyType<Integer> { 8 8 9 10 11 12 9 public <N, W, R, M> Integer evaluateN(N node, AccessParameters accessParameters, 10 DataSource<N,W,R,M> dataSource) { 11 return null; 12 }; 13 13 14 15 16 14 public <N, W, R, M> Integer evaluateW(W way, boolean forward, AccessParameters accessParameters, 15 DataSource<N,W,R,M> dataSource) { 16 assert way != null && accessParameters != null && dataSource != null; 17 17 18 19 18 TagGroup tags = dataSource.getTagsW(way); 19 String tracktypeString = tags.getValue("tracktype"); 20 20 21 22 23 24 25 26 27 28 29 30 31 32 33 21 if (tracktypeString != null) { 22 if ("grade1".equals(tracktypeString)) { 23 return 1; 24 } else if ("grade2".equals(tracktypeString)) { 25 return 2; 26 } else if ("grade3".equals(tracktypeString)) { 27 return 3; 28 } else if ("grade4".equals(tracktypeString)) { 29 return 4; 30 } else if ("grade5".equals(tracktypeString)) { 31 return 5; 32 } 33 } 34 34 35 36 35 return null; 36 }; 37 37 38 39 38 public boolean isUsable(Object propertyValue, AccessParameters accessParameters) { 39 assert propertyValue instanceof Integer; 40 40 41 41 int tracktype = (Integer)propertyValue; 42 42 43 44 43 Integer maxTracktype = 44 accessParameters.getVehiclePropertyValue(VehiclePropertyTypes.MAX_TRACKTYPE); 45 45 46 47 48 49 50 51 46 if (maxTracktype != null && tracktype > maxTracktype) { 47 return false; 48 } else { 49 return true; 50 } 51 } 52 52 53 53 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadValueLimit.java
r20243 r23189 11 11 abstract public class RoadValueLimit implements RoadPropertyType<Float> { 12 12 13 13 protected static enum LimitType {MINIMUM, MAXIMUM}; 14 14 15 16 15 private final String keyName; 16 private final VehiclePropertyType<Float> vehicleProperty; 17 17 18 18 private final LimitType upperLimit; 19 19 20 21 22 23 24 25 26 27 28 29 20 /** 21 * @param keyName key that is used to add this property to a way; value must be in 22 * format readable by {@link ValueStringParser#parseOsmDecimal(String)}; 23 * != null 24 * @param vehicleProperty vehicle property that is limited by this road property; != null 25 * @param upperLimit type of limit; != null 26 */ 27 protected RoadValueLimit(String keyName, VehiclePropertyType<Float> vehicleProperty, 28 LimitType upperLimit) { 29 assert keyName != null && vehicleProperty != null && upperLimit != null; 30 30 31 32 33 34 31 this.keyName = keyName; 32 this.vehicleProperty = vehicleProperty; 33 this.upperLimit = upperLimit; 34 } 35 35 36 36 protected abstract Float parse(String valueString); 37 37 38 39 40 41 42 38 public <N, W, R, M> Float evaluateW(W way, boolean forward, 39 AccessParameters accessParameters, DataSource<N, W, R, M> dataSource) { 40 assert way != null && accessParameters != null && dataSource != null; 41 return evaluateTags(dataSource.getTagsW(way)); 42 } 43 43 44 45 46 47 48 44 public <N, W, R, M> Float evaluateN(N node, 45 AccessParameters accessParameters, DataSource<N, W, R, M> dataSource) { 46 assert node != null && accessParameters != null && dataSource != null; 47 return evaluateTags(dataSource.getTagsN(node)); 48 } 49 49 50 51 52 53 54 55 56 57 58 50 private final Float evaluateTags(TagGroup tags) { 51 String valueString = tags.getValue(keyName); 52 if (valueString != null) { 53 Float value = parse(valueString); 54 return value; 55 } else { 56 return null; 57 } 58 } 59 59 60 61 60 public boolean isUsable(Object propertyValue, AccessParameters accessParameters) { 61 assert propertyValue instanceof Float; 62 62 63 63 Float vehicleValue = accessParameters.getVehiclePropertyValue(vehicleProperty); 64 64 65 66 67 68 69 70 71 72 73 74 65 if (vehicleValue != null) { 66 switch(upperLimit) { 67 case MINIMUM: return vehicleValue >= (Float) propertyValue; 68 case MAXIMUM: return vehicleValue <= (Float) propertyValue; 69 default: throw new Error("unhandled LimitType"); 70 } 71 } else { 72 return true; 73 } 74 } 75 75 76 76 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/RoadWidth.java
r20243 r23189 6 6 7 7 public class RoadWidth extends RoadValueLimit { 8 9 10 11 12 13 14 8 public RoadWidth() { 9 super("width", WIDTH, LimitType.MAXIMUM); 10 } 11 @Override 12 protected Float parse(String valueString) { 13 return ValueStringParser.parseMeasure(valueString); 14 } 15 15 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/VehiclePropertyType.java
r16520 r23189 11 11 public interface VehiclePropertyType<V> { 12 12 13 14 15 16 17 13 /** 14 * determines whether a value is valid. 15 * null is never a valid value and must not be used as parameter. 16 */ 17 public boolean isValidValue(Object value); 18 18 19 19 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/property/VehiclePropertyTypes.java
r16520 r23189 9 9 public final class VehiclePropertyTypes { 10 10 11 12 11 /** prevents instantiation */ 12 private VehiclePropertyTypes() { } 13 13 14 15 16 17 18 14 private static final class NonnegativeFloatProperty implements VehiclePropertyType<Float> { 15 public boolean isValidValue(Object value) { 16 return value instanceof Float && (Float)value >= 0; 17 } 18 } 19 19 20 21 20 /** length of a vehicle in meters; negative values are invalid */ 21 public static final VehiclePropertyType<Float> LENGTH = new NonnegativeFloatProperty(); 22 22 23 24 23 /** width of a vehicle in meters; negative values are invalid */ 24 public static final VehiclePropertyType<Float> WIDTH = new NonnegativeFloatProperty(); 25 25 26 27 26 /** height of a vehicle in meters; negative values are invalid */ 27 public static final VehiclePropertyType<Float> HEIGHT = new NonnegativeFloatProperty(); 28 28 29 30 29 /** weight of a vehicle in tons; negative values are invalid */ 30 public static final VehiclePropertyType<Float> WEIGHT = new NonnegativeFloatProperty(); 31 31 32 33 32 /** axleload of a vehicle in tons; negative values are invalid */ 33 public static final VehiclePropertyType<Float> AXLELOAD = new NonnegativeFloatProperty(); 34 34 35 36 35 /** speed a vehicle can reach in km/h; negative values are invalid */ 36 public static final VehiclePropertyType<Float> SPEED = new NonnegativeFloatProperty(); 37 37 38 39 38 /** maximum incline a vehicle can go up; negative values are invalid */ 39 public static final VehiclePropertyType<Float> MAX_INCLINE_UP = new NonnegativeFloatProperty(); 40 40 41 42 41 /** maximum incline a vehicle can go down; negative values are invalid */ 42 public static final VehiclePropertyType<Float> MAX_INCLINE_DOWN = new NonnegativeFloatProperty(); 43 43 44 45 46 44 /** surface types ("surface" key values) the vehicle cannot use */ 45 public static final VehiclePropertyType<Collection<String>> SURFACE_BLACKLIST = new VehiclePropertyType<Collection<String>>() { 46 public boolean isValidValue(Object value) { 47 47 48 49 50 48 if (!(value instanceof Collection)) { 49 return false; 50 } 51 51 52 53 54 55 56 52 for (Object contentObject : (Collection<?>)value) { 53 if (!(contentObject instanceof String)) { 54 return false; 55 } 56 } 57 57 58 59 60 58 return true; 59 } 60 }; 61 61 62 63 64 65 66 67 68 69 70 71 62 /** 63 * maximum tracktype grade the vehicle can use; 64 * values are integers from = to 5 65 * (values of key "tracktype" without "grade_" prefix, 0 is for "none") 66 */ 67 public static final VehiclePropertyType<Integer> MAX_TRACKTYPE = new VehiclePropertyType<Integer>() { 68 public boolean isValidValue(Object value) { 69 return value instanceof Integer && (Integer)value >= 0 && (Integer)value <= 5; 70 } 71 }; 72 72 73 73 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/transition/GenericTransitionStructure.java
r19216 r23189 30 30 public class GenericTransitionStructure<N, W, R, M> implements TransitionStructure, DataSourceObserver { 31 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 32 private static final Collection<Segment> EMPTY_SEGMENT_LIST = 33 Collections.unmodifiableList(new ArrayList<Segment>(0)); 34 private static final Collection<Restriction> EMPTY_RESTRICTION_COLLECTION = 35 new ArrayList<Restriction>(0); 36 37 private static class SegmentNodeImpl implements SegmentNode { 38 private final double lat; 39 private final double lon; 40 private final List<Segment> inboundSegments = new LinkedList<Segment>(); 41 private final List<Segment> outboundSegments = new LinkedList<Segment>(); 42 private final Map<RoadPropertyType<?>, Object> properties; 43 public SegmentNodeImpl(double lat, double lon, Map<RoadPropertyType<?>, Object> properties) { 44 assert properties != null; 45 this.lat = lat; 46 this.lon = lon; 47 this.properties = properties; 48 } 49 public double getLat() { 50 return lat; 51 } 52 public double getLon() { 53 return lon; 54 } 55 public void addInboundSegment(Segment segment) { 56 inboundSegments.add(segment); 57 } 58 public void addOutboundSegment(Segment segment) { 59 outboundSegments.add(segment); 60 } 61 public Collection<Segment> getOutboundSegments() { 62 return outboundSegments; 63 } 64 public Collection<Segment> getInboundSegments() { 65 return inboundSegments; 66 } 67 68 public <P> void setProperty(RoadPropertyType<P> property, P value) { 69 properties.put(property, value); 70 } 71 public Collection<RoadPropertyType<?>> getAvailableProperties() { 72 return properties.keySet(); 73 } 74 public <P> P getPropertyValue(RoadPropertyType<P> property) { 75 @SuppressWarnings("unchecked") //cast is safe due to type parameter of setProperty 76 P result = (P) properties.get(property); 77 return result; 78 } 79 public Map<RoadPropertyType<?>, Object> getProperties() { 80 return properties; 81 } 82 83 @Override 84 public String toString() { 85 return "(" + lat + ", " + lon + ")"; 86 } 87 } 88 89 private static class SegmentImpl implements Segment { 90 private final SegmentNode node1; 91 private final SegmentNode node2; 92 private final Map<RoadPropertyType<?>, Object> properties; 93 public SegmentImpl(SegmentNode node1, SegmentNode node2, Map<RoadPropertyType<?>, Object> properties) { 94 this.node1 = node1; 95 this.node2 = node2; 96 this.properties = properties; 97 } 98 public SegmentNode getNode1() { 99 return node1; 100 } 101 public SegmentNode getNode2() { 102 return node2; 103 } 104 public <P> void setProperty(RoadPropertyType<P> property, P value) { 105 properties.put(property, value); 106 } 107 public Collection<RoadPropertyType<?>> getAvailableProperties() { 108 return properties.keySet(); 109 } 110 public <P> P getPropertyValue(RoadPropertyType<P> property) { 111 @SuppressWarnings("unchecked") //cast is safe due to type parameter of setProperty 112 P result = (P) properties.get(property); 113 return result; 114 } 115 116 @Override 117 public String toString() { 118 return "(" + node1 + "->" + node2 + ")"; 119 } 120 } 121 122 private static class RestrictionImpl implements Restriction { 123 private final Segment from; 124 private final Collection<Segment> vias; 125 private final Collection<Segment> tos; 126 127 /** constructor, will directly use collection references, collections must not be changed after usage as constructor param */ 128 public RestrictionImpl(Segment from, Collection<Segment> vias, Collection<Segment> tos) { 129 this.from = from; 130 this.vias = Collections.unmodifiableCollection(vias); 131 this.tos = Collections.unmodifiableCollection(tos); 132 } 133 134 public Segment getFrom() { 135 return from; 136 } 137 public Collection<Segment> getVias() { 138 return vias; 139 } 140 public Collection<Segment> getTos() { 141 return tos; 142 } 143 144 @Override 145 public String toString() { 146 return from + " -> " + vias + " -> " + tos; 147 } 148 } 149 150 private final Set<TransitionStructureObserver> observers = new HashSet<TransitionStructureObserver>(); 151 152 private final Collection<RoadPropertyType<?>> properties; 153 154 private final DataSource<N, W, R, M> dataSource; 155 156 private AccessParameters accessParameters; 157 private AccessRuleset ruleset; 158 159 private AccessEvaluator<N, W> accessEvaluator; 160 161 private Collection<SegmentNode> nodes = null; 162 private Collection<Segment> segments = new LinkedList<Segment>(); 163 private Collection<Restriction> restrictions = new LinkedList<Restriction>(); 164 165 public GenericTransitionStructure( 166 AccessParameters accessParameters, AccessRuleset ruleset, 167 DataSource<N, W, R, M> dataSource, 168 Collection<RoadPropertyType<?>> properties) { 169 170 assert accessParameters != null && ruleset != null; 171 assert dataSource != null; 172 assert properties != null; 173 174 this.dataSource = dataSource; 175 176 this.properties = properties; 177 178 setAccessParametersAndRuleset(accessParameters, ruleset); 179 180 dataSource.addObserver(this); 181 } 182 183 /** 184 * sets new access parameters and/or a new ruleset. 185 * Causes a data update if at least one is actually changed. 186 * 187 * @param accessParameters new access parameters, null indicates no change 188 * @param ruleset new ruleset, null indicates no change 189 */ 190 public void setAccessParametersAndRuleset(AccessParameters accessParameters, AccessRuleset ruleset) { 191 192 if (accessParameters != null) { 193 this.accessParameters = accessParameters; 194 } 195 if (ruleset != null) { 196 this.ruleset = ruleset; 197 } 198 199 if (accessParameters != null || ruleset != null) { 200 201 assert dataSource != null; 202 203 accessEvaluator = new RulesetAccessEvaluator<N, W, R, M>( 204 dataSource, 205 this.ruleset, 206 this.accessParameters); 207 208 updateData(); 209 notifyObservers(); 210 211 } 212 213 } 214 215 public Collection<SegmentNode> getNodes() { 216 return nodes; 217 } 218 219 public Collection<Segment> getSegments() { 220 return segments; 221 } 222 223 public Collection<Restriction> getRestrictions() { 224 return restrictions; 225 } 226 227 /** 228 * creates nodes, segments and restrictions based on the data source 229 */ 230 protected void updateData() { 231 232 ArrayList<SegmentNode> nodes = new ArrayList<SegmentNode>(); 233 ArrayList<Segment> segments = new ArrayList<Segment>(); 234 235 Map<N, SegmentNodeImpl> nodeCreationMap = new HashMap<N, SegmentNodeImpl>(); 236 Map<W, List<Segment>> waySegmentMap = new HashMap<W, List<Segment>>(); 237 238 /* create segments (nodes are created only when included in a segment) */ 239 240 for (W way : dataSource.getWays()) { 241 createSegmentsAndSegmentNodes(way, accessEvaluator, nodes, segments, nodeCreationMap, waySegmentMap); 242 } 243 244 nodes.trimToSize(); 245 segments.trimToSize(); 246 247 /* create restrictions */ 248 249 Collection<Restriction> restrictions = 250 createRestrictionsFromTurnRestrictions(dataSource.getRelations(), nodeCreationMap, waySegmentMap); 251 252 restrictions.addAll(createRestrictionsFromBarrierNodes(nodeCreationMap, waySegmentMap)); 253 254 /* keep data and inform observers */ 255 256 this.nodes = nodes; 257 this.segments = segments; 258 this.restrictions = restrictions; 259 260 notifyObservers(); 261 262 } 263 264 /** 265 * creates all Segments and SegmentNodes for a way 266 * 267 * @param way way to create Segments and SegmentNodes from; != null 268 * @param wayAccessEvaluator evaluator object that decides whether way is usable; != null 269 * @param nodes collection of SegmentNodes, new SegmentNodes will be added here; != null 270 * @param segments collection of Segments, new Segments will be added here; != null 271 * @param nodeCreationMap map providing the SegmentNode that has been created from a Node, 272 * if new SegmentNodes are created, they will be added appropriately; != null 273 * @param waySegmentMap map providing the Segments that have been created from a Way, 274 * if new Segments are created, they will be added appropriately; != null 275 */ 276 private void createSegmentsAndSegmentNodes(W way, AccessEvaluator<N, W> wayAccessEvaluator, 277 Collection<SegmentNode> nodes, Collection<Segment> segments, 278 Map<N, SegmentNodeImpl> nodeCreationMap, Map<W, List<Segment>> waySegmentMap) { 279 280 assert way != null && wayAccessEvaluator != null && nodes != null && segments != null && nodeCreationMap != null && waySegmentMap != null; 281 282 /* calculate property values */ 283 284 Map<RoadPropertyType<?>, Object> forwardPropertyValues = getWayPropertyMap(way, true); 285 Map<RoadPropertyType<?>, Object> backwardPropertyValues = getWayPropertyMap(way, false); 286 287 /* create segments from the way if it can be accessed and isn't incomplete or deleted */ 288 289 boolean forwardAccess = wayAccessEvaluator.wayUsable(way, true, forwardPropertyValues); 290 boolean backwardAccess = wayAccessEvaluator.wayUsable(way, false, backwardPropertyValues); 291 292 if (forwardAccess || backwardAccess) { 293 294 if (!waySegmentMap.containsKey(way)) { 295 waySegmentMap.put(way, new LinkedList<Segment>()); 296 } 297 298 /* create segments from all pairs of subsequent nodes */ 299 300 N previousNode = null; 301 for (N node : dataSource.getNodes(way)) { 302 if (previousNode != null) { 303 304 SegmentNodeImpl node1 = 305 getOrCreateSegmentNodeForNode(previousNode, nodes, nodeCreationMap); 306 SegmentNodeImpl node2 = 307 getOrCreateSegmentNodeForNode(node, nodes, nodeCreationMap); 308 309 if (forwardAccess) { 310 SegmentImpl segment = new SegmentImpl(node1, node2, forwardPropertyValues); 311 segments.add(segment); 312 waySegmentMap.get(way).add(segment); 313 node1.addOutboundSegment(segment); 314 node2.addInboundSegment(segment); 315 } 316 if (backwardAccess) { //no "else if" because both can be valid 317 SegmentImpl segment = new SegmentImpl(node2, node1, backwardPropertyValues); 318 segments.add(segment); 319 waySegmentMap.get(way).add(segment); 320 node1.addInboundSegment(segment); 321 node2.addOutboundSegment(segment); 322 } 323 324 } 325 previousNode = node; 326 } 327 328 } 329 } 330 331 /** 332 * if no segment node for a node exists in the nodeCreationMap, 333 * creates a segment node for it and adds it to the nodeCreationMap and the nodes collection 334 * and returns it; otherwise returns the existing segment node. 335 */ 336 private SegmentNodeImpl getOrCreateSegmentNodeForNode(N node, 337 Collection<SegmentNode> nodes, Map<N, SegmentNodeImpl> nodeCreationMap) { 338 339 SegmentNodeImpl segmentNode = nodeCreationMap.get(node); 340 341 if (segmentNode == null) { 342 343 Map<RoadPropertyType<?>, Object> nodePropertyValues = getNodePropertyMap(node); 344 segmentNode = new SegmentNodeImpl(dataSource.getLat(node), dataSource.getLon(node), 345 nodePropertyValues); 346 347 nodeCreationMap.put(node, segmentNode); 348 nodes.add(segmentNode); 349 350 } 351 352 return segmentNode; 353 } 354 355 /** 356 * creates all Restrictions from a collection of Relations. 357 * Only "type=restriction" relations are relevant for restrictions. 358 * 359 * @param relations Relations to create Restrictions from. 360 * They can have any type key, as filtering is done inside this method. 361 * @param nodeCreationMap map providing the SegmentNode that has been created from a Node, 362 * will not be modified; != null 363 * @param waySegmentMap map providing the Segments that have been created from a Way, 364 * will not be modified; != null 365 * @return Restrictions created from the Relations; != null, but may be empty 366 */ 367 private Collection<Restriction> createRestrictionsFromTurnRestrictions( 368 Iterable<R> relations, 369 Map<N, SegmentNodeImpl> nodeCreationMap, 370 Map<W, List<Segment>> waySegmentMap) { 371 372 assert relations != null && nodeCreationMap != null && waySegmentMap != null; 373 374 Collection<Restriction> results = new LinkedList<Restriction>(); 375 376 for (R relation : relations) { 377 378 TagGroup tags = dataSource.getTagsR(relation); 379 380 if ("restriction".equals(tags.getValue("type")) 381 && tags.getValue("restriction") != null ) { 382 383 //evaluate relation 384 if (tags.getValue("restriction").startsWith("no_")) { 385 results.addAll(createRestrictionsFromRestrictionRelation(relation, true, nodeCreationMap, waySegmentMap)); 386 } else if (tags.getValue("restriction").startsWith("only_")) { 387 results.addAll(createRestrictionsFromRestrictionRelation(relation, false, nodeCreationMap, waySegmentMap)); 388 } 389 390 } 391 } 392 393 return results; 394 } 395 396 @SuppressWarnings("unchecked") //several generic casts that are checked with isInstance 397 private Collection<Restriction> createRestrictionsFromRestrictionRelation( 398 R relation, 399 boolean restrictive, 400 Map<N, SegmentNodeImpl> nodeCreationMap, 401 Map<W, List<Segment>> waySegmentMap) { 402 403 /* collect information about the relation */ 404 405 W fromWay = null; 406 Collection<N> viaNodes = new LinkedList<N>(); 407 Collection<W> viaWays = new LinkedList<W>(); 408 Collection<W> toWays = new LinkedList<W>(); 409 410 for (M member : dataSource.getMembers(relation)) { 411 412 if ("from".equals(dataSource.getRole(member))) { 413 if (fromWay != null || !dataSource.isWMember(member)) { 414 //broken restriction 415 return EMPTY_RESTRICTION_COLLECTION; 416 } else { 417 fromWay = (W)dataSource.getMember(member); 418 } 419 } else if ("to".equals(dataSource.getRole(member))) { 420 if (!dataSource.isWMember(member)) { 421 //broken restriction 422 return EMPTY_RESTRICTION_COLLECTION; 423 } else { 424 toWays.add((W)dataSource.getMember(member)); 425 } 426 } else if ("via".equals(dataSource.getRole(member))) { 427 if (dataSource.isWMember(member)) { 428 viaWays.add((W)dataSource.getMember(member)); 429 } else if (dataSource.isNMember(member)) { 430 viaNodes.add((N)dataSource.getMember(member)); 431 } 432 } 433 434 } 435 436 if (fromWay != null && toWays.size() > 0 && 437 (viaNodes.size() > 0 || viaWays.size() > 0)) { 438 439 return createRestrictionsFromRestrictionRelationMembers( 440 restrictive, nodeCreationMap, waySegmentMap, 441 fromWay, viaNodes, viaWays, toWays); 442 443 } else { 444 return new ArrayList<Restriction>(0); 445 } 446 } 447 448 private Collection<Restriction> createRestrictionsFromRestrictionRelationMembers( 449 boolean restrictive, 450 Map<N, SegmentNodeImpl> nodeCreationMap, Map<W, List<Segment>> waySegmentMap, 451 W fromWay, Collection<N> viaNodes, Collection<W> viaWays, Collection<W> toWays) { 452 453 Collection<SegmentNode> nodesCreatedFromViaNodes = new ArrayList<SegmentNode>(viaNodes.size()); 454 for (N viaNode : viaNodes) { 455 if (nodeCreationMap.containsKey(viaNode)) { 456 nodesCreatedFromViaNodes.add(nodeCreationMap.get(viaNode)); 457 } 458 } 459 460 /* check completeness of restriction to avoid dealing with incomplete restriction info */ 461 462 if (!waySegmentMap.containsKey(fromWay)) { 463 //broken restriction 464 return EMPTY_RESTRICTION_COLLECTION; 465 } 466 467 for (W viaWay : viaWays) { 468 if (!waySegmentMap.containsKey(viaWay)) { 469 //broken restriction 470 return EMPTY_RESTRICTION_COLLECTION; 471 } 472 } 473 474 for (W toWay : toWays) { 475 if (!waySegmentMap.containsKey(toWay)) { 476 //broken restriction 477 return EMPTY_RESTRICTION_COLLECTION; 478 } 479 } 480 481 /* find all via segments: 482 * via segments are segments created from via ways 483 * or segments starting and ending with nodes created from via nodes */ 484 485 ArrayList<Segment> viaSegments = new ArrayList<Segment>(); 486 487 for (W viaWay : viaWays) { 488 viaSegments.addAll(waySegmentMap.get(viaWay)); 489 } 490 491 for (SegmentNode nodeCreatedFromViaNode : nodesCreatedFromViaNodes) { 492 for (Segment segment : nodeCreatedFromViaNode.getOutboundSegments()) { 493 if (nodesCreatedFromViaNodes.contains(segment.getNode2())) { 494 viaSegments.add(segment); 495 } 496 } 497 } 498 499 viaSegments.trimToSize(); 500 501 /* create a set with all nodes that are based on via members */ 502 503 Set<SegmentNode> nodesCreatedFromViaMembers 504 = new HashSet<SegmentNode>(nodesCreatedFromViaNodes); 505 506 for (W viaWay : viaWays) { 507 for (N viaWayNode : dataSource.getNodes(viaWay)) { 508 nodesCreatedFromViaMembers.add(nodeCreationMap.get(viaWayNode)); 509 } 510 } 511 512 /* 513 * find from segment and to segments: 514 * Such a segment contains a node based on a via member. 515 * Each way should contain only one possible segment 516 * connecting to via members (due to splitting). 517 */ 518 519 Segment fromSegment = null; 520 Collection<Segment> toSegments = new ArrayList<Segment>(); 521 522 for (Segment possibleFromSegment : waySegmentMap.get(fromWay)) { 523 if (nodesCreatedFromViaMembers.contains(possibleFromSegment.getNode2())) { 524 525 if (fromSegment == null) { 526 fromSegment = possibleFromSegment; 527 } else { 528 //broken restriction 529 return EMPTY_RESTRICTION_COLLECTION; 530 } 531 532 } 533 } 534 if (fromSegment == null) { 535 //broken restriction 536 return EMPTY_RESTRICTION_COLLECTION; 537 } 538 539 if (restrictive) { 540 541 for (W toWay : toWays) { 542 if (waySegmentMap.containsKey(toWay)) { 543 Segment toSegment = null; 544 for (Segment possibleToSegment : waySegmentMap.get(toWay)) { 545 if (nodesCreatedFromViaMembers.contains(possibleToSegment.getNode1())) { 546 547 if (toSegment == null) { 548 toSegment = possibleToSegment; 549 } else { 550 //broken restriction 551 return EMPTY_RESTRICTION_COLLECTION; 552 } 553 554 } 555 } 556 if (toSegment == null) { 557 //broken restriction 558 return EMPTY_RESTRICTION_COLLECTION; 559 } else { 560 toSegments.add(toSegment); 561 } 562 } 563 } 564 565 } else { //!restrictive 566 567 /* forbidden "to" segments are all segments that start at a "via" node 568 * and are neither a via segment nor created from an allowed "to" way */ 569 570 for (SegmentNode toStartingNode : nodesCreatedFromViaMembers) { 571 for (Segment outboundSegment : toStartingNode.getOutboundSegments()) { 572 573 if (!viaSegments.contains(outboundSegment)) { 574 575 boolean isAllowed = false; 576 577 for (W toWay : toWays) { 578 if (waySegmentMap.get(toWay).contains(outboundSegment)) { 579 isAllowed = true; 580 break; 581 } 582 } 583 584 if (!isAllowed) { 585 toSegments.add(outboundSegment); 586 } 587 588 } 589 590 } 591 } 592 593 } 594 595 /* create restriction */ 596 597 Collection<Restriction> results = new ArrayList<Restriction>(1); 598 results.add(new RestrictionImpl(fromSegment, viaSegments, toSegments)); 599 return results; 600 } 601 602 /** 603 * creates Restrictions from barrier nodes (nodes that are considered impassable by the 604 * {@link #accessEvaluator}). These restrictions prevent moving from a segment before the 605 * barrier node to a segment after the barrier node. 606 * 607 * @param nodeCreationMap map providing the SegmentNode that has been created from a node, 608 * will not be modified; != null 609 * @param waySegmentMap map providing the Segments that have been created from a way, 610 * will not be modified; != null 611 * @return Restrictions created from barrier nodes; != null, but may be empty 612 */ 613 private Collection<Restriction> createRestrictionsFromBarrierNodes( 614 Map<N, SegmentNodeImpl> nodeCreationMap, 615 Map<W, List<Segment>> waySegmentMap) { 616 617 assert nodeCreationMap != null; 618 assert waySegmentMap != null; 619 620 Collection<Restriction> results = new LinkedList<Restriction>(); 621 622 for (N node : nodeCreationMap.keySet()) { 623 624 if (!accessEvaluator.nodeUsable(node, nodeCreationMap.get(node).getProperties())) { 625 626 SegmentNode barrierNode = nodeCreationMap.get(node); 627 628 for (Segment inboundSegment : barrierNode.getInboundSegments()) { 629 for (Segment outboundSegment : barrierNode.getOutboundSegments()) { 630 results.add(new RestrictionImpl(inboundSegment, EMPTY_SEGMENT_LIST, Arrays.asList(outboundSegment))); 631 } 632 } 633 634 } 635 636 } 637 638 return results; 639 } 640 641 /** 642 * determines the values of all RoadPropertyTypes from {@link #properties} for a way and 643 * creates a map with the types that have non-null values as keys, property values as content 644 */ 645 private Map<RoadPropertyType<?>, Object> getWayPropertyMap(W way, boolean forward) { 646 Map<RoadPropertyType<?>, Object> propertyValues; 647 propertyValues = new HashMap<RoadPropertyType<?>, Object>(); 648 for (RoadPropertyType<?> property : properties) { 649 Object value = property.evaluateW(way, forward, accessParameters, dataSource); 650 if (value != null) { 651 propertyValues.put(property, value); 652 } 653 } 654 return propertyValues; 655 } 656 657 /** 658 * determines the values of all RoadPropertyTypes from {@link #properties} for a node and 659 * creates a map with the types that have non-null values as keys, property values as content 660 */ 661 private Map<RoadPropertyType<?>, Object> getNodePropertyMap(N node) { 662 Map<RoadPropertyType<?>, Object> propertyValues; 663 propertyValues = new HashMap<RoadPropertyType<?>, Object>(); 664 for (RoadPropertyType<?> property : properties) { 665 Object value = property.evaluateN(node, accessParameters, dataSource); 666 if (value != null) { 667 propertyValues.put(property, value); 668 } 669 } 670 return propertyValues; 671 } 672 673 public void update(DataSource<?, ?, ?, ?> dataSource) { 674 assert this.dataSource == dataSource; 675 updateData(); 676 } 677 678 public void addObserver(TransitionStructureObserver observer) { 679 observers.add(observer); 680 } 681 682 public void deleteObserver(TransitionStructureObserver observer) { 683 observers.remove(observer); 684 } 685 686 protected void notifyObservers() { 687 for (TransitionStructureObserver observer : observers) { 688 observer.update(this); 689 } 690 } 691 691 692 692 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/transition/Restriction.java
r16520 r23189 10 10 public interface Restriction { 11 11 12 13 14 15 16 12 /** 13 * returns the starting segment that will trigger the restriction when used; 14 * != null 15 */ 16 public Segment getFrom(); 17 17 18 19 20 21 22 23 24 18 /** 19 * returns the "via" segments. 20 * The restriction will remain active as long as only via segments are used. 21 * 22 * @return unmodifiable collection of segments; != null 23 */ 24 public Collection<Segment> getVias(); 25 25 26 27 28 29 30 31 32 26 /** 27 * returns the forbidden "to" segments. 28 * The restriction prevents leaving the via segment set by using one of the to segments. 29 * 30 * @return unmodifiable collection of segments; != null 31 */ 32 public Collection<Segment> getTos(); 33 33 34 34 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/transition/Segment.java
r16520 r23189 6 6 public interface Segment extends TransitionStructureElement { 7 7 8 9 10 11 8 /** 9 * returns the node this segment starts at; != null 10 */ 11 public SegmentNode getNode1(); 12 12 13 14 15 16 13 /** 14 * returns the node this segment leads to; != null 15 */ 16 public SegmentNode getNode2(); 17 17 18 18 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/transition/SegmentNode.java
r16520 r23189 8 8 public interface SegmentNode extends TransitionStructureElement { 9 9 10 11 10 /** returns the node's latitude */ 11 public double getLat(); 12 12 13 14 13 /** returns the node's longitude */ 14 public double getLon(); 15 15 16 17 16 /** returns all segments that end at this node */ 17 Collection<Segment> getInboundSegments(); 18 18 19 20 19 /** returns all segments that start at this node */ 20 Collection<Segment> getOutboundSegments(); 21 21 22 22 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/transition/TransitionStructure.java
r16520 r23189 9 9 public interface TransitionStructure { 10 10 11 12 13 11 public Collection<SegmentNode> getNodes(); 12 public Collection<Segment> getSegments(); 13 public Collection<Restriction> getRestrictions(); 14 14 15 16 17 18 19 20 21 15 /** 16 * adds an observer. 17 * Does nothing if the parameter is already an observer of this TransitionStructure. 18 * 19 * @param observer observer object, != null 20 */ 21 public void addObserver(TransitionStructureObserver observer); 22 22 23 24 25 26 27 28 29 23 /** 24 * deletes an observer that has been added using {@link #addObserver(TransitionStructureObserver)}. 25 * Does nothing if the parameter isn't currently an observer of this TransitionStructure. 26 * 27 * @param observer observer object, != null 28 */ 29 public void deleteObserver(TransitionStructureObserver observer); 30 30 31 31 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/transition/TransitionStructureElement.java
r16520 r23189 10 10 public interface TransitionStructureElement { 11 11 12 13 14 15 16 17 12 /** 13 * returns the types of this object's properties 14 * 15 * @return property type collection; != null 16 */ 17 public Collection<RoadPropertyType<?>> getAvailableProperties(); 18 18 19 20 21 22 23 24 25 19 /** 20 * returns the value of a property for this object 21 * 22 * @param propertyType property type to return value for; != null 23 * @return property value of the property for this segment; null if not available 24 */ 25 public <P> P getPropertyValue(RoadPropertyType<P> propertyType); 26 26 27 27 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/transition/TransitionStructureObserver.java
r16520 r23189 7 7 public interface TransitionStructureObserver { 8 8 9 10 11 12 13 9 /** 10 * informs this observer about changes in an observed transition structure 11 * @param transitionStructure observed transition structure that has changed; != null 12 */ 13 public void update(TransitionStructure transitionStructure); 14 14 15 15 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/util/GraphUtil.java
r16520 r23189 11 11 public final class GraphUtil { 12 12 13 14 13 /** prevents instantiation */ 14 private GraphUtil() { } 15 15 16 17 18 19 20 16 /** 17 * checks whether a node is an "end node" 18 * (a node whose {@link SegmentNode} is connected to at most one other {@link SegmentNode}) 19 */ 20 public static final boolean isEndNode(GraphNode node) { 21 21 22 22 SegmentNode ownSegmentNode = node.getSegmentNode(); 23 23 24 24 SegmentNode connectedNode = null; 25 25 26 27 28 29 30 31 32 33 34 35 26 for (Segment inboundSegment : node.getSegmentNode().getInboundSegments()) { 27 SegmentNode otherSegmentNode = inboundSegment.getNode1(); 28 if (otherSegmentNode != ownSegmentNode) { 29 if (connectedNode == null) { 30 connectedNode = otherSegmentNode; 31 } else if (connectedNode != otherSegmentNode) { 32 return false; 33 } 34 } 35 } 36 36 37 38 39 40 41 42 43 44 45 46 37 for (Segment outboundSegment : node.getSegmentNode().getOutboundSegments()) { 38 SegmentNode otherSegmentNode = outboundSegment.getNode2(); 39 if (otherSegmentNode != ownSegmentNode) { 40 if (connectedNode == null) { 41 connectedNode = otherSegmentNode; 42 } else if (connectedNode != otherSegmentNode) { 43 return false; 44 } 45 } 46 } 47 47 48 48 return true; 49 49 50 50 } 51 51 52 52 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/util/TagCondition.java
r16520 r23189 9 9 public interface TagCondition { 10 10 11 12 13 14 15 16 11 /** 12 * returns true if the tags match the condition 13 * 14 * @param tags tags to check against the condition; != null 15 */ 16 boolean matches(TagGroup tags); 17 17 18 18 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/util/TagConditionLogic.java
r16520 r23189 11 11 public final class TagConditionLogic { 12 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 13 /** prevents instantiation */ 14 private TagConditionLogic(){ } 15 16 /** 17 * creates a condition that is fulfilled if the set of tags contains a given tag 18 * 19 * @param tag tag that must be in the tag collection; != null 20 */ 21 public static TagCondition tag(final Tag tag) { 22 assert tag != null; 23 return new TagCondition() { 24 public boolean matches(TagGroup tags) { 25 return tags.contains(tag); 26 } 27 @Override 28 public String toString() { 29 return tag.toString(); 30 } 31 }; 32 } 33 34 /** 35 * creates a condition that is fulfilled if the set of tags contains a tag with the given key 36 * 37 * @param key the key to look for; != null 38 */ 39 public static TagCondition key(final String key) { 40 assert key != null; 41 return new TagCondition() { 42 public boolean matches(TagGroup tags) { 43 return tags.containsKey(key); 44 } 45 @Override 46 public String toString() { 47 return key; 48 } 49 }; 50 } 51 52 /** 53 * combines conditions using a boolean "and" 54 * 55 * @param condition first condition; != null 56 * @param conditions more conditions; each != null 57 */ 58 public static TagCondition and(final TagCondition condition, final TagCondition... conditions) { 59 return new TagCondition() { 60 public boolean matches(TagGroup tags) { 61 for (TagCondition c : conditions) { 62 if (!c.matches(tags)) { 63 return false; 64 } 65 } 66 return condition.matches(tags); 67 } 68 @Override 69 public String toString() { 70 StringBuilder result = new StringBuilder(); 71 result.append("("); 72 result.append(condition); 73 for (TagCondition c : conditions) { 74 result.append(" && "); 75 result.append(c); 76 } 77 result.append(")"); 78 return result.toString(); 79 } 80 }; 81 } 82 83 /** 84 * combines conditions using a boolean "and" 85 * 86 * @param conditions collection of conditions, must contain at least one element; != null 87 */ 88 public static TagCondition and(final Collection<TagCondition> conditions) { 89 if (conditions.size() == 0) { 90 throw new IllegalArgumentException("collection must contain at least one condition"); 91 } 92 return new TagCondition() { 93 public boolean matches(TagGroup tags) { 94 for (TagCondition c : conditions) { 95 if (!c.matches(tags)) { 96 return false; 97 } 98 } 99 return true; 100 } 101 @Override 102 public String toString() { 103 StringBuilder result = new StringBuilder(); 104 result.append("("); 105 boolean firstCondition = true; 106 for (TagCondition c : conditions) { 107 if (!firstCondition) { 108 result.append(" && "); 109 } 110 firstCondition = false; 111 result.append(c); 112 } 113 result.append(")"); 114 return result.toString(); 115 } 116 }; 117 } 118 119 /** 120 * combines conditions using a boolean "or" 121 * 122 * @param condition first condition; != null 123 * @param conditions more conditions; each != null 124 */ 125 public static TagCondition or(final TagCondition condition, final TagCondition... conditions) { 126 return new TagCondition() { 127 public boolean matches(TagGroup tags) { 128 for (TagCondition c : conditions) { 129 if (c.matches(tags)) { 130 return true; 131 } 132 } 133 return condition.matches(tags); 134 } 135 @Override 136 public String toString() { 137 StringBuilder result = new StringBuilder(); 138 result.append("("); 139 result.append(condition); 140 for (TagCondition c : conditions) { 141 result.append(" || "); 142 result.append(c); 143 } 144 result.append(")"); 145 return result.toString(); 146 } 147 }; 148 } 149 150 /** 151 * combines conditions using a boolean "or" 152 * 153 * @param conditions collection of conditions, must contain at least one element; != null 154 */ 155 public static TagCondition or(final Collection<TagCondition> conditions) { 156 if (conditions.size() == 0) { 157 throw new IllegalArgumentException("collection must contain at least one condition"); 158 } 159 return new TagCondition() { 160 public boolean matches(TagGroup tags) { 161 for (TagCondition c : conditions) { 162 if (c.matches(tags)) { 163 return true; 164 } 165 } 166 return false; 167 } 168 @Override 169 public String toString() { 170 StringBuilder result = new StringBuilder(); 171 result.append("("); 172 boolean firstCondition = true; 173 for (TagCondition c : conditions) { 174 if (!firstCondition) { 175 result.append(" || "); 176 } 177 firstCondition = false; 178 result.append(c); 179 } 180 result.append(")"); 181 return result.toString(); 182 } 183 }; 184 } 185 186 /** 187 * inverts a condition 188 * 189 * @param condition condition to invert, != null 190 */ 191 public static TagCondition not(final TagCondition condition) { 192 return new TagCondition() { 193 public boolean matches(TagGroup tags) { 194 return !condition.matches(tags); 195 } 196 @Override 197 public String toString() { 198 return "!" + condition; 199 } 200 }; 201 } 202 202 203 203 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/util/ValueStringParser.java
r20243 r23189 6 6 public final class ValueStringParser { 7 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 8 /** prevents instantiation */ 9 private ValueStringParser() { } 10 11 /** pattern that splits into a part before and after a decimal point */ 12 private static final Pattern DEC_POINT_PATTERN = Pattern.compile("^(\\-?\\d+)\\.(\\d+)$"); 13 14 public static final Float parseOsmDecimal(String value, boolean allowNegative) { 15 16 /* positive integer */ 17 18 try { 19 20 int weight = Integer.parseInt(value); 21 if (weight >= 0 || allowNegative) { 22 return (float)weight; 23 } 24 25 } catch (NumberFormatException nfe) {} 26 27 /* positive number with decimal point */ 28 29 Matcher matcher = DEC_POINT_PATTERN.matcher(value); 30 31 if (matcher.matches()) { 32 33 String stringBeforePoint = matcher.group(1); 34 String stringAfterPoint = matcher.group(2); 35 36 if (stringBeforePoint.length() > 0 || stringAfterPoint.length() > 0) { 37 38 try { 39 40 float beforePoint = Integer.parseInt(stringBeforePoint); 41 float afterPoint = Integer.parseInt(stringAfterPoint); 42 43 double result = Math.signum(beforePoint) * 44 (Math.abs(beforePoint) 45 + Math.pow(10, -stringAfterPoint.length()) * afterPoint); 46 47 if (result >= 0 || allowNegative) { 48 return (float)result; 49 } 50 51 } catch (NumberFormatException nfe) {} 52 53 } 54 } 55 56 return null; 57 } 58 59 private static final Pattern KMH_PATTERN = Pattern.compile("^(\\d+)\\s*km/h$"); 60 private static final Pattern MPH_PATTERN = Pattern.compile("^(\\d+)\\s*mph$"); 61 62 private static final float KM_PER_MILE = 1.609344f; 63 64 /** 65 * parses a speed value given e.g. for the "maxspeed" key. 66 * 67 * @return speed in km/h; null if value had syntax errors 68 */ 69 public static final Float parseSpeed(String value) { 70 71 /* try numeric speed (implied km/h) */ 72 73 Float speed = parseOsmDecimal(value, false); 74 if (speed != null) { 75 return speed; 76 } 77 78 /* try km/h speed */ 79 80 Matcher kmhMatcher = KMH_PATTERN.matcher(value); 81 if (kmhMatcher.matches()) { 82 String kmhString = kmhMatcher.group(1); 83 try { 84 return (float)Integer.parseInt(kmhString); 85 } catch (NumberFormatException nfe) {} 86 } 87 88 /* try mph speed */ 89 90 Matcher mphMatcher = MPH_PATTERN.matcher(value); 91 if (mphMatcher.matches()) { 92 String mphString = mphMatcher.group(1); 93 try { 94 int mph = Integer.parseInt(mphString); 95 return KM_PER_MILE * mph; 96 } catch (NumberFormatException nfe) {} 97 } 98 99 /* all possibilities failed */ 100 101 return null; 102 } 103 104 private static final Pattern M_PATTERN = Pattern.compile("^([\\d\\.]+)\\s*m$"); 105 private static final Pattern KM_PATTERN = Pattern.compile("^([\\d\\.]+)\\s*km$"); 106 private static final Pattern MI_PATTERN = Pattern.compile("^([\\d\\.]+)\\s*mi$"); 107 private static final Pattern FEET_INCHES_PATTERN = Pattern.compile("^([\\d]+)'\\s*([\\d]+)\""); 108 109 private static final double M_PER_MI = 1609.344; 110 private static final double M_PER_INCH = 0.0254f; 111 112 /** 113 * parses a measure value given e.g. for the "width" or "length" key. 114 * 115 * @return measure in m; null if value had syntax errors 116 */ 117 public static final Float parseMeasure(String value) { 118 119 /* try numeric measure (implied m) */ 120 121 Float measure = parseOsmDecimal(value, false); 122 if (measure != null) { 123 return measure; 124 } 125 126 /* try m measure */ 127 128 Matcher mMatcher = M_PATTERN.matcher(value); 129 if (mMatcher.matches()) { 130 String mString = mMatcher.group(1); 131 return parseOsmDecimal(mString, false); 132 } 133 134 /* try km measure */ 135 136 Matcher kmMatcher = KM_PATTERN.matcher(value); 137 if (kmMatcher.matches()) { 138 String kmString = kmMatcher.group(1); 139 float km = parseOsmDecimal(kmString, false); 140 return 1000 * km; 141 } 142 143 /* try mi measure */ 144 145 Matcher miMatcher = MI_PATTERN.matcher(value); 146 if (miMatcher.matches()) { 147 String miString = miMatcher.group(1); 148 float mi = parseOsmDecimal(miString, false); 149 return (float)(M_PER_MI * mi); 150 } 151 152 /* try feet/inches measure */ 153 154 Matcher feetInchesMatcher = FEET_INCHES_PATTERN.matcher(value); 155 if (feetInchesMatcher.matches()) { 156 String feetString = feetInchesMatcher.group(1); 157 String inchesString = feetInchesMatcher.group(2); 158 try { 159 int feet = Integer.parseInt(feetString); 160 int inches = Integer.parseInt(inchesString); 161 if (feet >= 0 && inches >= 0 && inches < 12) { 162 return (float)(M_PER_INCH * (12 * feet + inches)); 163 } 164 } catch (NumberFormatException nfe) {} 165 } 166 167 /* all possibilities failed */ 168 169 return null; 170 } 171 172 private static final Pattern T_PATTERN = Pattern.compile("^([\\d\\.]+)\\s*t$"); 173 174 /** 175 * parses a weight value given e.g. for the "maxweight" or "maxaxleload" key. 176 * 177 * @return weight in t; null if value had syntax errors 178 */ 179 public static Float parseWeight(String value) { 180 181 /* try numeric weight (implied t) */ 182 183 Float weight = parseOsmDecimal(value, false); 184 if (weight != null) { 185 return weight; 186 } 187 188 /* try t weight */ 189 190 Matcher tMatcher = T_PATTERN.matcher(value); 191 if (tMatcher.matches()) { 192 String tString = tMatcher.group(1); 193 return parseOsmDecimal(tString, false); 194 } 195 196 /* all possibilities failed */ 197 198 return null; 199 200 } 201 202 private static final Pattern INCLINE_PATTERN = Pattern.compile("^(\\-?\\d+(?:\\.\\d+)?)\\s*%$"); 203 204 /** 205 * parses an incline value as given for the "incline" key. 206 * 207 * @return incline in percents; null if value had syntax errors 208 */ 209 public static final Float parseIncline(String value) { 210 211 Matcher inclineMatcher = INCLINE_PATTERN.matcher(value); 212 if (inclineMatcher.matches()) { 213 String inclineString = inclineMatcher.group(1); 214 return parseOsmDecimal(inclineString, true); 215 } 216 217 return null; 218 } 219 219 220 220 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/ColorScheme.java
r16520 r23189 11 11 public interface ColorScheme { 12 12 13 14 15 16 17 13 /** 14 * returns the color to be used for a node in a WayGraph 15 * @param edge GraphNode to determine the color for; != null 16 */ 17 public Color getNodeColor(GraphNode node); 18 18 19 20 21 22 23 19 /** 20 * returns the color to be used for an edge in a WayGraph 21 * @param segment segment to determine the color for; != null 22 */ 23 public Color getSegmentColor(Segment segment); 24 24 25 25 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/DefaultNodePositioner.java
r16520 r23189 11 11 public class DefaultNodePositioner implements NodePositioner { 12 12 13 13 public LatLonCoords getPosition(GraphNode node) { 14 14 15 15 SegmentNode segmentNode = node.getSegmentNode(); 16 16 17 18 17 if (2 >= segmentNode.getInboundSegments().size() 18 + segmentNode.getOutboundSegments().size() ) { 19 19 20 21 22 20 return new LatLonCoords( 21 node.getSegmentNode().getLat(), 22 node.getSegmentNode().getLon()); 23 23 24 24 } else { 25 25 26 27 26 SegmentNode node1 = node.getSegment().getNode1(); 27 SegmentNode node2 = node.getSegment().getNode2(); 28 28 29 29 assert segmentNode == node1 || segmentNode == node2; 30 30 31 31 LatLonCoords result; 32 32 33 34 35 36 37 38 39 40 41 33 if (segmentNode == node1) { 34 result = new LatLonCoords( 35 (2 * node1.getLat() + node2.getLat()) / 3, 36 (2 * node1.getLon() + node2.getLon()) / 3); 37 } else { 38 result = new LatLonCoords( 39 (node1.getLat() + 2 * node2.getLat()) / 3, 40 (node1.getLon() + 2 * node2.getLon()) / 3); 41 } 42 42 43 44 45 43 return result; 44 } 45 } 46 46 47 47 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/EndNodeColorScheme.java
r16520 r23189 12 12 public class EndNodeColorScheme implements ColorScheme { 13 13 14 15 16 14 private final Color nodeColor; 15 private final Color endNodeColor; 16 private final Color segmentColor; 17 17 18 19 20 21 22 18 public EndNodeColorScheme(Color nodeColor, Color endNodeColor, Color segmentColor) { 19 this.nodeColor = nodeColor; 20 this.endNodeColor = endNodeColor; 21 this.segmentColor = segmentColor; 22 } 23 23 24 25 26 24 public Color getNodeColor(GraphNode node) { 25 return GraphUtil.isEndNode(node) ? endNodeColor : nodeColor; 26 } 27 27 28 29 30 28 public Color getSegmentColor(Segment segment) { 29 return segmentColor; 30 } 31 31 32 32 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/FloatPropertyColorScheme.java
r18130 r23189 22 22 public class FloatPropertyColorScheme implements ColorScheme { 23 23 24 25 26 24 private final Class<? extends RoadPropertyType<Float>> propertyClass; 25 private final Map<Float, Color> colorMap; 26 private final Color defaultColor; 27 27 28 29 30 31 32 33 34 35 36 37 28 /** 29 * @param propertyClass property type to get values for; != null 30 * @param colorMap map from some values to colors. 31 * Colors for all other values are interpolated. 32 * This map will be copied and not used directly later. != null 33 * @param defaultColor color that is used when the property is not available; != null 34 */ 35 public FloatPropertyColorScheme(Class<? extends RoadPropertyType<Float>> propertyClass, 36 Map<Float, Color> colorMap, Color defaultColor) { 37 assert propertyClass != null && colorMap != null && defaultColor != null; 38 38 39 40 41 42 39 this.propertyClass = propertyClass; 40 this.colorMap = new HashMap<Float, Color>(colorMap); 41 this.defaultColor = defaultColor; 42 } 43 43 44 45 44 public Color getSegmentColor(Segment segment) { 45 assert segment != null; 46 46 47 48 49 50 51 52 53 54 55 56 47 Float propertyValue = null; 48 Collection<RoadPropertyType<?>> availableProperties = segment.getAvailableProperties(); 49 for (RoadPropertyType<?> property : availableProperties) { 50 if (propertyClass.isInstance(property)) { 51 @SuppressWarnings("unchecked") //has been checked using isInstance 52 RoadPropertyType<Float> floatProperty = (RoadPropertyType<Float>)property; 53 propertyValue = segment.getPropertyValue(floatProperty); 54 break; 55 } 56 } 57 57 58 59 60 61 62 63 58 if (propertyValue != null) { 59 return getColorForValue(propertyValue); 60 } else { 61 return defaultColor; 62 } 63 } 64 64 65 65 public Color getNodeColor(GraphNode node) { 66 66 67 67 List<Color> segmentColors = new ArrayList<Color>(); 68 68 69 70 71 for (GraphEdge edge : node.getInboundEdges()) {72 List<Segment> edgeSegments = edge.getPropertyValue(GraphEdgeSegments.PROPERTY);73 if (edgeSegments.size() > 0) {74 Segment firstSegment = edgeSegments.get(0);75 segmentColors.add(getSegmentColor(firstSegment));76 }77 }78 for (GraphEdge edge : node.getOutboundEdges()) {79 List<Segment> edgeSegments = edge.getPropertyValue(GraphEdgeSegments.PROPERTY);80 if (edgeSegments.size() > 0) {81 Segment lastSegment = edgeSegments.get(edgeSegments.size()-1);82 segmentColors.add(getSegmentColor(lastSegment));83 }84 }85 69 86 if (segmentColors.size() > 0) {87 return averageColor(segmentColors);88 } else {89 return Color.WHITE;90 }91 70 92 } 71 for (GraphEdge edge : node.getInboundEdges()) { 72 List<Segment> edgeSegments = edge.getPropertyValue(GraphEdgeSegments.PROPERTY); 73 if (edgeSegments.size() > 0) { 74 Segment firstSegment = edgeSegments.get(0); 75 segmentColors.add(getSegmentColor(firstSegment)); 76 } 77 } 78 for (GraphEdge edge : node.getOutboundEdges()) { 79 List<Segment> edgeSegments = edge.getPropertyValue(GraphEdgeSegments.PROPERTY); 80 if (edgeSegments.size() > 0) { 81 Segment lastSegment = edgeSegments.get(edgeSegments.size()-1); 82 segmentColors.add(getSegmentColor(lastSegment)); 83 } 84 } 93 85 94 /** 95 * returns the color for a value 96 * @param value value to get color for; != null 97 * @return color; != null 98 */ 99 protected Color getColorForValue(Float value) { 100 assert value != null; 86 if (segmentColors.size() > 0) { 87 return averageColor(segmentColors); 88 } else { 89 return Color.WHITE; 90 } 101 91 102 if (colorMap.containsKey(value)) { 92 } 103 93 104 return colorMap.get(value); 94 /** 95 * returns the color for a value 96 * @param value value to get color for; != null 97 * @return color; != null 98 */ 99 protected Color getColorForValue(Float value) { 100 assert value != null; 105 101 106 } else{102 if (colorMap.containsKey(value)) { 107 103 108 LinkedList<Float> valuesWithDefinedColor = new LinkedList<Float>(colorMap.keySet()); 109 Collections.sort(valuesWithDefinedColor); 104 return colorMap.get(value); 110 105 111 if (value <= valuesWithDefinedColor.getFirst()){106 } else { 112 107 113 return colorMap.get(valuesWithDefinedColor.getFirst()); 108 LinkedList<Float> valuesWithDefinedColor = new LinkedList<Float>(colorMap.keySet()); 109 Collections.sort(valuesWithDefinedColor); 114 110 115 } elseif (value>= valuesWithDefinedColor.getLast()) {111 if (value <= valuesWithDefinedColor.getFirst()) { 116 112 117 Last());113 return colorMap.get(valuesWithDefinedColor.getFirst()); 118 114 119 } else{115 } else if (value >= valuesWithDefinedColor.getLast()) { 120 116 121 /* interpolate */ 117 return colorMap.get(valuesWithDefinedColor.getLast()); 122 118 123 Float lowerValue = valuesWithDefinedColor.getFirst(); 124 Float higherValue = null; 119 } else { 125 120 126 for (Float v : valuesWithDefinedColor) { 127 if (v >= value) { 128 higherValue = v; 129 break; 130 } 131 lowerValue = v; 132 } 121 /* interpolate */ 133 122 134 assert lowerValue != null && higherValue != null; 123 Float lowerValue = valuesWithDefinedColor.getFirst(); 124 Float higherValue = null; 135 125 136 Color lowerColor = colorMap.get(lowerValue); 137 Color higherColor = colorMap.get(higherValue); 126 for (Float v : valuesWithDefinedColor) { 127 if (v >= value) { 128 higherValue = v; 129 break; 130 } 131 lowerValue = v; 132 } 138 133 139 float weightHigherColor = (value -lowerValue) / (higherValue- lowerValue);134 assert lowerValue != null && higherValue != null; 140 135 141 return weightedAverageColor(lowerColor, higherColor, weightHigherColor); 136 Color lowerColor = colorMap.get(lowerValue); 137 Color higherColor = colorMap.get(higherValue); 142 138 143 } 139 float weightHigherColor = (value - lowerValue) / (higherValue - lowerValue); 144 140 145 } 141 return weightedAverageColor(lowerColor, higherColor, weightHigherColor); 146 142 147 143 } 148 144 149 /** 150 * returns an average of all colors that have been passed as parameter 151 * 152 * @param colors colors to calculate average from; not empty or null 153 * @return average color; != null 154 */ 155 private static Color averageColor(List<Color> colors) { 156 assert colors != null && colors.size() > 0; 145 } 157 146 158 float weightPerColor = 1.0f / colors.size(); 147 } 159 148 160 Color average = new Color(0,0,0); 149 /** 150 * returns an average of all colors that have been passed as parameter 151 * 152 * @param colors colors to calculate average from; not empty or null 153 * @return average color; != null 154 */ 155 private static Color averageColor(List<Color> colors) { 156 assert colors != null && colors.size() > 0; 161 157 162 for (Color color : colors) { 163 average = new Color( 164 Math.min(Math.round(average.getRed() + weightPerColor*color.getRed()), 255), 165 Math.min(Math.round(average.getGreen() + weightPerColor*color.getGreen()), 255), 166 Math.min(Math.round(average.getBlue() + weightPerColor*color.getBlue()), 255)); 167 } 158 float weightPerColor = 1.0f / colors.size(); 168 159 169 return average; 170 } 160 Color average = new Color(0,0,0); 171 161 172 /** 173 * returns a weighted average of two colors 174 * 175 * @param color1 first color for the average; != null 176 * @param color2 second color for the average; != null 177 * @param weightColor2 weight of color2; must be in [0..1] 178 * @return average color; != null 179 */ 180 private static Color weightedAverageColor(Color color1, Color color2, float weightColor2) { 181 assert color1 != null && color2 != null; 182 assert 0 <= weightColor2 && weightColor2 <= 1; 162 for (Color color : colors) { 163 average = new Color( 164 Math.min(Math.round(average.getRed() + weightPerColor*color.getRed()), 255), 165 Math.min(Math.round(average.getGreen() + weightPerColor*color.getGreen()), 255), 166 Math.min(Math.round(average.getBlue() + weightPerColor*color.getBlue()), 255)); 167 } 183 168 184 return new Color( 185 Math.round((1 - weightColor2) * color1.getRed() + weightColor2 * color2.getRed()), 186 Math.round((1 - weightColor2) * color1.getGreen() + weightColor2 * color2.getGreen()), 187 Math.round((1 - weightColor2) * color1.getBlue() + weightColor2 * color2.getBlue())); 188 } 169 return average; 170 } 171 172 /** 173 * returns a weighted average of two colors 174 * 175 * @param color1 first color for the average; != null 176 * @param color2 second color for the average; != null 177 * @param weightColor2 weight of color2; must be in [0..1] 178 * @return average color; != null 179 */ 180 private static Color weightedAverageColor(Color color1, Color color2, float weightColor2) { 181 assert color1 != null && color2 != null; 182 assert 0 <= weightColor2 && weightColor2 <= 1; 183 184 return new Color( 185 Math.round((1 - weightColor2) * color1.getRed() + weightColor2 * color2.getRed()), 186 Math.round((1 - weightColor2) * color1.getGreen() + weightColor2 * color2.getGreen()), 187 Math.round((1 - weightColor2) * color1.getBlue() + weightColor2 * color2.getBlue())); 188 } 189 189 190 190 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/InclineColorScheme.java
r16520 r23189 12 12 public class InclineColorScheme extends FloatPropertyColorScheme { 13 13 14 14 private static final Map<Float, Color> COLOR_MAP; 15 15 16 17 18 19 20 21 16 static { 17 COLOR_MAP = new HashMap<Float, Color>(); 18 COLOR_MAP.put(-30f, Color.BLUE); 19 COLOR_MAP.put(0f, Color.WHITE); 20 COLOR_MAP.put(30f, Color.RED); 21 } 22 22 23 24 25 23 public InclineColorScheme() { 24 super(RoadIncline.class, COLOR_MAP, Color.GRAY); 25 } 26 26 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/LatLonCoords.java
r16520 r23189 6 6 public final class LatLonCoords { 7 7 8 9 8 private final double lat; 9 private final double lon; 10 10 11 12 13 14 11 public LatLonCoords(double lat, double lon) { 12 this.lat = lat; 13 this.lon = lon; 14 } 15 15 16 17 18 16 public double getLat() { 17 return lat; 18 } 19 19 20 21 22 20 public double getLon() { 21 return lon; 22 } 23 23 24 24 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/MaxheightColorScheme.java
r16520 r23189 12 12 public class MaxheightColorScheme extends FloatPropertyColorScheme { 13 13 14 14 private static final Map<Float, Color> COLOR_MAP; 15 15 16 17 18 19 20 21 16 static { 17 COLOR_MAP = new HashMap<Float, Color>(); 18 COLOR_MAP.put(0f, new Color(0, 0, 50)); 19 COLOR_MAP.put(10f, new Color(100, 100, 255)); 20 COLOR_MAP.put(30f, new Color(200, 200, 255)); 21 } 22 22 23 24 25 23 public MaxheightColorScheme() { 24 super(RoadMaxheight.class, COLOR_MAP, Color.WHITE); 25 } 26 26 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/MaxspeedColorScheme.java
r16520 r23189 12 12 public class MaxspeedColorScheme extends FloatPropertyColorScheme { 13 13 14 14 private static final Map<Float, Color> COLOR_MAP; 15 15 16 17 18 19 20 21 22 23 16 static { 17 COLOR_MAP = new HashMap<Float, Color>(); 18 COLOR_MAP.put(0f, new Color(50, 0, 0)); 19 COLOR_MAP.put(30f, Color.RED); 20 COLOR_MAP.put(60f, Color.YELLOW); 21 COLOR_MAP.put(90f, Color.GREEN); 22 COLOR_MAP.put(150f, Color.BLUE); 23 } 24 24 25 26 27 25 public MaxspeedColorScheme() { 26 super(RoadMaxspeed.class, COLOR_MAP, Color.GRAY); 27 } 28 28 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/MaxweightColorScheme.java
r16520 r23189 12 12 public class MaxweightColorScheme extends FloatPropertyColorScheme { 13 13 14 14 private static final Map<Float, Color> COLOR_MAP; 15 15 16 17 18 19 20 21 16 static { 17 COLOR_MAP = new HashMap<Float, Color>(); 18 COLOR_MAP.put(0f, new Color(0, 0, 50)); 19 COLOR_MAP.put(20f, new Color(100, 100, 255)); 20 COLOR_MAP.put(50f, new Color(200, 200, 255)); 21 } 22 22 23 24 25 23 public MaxweightColorScheme() { 24 super(RoadMaxweight.class, COLOR_MAP, Color.WHITE); 25 } 26 26 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/NodePositioner.java
r16520 r23189 9 9 public interface NodePositioner { 10 10 11 11 LatLonCoords getPosition(GraphNode node); 12 12 13 13 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/NonMovingNodePositioner.java
r16520 r23189 5 5 public class NonMovingNodePositioner implements NodePositioner { 6 6 7 8 9 10 11 7 public LatLonCoords getPosition(GraphNode node) { 8 return new LatLonCoords( 9 node.getSegmentNode().getLat(), 10 node.getSegmentNode().getLon()); 11 } 12 12 13 13 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/core/visualisation/SingleColorScheme.java
r16520 r23189 12 12 public class SingleColorScheme implements ColorScheme { 13 13 14 15 14 private final Color nodeColor; 15 private final Color segmentColor; 16 16 17 18 19 20 17 public SingleColorScheme(Color nodeColor, Color segmentColor) { 18 this.nodeColor = nodeColor; 19 this.segmentColor = segmentColor; 20 } 21 21 22 23 24 22 public Color getNodeColor(GraphNode node) { 23 return nodeColor; 24 } 25 25 26 27 28 26 public Color getSegmentColor(Segment segment) { 27 return segmentColor; 28 } 29 29 30 30 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/GraphViewPlugin.java
r19441 r23189 52 52 public class GraphViewPlugin extends Plugin implements LayerChangeListener, Observer { 53 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 54 private static final Collection<RoadPropertyType<?>> PROPERTIES; 55 56 static { 57 PROPERTIES = new LinkedList<RoadPropertyType<?>>(); 58 PROPERTIES.add(new RoadIncline()); 59 PROPERTIES.add(new RoadMaxaxleload()); 60 PROPERTIES.add(new RoadMaxheight()); 61 PROPERTIES.add(new RoadMaxlength()); 62 PROPERTIES.add(new RoadMaxspeed()); 63 PROPERTIES.add(new RoadMaxweight()); 64 PROPERTIES.add(new RoadMaxwidth()); 65 PROPERTIES.add(new RoadMinspeed()); 66 PROPERTIES.add(new RoadSurface()); 67 PROPERTIES.add(new RoadTracktype()); 68 PROPERTIES.add(new RoadWidth()); 69 } 70 71 private final GraphViewPreferences preferences; 72 73 private JOSMTransitionStructure transitionStructure; 74 private GraphViewLayer graphViewLayer; 75 76 /** creates the plugin */ 77 public GraphViewPlugin(PluginInformation info) { 78 super(info); 79 preferences = GraphViewPreferences.getInstance(); 80 this.preferences.addObserver(this); 81 82 } 83 84 /** allows creation/update of GraphViewLayer */ 85 public void createGraphViewLayer() { 86 87 try { 88 89 if (graphViewLayer != null) { 90 91 AccessRuleset accessRuleset = getAccessRuleset(); 92 93 if (accessRuleset == null) { 94 JOptionPane.showMessageDialog(Main.parent, "No ruleset has been selected!", "No ruleset", JOptionPane.ERROR_MESSAGE); 95 } else { 96 transitionStructure.setAccessParametersAndRuleset(preferences.getCurrentParameterBookmark(), accessRuleset); 97 transitionStructure.forceUpdate(); 98 } 99 100 } else { 101 102 AccessRuleset accessRuleset = getAccessRuleset(); 103 104 if (accessRuleset == null) { 105 JOptionPane.showMessageDialog(Main.parent, "No ruleset has been selected!", 106 "No ruleset", JOptionPane.ERROR_MESSAGE); 107 } else { 108 109 transitionStructure = new JOSMTransitionStructure( 110 preferences.getCurrentParameterBookmark(), 111 accessRuleset, 112 PROPERTIES); 113 114 WayGraph graph = new TSBasedWayGraph(transitionStructure); 115 116 graphViewLayer = new GraphViewLayer(); 117 graphViewLayer.setWayGraph(graph); 118 graphViewLayer.setColorScheme(preferences.getCurrentColorScheme()); 119 graphViewLayer.setNodePositioner(new DefaultNodePositioner()); 120 121 Main.main.addLayer(graphViewLayer); 122 123 } 124 125 } 126 127 } catch (AccessRulesetSyntaxException e) { 128 JOptionPane.showMessageDialog(Main.parent, "syntax exception in access ruleset:\n" + e); 129 e.printStackTrace(); 130 } catch (FileNotFoundException e) { 131 JOptionPane.showMessageDialog(Main.parent, "file not found:\n" + e); 132 e.printStackTrace(); 133 } catch (IOException e) { 134 JOptionPane.showMessageDialog(Main.parent, "problem when accessing a file:\n" + e); 135 e.printStackTrace(); 136 } 137 138 } 139 140 /** allows update of GraphViewLayer */ 141 public void updateGraphViewLayer() { 142 143 try { 144 145 if (graphViewLayer != null) { 146 147 AccessRuleset accessRuleset = getAccessRuleset(); 148 149 if (accessRuleset == null) { 150 JOptionPane.showMessageDialog(Main.parent, "No ruleset has been selected!", 151 "No ruleset", JOptionPane.ERROR_MESSAGE); 152 } else { 153 transitionStructure.setAccessParametersAndRuleset( 154 preferences.getCurrentParameterBookmark(), accessRuleset); 155 transitionStructure.forceUpdate(); 156 } 157 158 } 159 160 } catch (AccessRulesetSyntaxException e) { 161 JOptionPane.showMessageDialog(Main.parent, "syntax exception in access ruleset:\n" + e); 162 e.printStackTrace(); 163 } catch (FileNotFoundException e) { 164 JOptionPane.showMessageDialog(Main.parent, "file not found:\n" + e); 165 e.printStackTrace(); 166 } catch (IOException e) { 167 JOptionPane.showMessageDialog(Main.parent, "problem when accessing a file:\n" + e); 168 e.printStackTrace(); 169 } 170 171 } 172 173 /** repaints the GraphViewLayer without recalculating the graph (visual update) */ 174 public void repaintGraphViewLayer() { 175 176 if (graphViewLayer != null) { 177 Main.panel.repaint(); 178 } 179 180 } 181 182 /** 183 * @return ruleset read from a source as specified by preferences, null if the preferences 184 * don't specify a ruleset source 185 * @throws AccessRulesetSyntaxException 186 * @throws IOException 187 * @throws FileNotFoundException 188 */ 189 private AccessRuleset getAccessRuleset() 190 throws AccessRulesetSyntaxException, IOException, FileNotFoundException { 191 192 InputStream rulesetInputStream; 193 194 if (preferences.getUseInternalRulesets()) { 195 196 InternalRuleset ruleset = preferences.getCurrentInternalRuleset(); 197 198 if (ruleset == null) { 199 return null; 200 } 201 202 ClassLoader classLoader = this.getClass().getClassLoader(); 203 URL rulesetURL = classLoader.getResource(ruleset.getResourceName()); 204 205 if (rulesetURL != null) { 206 rulesetInputStream = rulesetURL.openStream(); 207 } else { 208 throw new FileNotFoundException("couldn't find built-in ruleset " + ruleset); 209 } 210 211 } else { 212 213 File rulesetFile = preferences.getCurrentRulesetFile(); 214 215 if (rulesetFile == null) { 216 return null; 217 } 218 219 rulesetInputStream = new FileInputStream(rulesetFile); 220 221 } 222 223 return AccessRulesetReader.readAccessRuleset(rulesetInputStream); 224 225 } 226 227 @Override 228 public PreferenceSetting getPreferenceSetting() { 229 return new GraphViewPreferenceEditor(); 230 } 231 232 @Override 233 public void mapFrameInitialized(MapFrame oldFrame, MapFrame newFrame) { 234 if (newFrame != null) { 235 if (oldFrame == null) { 236 final GraphViewDialog laneDialog 237 = new GraphViewDialog(this); 238 newFrame.addToggleDialog(laneDialog); 239 } 240 MapView.addLayerChangeListener(this); 241 } else { 242 MapView.removeLayerChangeListener(this); 243 } 244 } 245 246 public void layerRemoved(Layer oldLayer) { 247 if (oldLayer == graphViewLayer) { 248 graphViewLayer = null; 249 } else if (oldLayer == Main.map.mapView.getEditLayer()) { //data layer removed 250 if (graphViewLayer != null) { 251 Main.map.mapView.removeLayer(graphViewLayer); 252 graphViewLayer = null; 253 } 254 } 255 } 256 257 public void activeLayerChange(Layer oldLayer, Layer newLayer) { 258 //do nothing 259 } 260 261 public void layerAdded(Layer newLayer) { 262 //do nothing 263 } 264 265 public void update(Observable arg0, Object arg1) { 266 if (arg0 == preferences) { 267 if (graphViewLayer != null) { 268 graphViewLayer.setColorScheme(preferences.getCurrentColorScheme()); 269 } 270 } 271 } 272 272 273 273 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/data/JOSMDataSource.java
r21610 r23189 26 26 public class JOSMDataSource implements DataSource<Node, Way, Relation, RelationMember> { 27 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 28 public double getLat(Node node) { 29 return node.getCoor().lat(); 30 } 31 32 public double getLon(Node node) { 33 return node.getCoor().lon(); 34 } 35 36 public Iterable<RelationMember> getMembers(Relation relation) { 37 return relation.getMembers(); 38 } 39 40 public Iterable<Node> getNodes(Way way) { 41 return new FilteredOsmPrimitiveIterable<Node>(way.getNodes()); 42 } 43 44 public Iterable<Node> getNodes() { 45 return new FilteredOsmPrimitiveIterable<Node>(Main.main.getCurrentDataSet().getNodes()); 46 } 47 48 public Iterable<Relation> getRelations() { 49 return new FilteredRelationIterable(Main.main.getCurrentDataSet().getRelations()); 50 } 51 52 public Iterable<Way> getWays() { 53 return new FilteredOsmPrimitiveIterable<Way>(Main.main.getCurrentDataSet().getWays()); 54 } 55 56 public TagGroup getTagsN(Node node) { 57 return getTags(node); 58 } 59 60 public TagGroup getTagsW(Way way) { 61 return getTags(way); 62 } 63 64 public TagGroup getTagsR(Relation relation) { 65 return getTags(relation); 66 } 67 68 private TagGroup getTags(OsmPrimitive primitive) { 69 if (primitive.getKeys() == null) { 70 return EMPTY_TAG_GROUP; 71 } else { 72 return new MapBasedTagGroup(primitive.getKeys()); 73 } 74 } 75 76 public Object getMember(RelationMember member) { 77 return member.getMember(); 78 } 79 80 public String getRole(RelationMember member) { 81 return member.getRole(); 82 } 83 84 public boolean isNMember(RelationMember member) { 85 return member.getMember() instanceof Node; 86 } 87 88 public boolean isWMember(RelationMember member) { 89 return member.getMember() instanceof Way; 90 } 91 92 public boolean isRMember(RelationMember member) { 93 return member.getMember() instanceof Relation; 94 } 95 96 97 private static final TagGroup EMPTY_TAG_GROUP; 98 static { 99 Map<String, String> emptyMap = new HashMap<String, String>(0); 100 EMPTY_TAG_GROUP = new MapBasedTagGroup(emptyMap); 101 } 102 103 /** 104 * Iterable of OsmPrimitive objects based on an existing Iterable, 105 * will filter incomplete and deleted objects from the iterator. 106 * 107 * @param <P> OsmPrimitive subtype 108 */ 109 public static class FilteredOsmPrimitiveIterable<P extends OsmPrimitive> implements Iterable<P> { 110 111 private final Iterable<P> originalIterable; 112 113 public FilteredOsmPrimitiveIterable(Iterable<P> originalIterable) { 114 this.originalIterable = originalIterable; 115 } 116 117 /** returns an iterator. The iterator does not support {@link Iterator#remove()}. */ 118 public Iterator<P> iterator() { 119 return new FilteredIterator(originalIterable.iterator()); 120 } 121 122 private class FilteredIterator implements Iterator<P> { 123 124 private final Iterator<P> originalIterator; 125 126 private P next; 127 128 public FilteredIterator(Iterator<P> originalIterator) { 129 this.originalIterator = originalIterator; 130 updateNext(); 131 } 132 133 public boolean hasNext() { 134 return next != null; 135 } 136 137 public P next() { 138 if (next != null) { 139 P result = next; 140 updateNext(); 141 return result; 142 } else { 143 throw new NoSuchElementException(); 144 } 145 } 146 147 public void remove() { 148 throw new UnsupportedOperationException(); 149 } 150 151 private void updateNext() { 152 next = null; 153 while (originalIterator.hasNext()) { 154 P originalNext = originalIterator.next(); 155 if (accept(originalNext)) { 156 next = originalNext; 157 break; 158 } 159 } 160 } 161 162 } 163 164 protected boolean accept(P primitive) { 165 return !primitive.isDeleted() && !primitive.isIncomplete(); 166 } 167 } 168 169 /** 170 * Relation-specific variant of the FilteredOsmPrimitiveIterable, 171 * also checks completeness of relation's members 172 */ 173 public static class FilteredRelationIterable extends FilteredOsmPrimitiveIterable<Relation> { 174 175 public FilteredRelationIterable(Iterable<Relation> originalIterable) { 176 super(originalIterable); 177 } 178 179 @Override 180 protected boolean accept(Relation relation) { 181 boolean complete = true; 182 for (org.openstreetmap.josm.data.osm.RelationMember member : relation.getMembers()) { 183 if (member.getMember() == null || member.getMember().isDeleted() || member.getMember().isIncomplete()) { 184 complete = false; 185 } 186 } 187 return complete && super.accept(relation); 188 } 189 } 190 191 static class RelationMemberImpl { 192 private final String role; 193 private final Object member; 194 public RelationMemberImpl(org.openstreetmap.josm.data.osm.RelationMember originalMember) { 195 this.role = originalMember.getRole(); 196 this.member = originalMember.getMember(); 197 } 198 public String getRole() { 199 return role; 200 } 201 public Object getMember() { 202 return member; 203 } 204 } 205 206 private final Set<DataSourceObserver> observers = new HashSet<DataSourceObserver>(); 207 208 public void addObserver(DataSourceObserver observer) { 209 observers.add(observer); 210 } 211 212 public void deleteObserver(DataSourceObserver observer) { 213 observers.remove(observer); 214 } 215 215 216 216 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/data/JOSMTransitionStructure.java
r19216 r23189 17 17 public class JOSMTransitionStructure extends GenericTransitionStructure<Node, Way, Relation, RelationMember> { 18 18 19 19 private static final JOSMDataSource DATA_SOURCE = new JOSMDataSource(); 20 20 21 22 21 public JOSMTransitionStructure(AccessParameters accessParameters, AccessRuleset ruleset, 22 Collection<RoadPropertyType<?>> properties) { 23 23 24 25 26 24 super(accessParameters, ruleset, 25 DATA_SOURCE, 26 properties); 27 27 28 28 } 29 29 30 31 32 33 30 /** causes an update (as if the DataSource had noticed a change) */ 31 public void forceUpdate() { 32 super.update(DATA_SOURCE); 33 } 34 34 35 35 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/dialogs/AccessParameterDialog.java
r16520 r23189 38 38 public class AccessParameterDialog extends JDialog { 39 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 40 public static interface BookmarkAction { 41 public void execute(String name, PreferenceAccessParameters parameters); 42 } 43 44 /** 45 * map that contains all float value vehicle properties (as those can be treated uniformly) 46 * and their labels 47 */ 48 private static final Map<VehiclePropertyType<Float>, String> FLOAT_PROPERTIES; 49 50 static { 51 FLOAT_PROPERTIES = new LinkedHashMap<VehiclePropertyType<Float>, String>(); 52 FLOAT_PROPERTIES.put(VehiclePropertyTypes.HEIGHT, "height (m)"); 53 FLOAT_PROPERTIES.put(VehiclePropertyTypes.WIDTH, "width (m)"); 54 FLOAT_PROPERTIES.put(VehiclePropertyTypes.LENGTH, "length (m)"); 55 FLOAT_PROPERTIES.put(VehiclePropertyTypes.SPEED, "speed (km/h)"); 56 FLOAT_PROPERTIES.put(VehiclePropertyTypes.WEIGHT, "weight (t)"); 57 FLOAT_PROPERTIES.put(VehiclePropertyTypes.AXLELOAD, "axleload (t)"); 58 } 59 60 private static final Collection<Character> FORBIDDEN_CHARS = 61 Arrays.asList(',', ';', '{', '}', '=', '|'); 62 63 private class BookmarkNamePanel extends JPanel { 64 65 private final JTextField bookmarkNameTextField; 66 67 public BookmarkNamePanel(String initialName) { 68 super(); 69 this.setBorder(BorderFactory.createTitledBorder("bookmark name")); 70 71 this.setLayout(new BoxLayout(this, BoxLayout.X_AXIS)); 72 73 bookmarkNameTextField = new JTextField(initialName); 74 this.add(bookmarkNameTextField); 75 76 } 77 78 public String getBookmarkName() { 79 80 String name = bookmarkNameTextField.getText(); 81 82 if (existingBookmarkNames.contains(name)) { 83 JOptionPane.showMessageDialog(this, "Bookmark name already exists!"); 84 return null; 85 } 86 87 for (char nameChar : name.toCharArray()) { 88 if (FORBIDDEN_CHARS.contains(nameChar)) { 89 JOptionPane.showMessageDialog(this, "Bookmark name must not contain '" + 90 nameChar + "'!"); 91 return null; 92 } 93 } 94 95 return name; 96 } 97 } 98 99 private static class AccessClassPanel extends JPanel { 100 101 private final JTextField accessClassTextField; 102 103 public AccessClassPanel(PreferenceAccessParameters initialParameters) { 104 super(); 105 this.setBorder(BorderFactory.createTitledBorder("access class")); 106 107 this.setLayout(new BoxLayout(this, BoxLayout.X_AXIS)); 108 109 accessClassTextField = new JTextField(initialParameters.getAccessClass()); 110 this.add(accessClassTextField); 111 112 } 113 114 public String getAccessClass() { 115 116 String name = accessClassTextField.getText(); 117 118 for (char nameChar : name.toCharArray()) { 119 if (FORBIDDEN_CHARS.contains(nameChar)) { 120 JOptionPane.showMessageDialog(this, "Access class must not contain '" + 121 nameChar + "'!"); 122 return null; 123 } 124 } 125 126 return name; 127 } 128 } 129 130 private static class AccessTypesPanel extends JPanel { 131 132 private static final int COLS = 4; 133 134 private final Map<AccessType, JCheckBox> accessTypeCheckBoxes = 135 new EnumMap<AccessType, JCheckBox>(AccessType.class); 136 137 public AccessTypesPanel(PreferenceAccessParameters initialParameters) { 138 super(); 139 this.setBorder(BorderFactory.createTitledBorder("access types")); 140 141 this.setLayout( 142 new GridLayout(((COLS-1 + AccessType.values().length) / COLS), COLS)); 143 144 for (AccessType accessType : AccessType.values()) { 145 JCheckBox checkBox = new JCheckBox(accessType.toString()); 146 checkBox.setSelected(initialParameters.getAccessTypeUsable(accessType)); 147 accessTypeCheckBoxes.put(accessType, checkBox); 148 this.add(checkBox); 149 } 150 151 } 152 153 public Collection<AccessType> getUsableAccessTypes() { 154 155 Collection<AccessType> usableAccessTypes = new LinkedList<AccessType>(); 156 157 for (AccessType accessType : AccessType.values()) { 158 if (accessTypeCheckBoxes.get(accessType).isSelected()) { 159 usableAccessTypes.add(accessType); 160 } 161 } 162 163 return usableAccessTypes; 164 } 165 } 166 167 private static class VehiclePropertiesPanel extends JPanel { 168 169 private static final int COLS = 2; 170 171 private final Map<VehiclePropertyType<Float>, JTextField> floatPropertyTextFields = 172 new HashMap<VehiclePropertyType<Float>, JTextField>(); 173 174 public VehiclePropertiesPanel(PreferenceAccessParameters initialParameters) { 175 super(); 176 this.setBorder(BorderFactory.createTitledBorder("vehicle properties")); 177 178 this.setLayout(new GridLayout(((COLS-1 + FLOAT_PROPERTIES.size()) / COLS), 179 2*COLS)); 180 181 for (VehiclePropertyType<Float> vehicleProperty : FLOAT_PROPERTIES.keySet()) { 182 183 JLabel label = new JLabel(FLOAT_PROPERTIES.get(vehicleProperty)); 184 this.add(label); 185 186 JTextField textField = new JTextField(); 187 188 String vehiclePropertyString = 189 initialParameters.getVehiclePropertyString(vehicleProperty); 190 if (vehiclePropertyString != null) { 191 textField.setText(vehiclePropertyString); 192 } 193 194 floatPropertyTextFields.put(vehicleProperty, textField); 195 this.add(textField); 196 } 197 198 } 199 200 public Map<VehiclePropertyType<?>, String> getVehiclePropertyStrings() { 201 202 Map<VehiclePropertyType<?>, String> vehiclePropertyStrings = 203 new HashMap<VehiclePropertyType<?>, String>(); 204 205 for (VehiclePropertyType<Float> vehicleProperty : floatPropertyTextFields.keySet()) { 206 String textFieldContent = floatPropertyTextFields.get(vehicleProperty).getText(); 207 if (textFieldContent.trim().length() > 0) { 208 vehiclePropertyStrings.put(vehicleProperty, textFieldContent); 209 } 210 } 211 212 return vehiclePropertyStrings; 213 } 214 } 215 216 private static class RoadQualityPanel extends JPanel { 217 218 private JTextField inclineUpTextField; 219 private JTextField inclineDownTextField; 220 private JTextField surfaceTextField; 221 private JTextField tracktypeTextField; 222 223 public RoadQualityPanel(PreferenceAccessParameters initialParameters) { 224 super(); 225 this.setBorder(BorderFactory.createTitledBorder("road requirements")); 226 227 228 this.setLayout(new GridLayout(4, 2)); 229 230 /* incline up */ 231 { 232 JLabel inclineUpLabel = new JLabel("max. incline up (%, pos.)"); 233 inclineUpLabel.setToolTipText("maximum incline the vehicle can go up"); 234 this.add(inclineUpLabel); 235 236 inclineUpTextField = new JTextField(); 237 238 String vehiclePropertyString = 239 initialParameters.getVehiclePropertyString(MAX_INCLINE_UP); 240 if (vehiclePropertyString != null) { 241 inclineUpTextField.setText(vehiclePropertyString); 242 } 243 inclineUpTextField.setToolTipText("maximum incline the vehicle can go up"); 244 245 this.add(inclineUpTextField); 246 } 247 248 /* incline down */ 249 { 250 JLabel inclineDownLabel = new JLabel("max. incline down (%, pos.)"); 251 inclineDownLabel.setToolTipText("maximum incline the vehicle can go down"); 252 this.add(inclineDownLabel); 253 254 inclineDownTextField = new JTextField(); 255 256 String vehiclePropertyString = 257 initialParameters.getVehiclePropertyString(MAX_INCLINE_DOWN); 258 if (vehiclePropertyString != null) { 259 inclineDownTextField.setText(vehiclePropertyString); 260 } 261 inclineDownTextField.setToolTipText("maximum incline the vehicle can go down"); 262 263 this.add(inclineDownTextField); 264 } 265 266 /* surface */ 267 { 268 JLabel surfaceLabel = new JLabel("surface blacklist"); 269 surfaceLabel.setToolTipText("list of surfaces the vehicle cannot use, " 270 + "values are separated by semicolons (;)"); 271 this.add(surfaceLabel); 272 273 surfaceTextField = new JTextField(); 274 275 String vehiclePropertyString = 276 initialParameters.getVehiclePropertyString(SURFACE_BLACKLIST); 277 278 if (vehiclePropertyString != null) { 279 surfaceTextField.setText(vehiclePropertyString); 280 } 281 282 surfaceTextField.setToolTipText("list of surfaces the vehicle cannot use, " 283 + "values are separated by semicolons (;)"); 284 285 this.add(surfaceTextField); 286 } 287 288 /* tracktype */ 289 { 290 JLabel tracktypeLabel = new JLabel("max. tracktype grade"); 291 tracktypeLabel.setToolTipText("worst tracktype (1-5) the vehicle can still use," 292 + " 0 for none"); 293 this.add(tracktypeLabel); 294 295 tracktypeTextField = new JTextField(); 296 297 String vehiclePropertyString = 298 initialParameters.getVehiclePropertyString(MAX_TRACKTYPE); 299 if (vehiclePropertyString != null) { 300 tracktypeTextField.setText(vehiclePropertyString); 301 } 302 tracktypeTextField.setToolTipText("worst tracktype (1-5) the vehicle can still use," 303 + " 0 for none"); 304 305 this.add(tracktypeTextField); 306 } 307 308 } 309 310 public Map<VehiclePropertyType<?>, String> getVehiclePropertyStrings() { 311 312 Map<VehiclePropertyType<?>, String> vehiclePropertyStrings = 313 new HashMap<VehiclePropertyType<?>, String>(); 314 315 String incUpString = inclineUpTextField.getText(); 316 if (incUpString.trim().length() > 0) { 317 vehiclePropertyStrings.put(MAX_INCLINE_UP, incUpString); 318 } 319 320 String incDownString = inclineDownTextField.getText(); 321 if (incDownString.trim().length() > 0) { 322 vehiclePropertyStrings.put(MAX_INCLINE_DOWN, incDownString); 323 } 324 325 String surfaceString = surfaceTextField.getText(); 326 if (surfaceString.trim().length() > 0) { 327 vehiclePropertyStrings.put(SURFACE_BLACKLIST, surfaceString); 328 } 329 330 String tracktypeString = tracktypeTextField.getText(); 331 if (tracktypeString.trim().length() > 0) { 332 vehiclePropertyStrings.put(MAX_TRACKTYPE, tracktypeString); 333 } 334 335 return vehiclePropertyStrings; 336 } 337 } 338 339 private class OkCancelPanel extends JPanel { 340 341 public OkCancelPanel() { 342 343 new BoxLayout(this, BoxLayout.X_AXIS); 344 345 JButton okButton = new JButton(existingBookmark?"Change bookmark":"Create bookmark"); 346 okButton.addActionListener(new ActionListener() { 347 public void actionPerformed(ActionEvent e) { 348 String bookmarkName = bookmarkNamePanel.getBookmarkName(); 349 if (bookmarkName != null) { 350 PreferenceAccessParameters parameters = getAccessParameters(); 351 if (parameters != null) { 352 okAction.execute(bookmarkName, parameters); 353 AccessParameterDialog.this.dispose(); 354 } 355 } 356 } 357 }); 358 this.add(okButton); 359 360 JButton cancelButton = new JButton("Cancel"); 361 cancelButton.addActionListener(new ActionListener() { 362 public void actionPerformed(ActionEvent e) { 363 AccessParameterDialog.this.dispose(); 364 } 365 }); 366 this.add(cancelButton); 367 368 } 369 370 } 371 372 private boolean existingBookmark = false; 373 private final Collection<String> existingBookmarkNames; 374 375 private final BookmarkAction okAction; 376 377 private final BookmarkNamePanel bookmarkNamePanel; 378 private final AccessClassPanel accessClassPanel; 379 private final AccessTypesPanel accessTypesPanel; 380 private final VehiclePropertiesPanel vehiclePropertiesPanel; 381 private final RoadQualityPanel roadQualityPanel; 382 383 public AccessParameterDialog(final Frame owner, boolean existingBookmark, String initialName, 384 Collection<String> existingBookmarkNames, 385 PreferenceAccessParameters initialAccessParameters, BookmarkAction okAction) { 386 super(owner, "edit access parameters", true); 387 388 this.existingBookmark = existingBookmark; 389 this.existingBookmarkNames = existingBookmarkNames; 390 this.okAction = okAction; 391 392 GridBagLayout layout = new GridBagLayout(); 393 this.setLayout(layout); 394 395 GridBagConstraints gbc = new GridBagConstraints(); 396 gbc.gridx = 0; 397 gbc.weighty = 1; 398 gbc.fill = GridBagConstraints.BOTH; 399 400 bookmarkNamePanel = new BookmarkNamePanel(initialName); 401 gbc.gridy = 0; 402 layout.setConstraints(bookmarkNamePanel, gbc); 403 this.add(bookmarkNamePanel); 404 405 accessClassPanel = new AccessClassPanel(initialAccessParameters); 406 gbc.gridy = 1; 407 layout.setConstraints(accessClassPanel, gbc); 408 this.add(accessClassPanel); 409 410 accessTypesPanel = new AccessTypesPanel(initialAccessParameters); 411 gbc.gridy = 2; 412 layout.setConstraints(accessTypesPanel, gbc); 413 this.add(accessTypesPanel); 414 415 vehiclePropertiesPanel = new VehiclePropertiesPanel(initialAccessParameters); 416 gbc.gridy = 3; 417 layout.setConstraints(vehiclePropertiesPanel, gbc); 418 this.add(vehiclePropertiesPanel); 419 420 roadQualityPanel = new RoadQualityPanel(initialAccessParameters); 421 gbc.gridy = 4; 422 layout.setConstraints(roadQualityPanel, gbc); 423 this.add(roadQualityPanel); 424 425 JPanel okCancelPanel = new OkCancelPanel(); 426 gbc.gridy = 5; 427 gbc.weighty = 0; 428 gbc.fill = GridBagConstraints.HORIZONTAL; 429 layout.setConstraints(okCancelPanel, gbc); 430 this.add(okCancelPanel); 431 432 this.pack(); 433 } 434 435 private PreferenceAccessParameters getAccessParameters() { 436 437 String accessClass = accessClassPanel.getAccessClass(); 438 Collection<AccessType> usableAccessTypes = accessTypesPanel.getUsableAccessTypes(); 439 Map<VehiclePropertyType<?>, String> vehiclePropertyStrings = 440 vehiclePropertiesPanel.getVehiclePropertyStrings(); 441 Map<VehiclePropertyType<?>, String> additionalVehiclePropertyStrings = 442 roadQualityPanel.getVehiclePropertyStrings(); 443 444 if (accessClass != null && usableAccessTypes != null && vehiclePropertyStrings != null 445 && additionalVehiclePropertyStrings != null) { 446 447 vehiclePropertyStrings.putAll(additionalVehiclePropertyStrings); 448 449 try { 450 return new PreferenceAccessParameters(accessClass, usableAccessTypes, vehiclePropertyStrings); 451 } catch (PropertyValueSyntaxException e) { 452 JOptionPane.showMessageDialog(this, e.getMessage()); 453 return null; 454 } 455 456 } else { 457 return null; 458 } 459 } 460 460 461 461 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/dialogs/GraphViewDialog.java
r16520 r23189 44 44 public class GraphViewDialog extends ToggleDialog implements Observer { 45 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 46 private static final int HEIGHT = 150; 47 48 /** map from labels to available color schemes */ 49 private final LinkedHashMap<String, ColorScheme> availableColorSchemes; 50 51 52 private final GraphViewPreferences preferences; 53 private final GraphViewPlugin plugin; 54 55 private final JComboBox rulesetComboBox; 56 private final JComboBox bookmarkComboBox; 57 private final JComboBox colorSchemeComboBox; 58 59 /** 60 * list of ruleset files in the order currently used by rulesetComboBox; 61 * null if internal rulesets are used 62 */ 63 private List<File> rulesetFiles; 64 65 public GraphViewDialog(final GraphViewPlugin plugin) { 66 67 super("Graph View Dialog", "graphview", 68 "Open the dialog for graph view configuration.", (Shortcut)null, HEIGHT); 69 70 this.preferences = GraphViewPreferences.getInstance(); 71 this.plugin = plugin; 72 73 availableColorSchemes = new LinkedHashMap<String, ColorScheme>(); 74 75 availableColorSchemes.put("default", 76 new PreferencesColorScheme(preferences)); 77 availableColorSchemes.put("end nodes", 78 new EndNodeColorScheme(Color.GRAY, Color.RED, Color.GRAY)); 79 availableColorSchemes.put("maxspeed", 80 new MaxspeedColorScheme()); 81 availableColorSchemes.put("maxweight", 82 new MaxweightColorScheme()); 83 availableColorSchemes.put("maxheight", 84 new MaxheightColorScheme()); 85 availableColorSchemes.put("incline", 86 new InclineColorScheme()); 87 88 JPanel selectionPanel = new JPanel(); 89 GridBagLayout selectionLayout = new GridBagLayout(); 90 selectionPanel.setLayout(selectionLayout); 91 92 GridBagConstraints gbcLabel = new GridBagConstraints(); 93 gbcLabel.gridx = 0; 94 gbcLabel.anchor = GridBagConstraints.WEST; 95 gbcLabel.insets = new Insets(0, 5, 0, 5); 96 97 GridBagConstraints gbcComboBox = new GridBagConstraints(); 98 gbcComboBox.gridx = 1; 99 gbcComboBox.fill = GridBagConstraints.HORIZONTAL; 100 gbcComboBox.weightx = 1; 101 102 103 /* create ruleset label and combo box */ 104 { 105 JLabel rulesetLabel = new JLabel("ruleset:"); 106 gbcLabel.gridy = 0; 107 selectionLayout.setConstraints(rulesetLabel, gbcLabel); 108 selectionPanel.add(rulesetLabel); 109 110 rulesetComboBox = new JComboBox(); 111 rulesetComboBox.addActionListener(rulesetActionListener); 112 gbcComboBox.gridy = 0; 113 selectionLayout.setConstraints(rulesetComboBox, gbcComboBox); 114 selectionPanel.add(rulesetComboBox); 115 } 116 117 /* create bookmark label and combo box */ 118 { 119 JLabel bookmarkLabel = new JLabel("parameters:"); 120 gbcLabel.gridy = 1; 121 selectionLayout.setConstraints(bookmarkLabel, gbcLabel); 122 selectionPanel.add(bookmarkLabel); 123 124 bookmarkComboBox = new JComboBox(); 125 bookmarkComboBox.addActionListener(bookmarkActionListener); 126 gbcComboBox.gridy = 1; 127 selectionLayout.setConstraints(bookmarkComboBox, gbcComboBox); 128 selectionPanel.add(bookmarkComboBox); 129 } 130 131 /* create color scheme label and combo box */ 132 { 133 JLabel colorSchemeLabel = new JLabel("coloring:"); 134 gbcLabel.gridy = 2; 135 selectionLayout.setConstraints(colorSchemeLabel, gbcLabel); 136 selectionPanel.add(colorSchemeLabel); 137 138 colorSchemeComboBox = new JComboBox(); 139 for (String colorSchemeName : availableColorSchemes.keySet()) { 140 colorSchemeComboBox.addItem(colorSchemeName); 141 ColorScheme colorScheme = availableColorSchemes.get(colorSchemeName); 142 if (colorScheme.getClass().equals(preferences.getCurrentColorScheme().getClass())) { 143 colorSchemeComboBox.setSelectedItem(colorSchemeName); 144 } 145 } 146 colorSchemeComboBox.addActionListener(colorSchemeActionListener); 147 gbcComboBox.gridy = 2; 148 selectionLayout.setConstraints(colorSchemeComboBox, gbcComboBox); 149 selectionPanel.add(colorSchemeComboBox); 150 } 151 152 this.add(BorderLayout.CENTER, selectionPanel); 153 154 155 JPanel buttonPanel = new JPanel(); 156 JButton showLayerButton = new JButton("create/update graph"); 157 showLayerButton.addActionListener(new ActionListener() { 158 public void actionPerformed(ActionEvent e) { 159 plugin.createGraphViewLayer(); 160 } 161 }); 162 buttonPanel.add(showLayerButton); 163 164 this.add(BorderLayout.SOUTH, buttonPanel); 165 166 updateSelections(); 167 this.preferences.addObserver(this); 168 169 } 170 171 private final ActionListener rulesetActionListener = new ActionListener() { 172 public void actionPerformed(ActionEvent e) { 173 if (rulesetComboBox.getSelectedItem() != null) { 174 int selectedRulesetIndex = rulesetComboBox.getSelectedIndex(); 175 if (rulesetFiles != null) { 176 File selectedRulesetFile = rulesetFiles.get(selectedRulesetIndex); 177 preferences.setCurrentRulesetFile(selectedRulesetFile); 178 preferences.distributeChanges(); 179 plugin.updateGraphViewLayer(); 180 } else { 181 if (selectedRulesetIndex < InternalRuleset.values().length) { 182 InternalRuleset selectedIRR = InternalRuleset.values()[selectedRulesetIndex]; 183 preferences.setCurrentInternalRuleset(selectedIRR); 184 preferences.distributeChanges(); 185 plugin.updateGraphViewLayer(); 186 } 187 } 188 } 189 } 190 }; 191 192 private final ActionListener bookmarkActionListener = new ActionListener() { 193 public void actionPerformed(ActionEvent e) { 194 String selectedBookmarkName = (String)bookmarkComboBox.getSelectedItem(); 195 if (selectedBookmarkName != null) { 196 preferences.setCurrentParameterBookmarkName(selectedBookmarkName); 197 preferences.distributeChanges(); 198 plugin.updateGraphViewLayer(); 199 } 200 } 201 }; 202 203 private final ActionListener colorSchemeActionListener = new ActionListener() { 204 public void actionPerformed(ActionEvent e) { 205 assert availableColorSchemes.containsKey(colorSchemeComboBox.getSelectedItem()); 206 String colorSchemeLabel = (String)colorSchemeComboBox.getSelectedItem(); 207 preferences.setCurrentColorScheme(availableColorSchemes.get(colorSchemeLabel)); 208 preferences.distributeChanges(); 209 plugin.repaintGraphViewLayer(); 210 } 211 }; 212 213 public void update(Observable observable, Object param) { 214 if (observable == preferences) { 215 updateSelections(); 216 } 217 } 218 219 protected void updateSelections() { 220 221 /* update rulesets */ 222 223 rulesetComboBox.removeActionListener(rulesetActionListener); 224 225 if (preferences.getUseInternalRulesets()) { 226 227 rulesetFiles = null; 228 229 rulesetComboBox.removeAllItems(); 230 for (int i=0; i < InternalRuleset.values().length; i++) { 231 InternalRuleset ruleset = InternalRuleset.values()[i]; 232 rulesetComboBox.addItem(ruleset.toString()); 233 if (ruleset == preferences.getCurrentInternalRuleset()) { 234 rulesetComboBox.setSelectedIndex(i); 235 } 236 } 237 238 if (preferences.getCurrentInternalRuleset() == null) { 239 rulesetComboBox.addItem(""); 240 rulesetComboBox.setSelectedIndex(InternalRuleset.values().length); 241 } 242 243 } else { 244 245 rulesetFiles = new LinkedList<File>(); 246 247 File[] filesInRulesetFolder = preferences.getRulesetFolder().listFiles(); 248 249 if (filesInRulesetFolder != null) { 250 for (File possibleRulesetFile : filesInRulesetFolder) { 251 try { 252 AccessRulesetReader.readAccessRuleset(new FileInputStream(possibleRulesetFile)); 253 rulesetFiles.add(possibleRulesetFile); 254 } catch (IOException ioe) { 255 //don't add to rulesetFiles 256 } 257 } 258 } 259 260 Collections.sort(rulesetFiles); 261 262 rulesetComboBox.removeAllItems(); 263 for (int i=0; i < rulesetFiles.size(); i++) { 264 File rulesetFile = rulesetFiles.get(i); 265 rulesetComboBox.addItem(rulesetFile.getName()); 266 if (rulesetFile.equals(preferences.getCurrentRulesetFile())) { 267 rulesetComboBox.setSelectedIndex(i); 268 } 269 } 270 271 } 272 273 rulesetComboBox.addActionListener(rulesetActionListener); 274 275 /* update bookmarks */ 276 277 bookmarkComboBox.removeActionListener(bookmarkActionListener); 278 279 String activeBookmarkName = preferences.getCurrentParameterBookmarkName(); 280 Set<String> bookmarkNames = new HashSet<String>(preferences.getParameterBookmarks().keySet()); 281 282 bookmarkComboBox.removeAllItems(); 283 for (String bookmarkName : bookmarkNames) { 284 bookmarkComboBox.addItem(bookmarkName); 285 if (bookmarkName.equals(activeBookmarkName)) { 286 bookmarkComboBox.setSelectedItem(bookmarkName); 287 } 288 } 289 290 bookmarkComboBox.addActionListener(bookmarkActionListener); 291 292 } 293 293 294 294 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/dialogs/GraphViewPreferenceEditor.java
r19292 r23189 33 33 public class GraphViewPreferenceEditor implements PreferenceSetting { 34 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 35 private File rulesetFolder; 36 private Map<String, PreferenceAccessParameters> parameterBookmarks; 37 38 private JPanel preferencePanel; 39 40 private JCheckBox internalRulesetCheckBox; 41 private JLabel rulesetFolderLabel; 42 private JTextField rulesetFolderTextField; 43 private JButton selectRulesetFolderButton; 44 45 private JComboBox bookmarkComboBox; 46 private JButton editBookmarkButton; 47 private JButton deleteBookmarkButton; 48 49 private JCheckBox separateDirectionsCheckBox; 50 51 public void addGui(PreferenceTabbedPane gui) { 52 53 readPreferences(); 54 55 preferencePanel = gui.createPreferenceTab("graphview", "Graphview", 56 "Settings for the Graphview plugin that visualizes routing graphs."); 57 58 JPanel mainPanel = createMainPanel(); 59 60 preferencePanel.add(mainPanel, GBC.eol().fill(GBC.BOTH)); 61 62 updateVehiclePanel(GraphViewPreferences.getInstance().getCurrentParameterBookmarkName()); 63 64 } 65 66 /** 67 * creates local versions of preference information 68 * that will only be written to real preferences if the OK button is pressed 69 */ 70 private void readPreferences() { 71 72 GraphViewPreferences preferences = GraphViewPreferences.getInstance(); 73 74 rulesetFolder = preferences.getRulesetFolder(); 75 76 parameterBookmarks = 77 new HashMap<String, PreferenceAccessParameters>(preferences.getParameterBookmarks()); 78 79 } 80 81 private JPanel createMainPanel() { 82 83 JPanel mainPanel = new JPanel(); 84 85 GridBagLayout mainLayout = new GridBagLayout(); 86 mainPanel.setLayout(mainLayout); 87 88 GridBagConstraints constraints = new GridBagConstraints(); 89 constraints.fill = GridBagConstraints.HORIZONTAL; 90 constraints.weightx = 1; 91 constraints.gridx = 0; 92 93 { 94 JPanel rulesetPanel = createRulesetPanel(); 95 constraints.gridy = 0; 96 mainLayout.setConstraints(rulesetPanel, constraints); 97 mainPanel.add(rulesetPanel); 98 } { 99 JPanel vehiclePanel = createVehiclePanel(); 100 constraints.gridy = 1; 101 mainLayout.setConstraints(vehiclePanel, constraints); 102 mainPanel.add(vehiclePanel); 103 } { 104 JPanel visualizationPanel = createVisualizationPanel(); 105 constraints.gridy = 2; 106 mainLayout.setConstraints(visualizationPanel, constraints); 107 mainPanel.add(visualizationPanel); 108 } 109 110 mainPanel.add(GBC.glue(0, 0)); 111 112 return mainPanel; 113 114 } 115 116 private JPanel createRulesetPanel() { 117 118 JPanel rulesetPanel = new JPanel(); 119 rulesetPanel.setBorder(BorderFactory.createTitledBorder("ruleset")); 120 rulesetPanel.setLayout(new BoxLayout(rulesetPanel, BoxLayout.Y_AXIS)); 121 122 internalRulesetCheckBox = new JCheckBox("use built-in rulesets"); 123 internalRulesetCheckBox.setSelected(GraphViewPreferences.getInstance().getUseInternalRulesets()); 124 internalRulesetCheckBox.addActionListener(internalRulesetActionListener); 125 rulesetPanel.add(internalRulesetCheckBox); 126 127 rulesetFolderLabel = new JLabel("external ruleset directory:"); 128 rulesetPanel.add(rulesetFolderLabel); 129 130 rulesetFolderTextField = new JTextField(); 131 rulesetFolderTextField.setText(rulesetFolder.getPath()); 132 rulesetFolderTextField.setEditable(false); 133 rulesetPanel.add(rulesetFolderTextField); 134 135 selectRulesetFolderButton = new JButton("select directory"); 136 selectRulesetFolderButton.addActionListener(selectRulesetFolderActionListener); 137 rulesetPanel.add(selectRulesetFolderButton); 138 139 updateRulesetPanel(); 140 141 return rulesetPanel; 142 } 143 144 private JPanel createVehiclePanel() { 145 146 JPanel vehiclePanel = new JPanel(); 147 vehiclePanel.setBorder(BorderFactory.createTitledBorder("vehicle")); 148 vehiclePanel.setLayout(new BoxLayout(vehiclePanel, BoxLayout.Y_AXIS)); 149 150 bookmarkComboBox = new JComboBox(); 151 vehiclePanel.add(bookmarkComboBox); 152 153 JPanel buttonPanel = new JPanel(); 154 buttonPanel.setLayout(new BoxLayout(buttonPanel, BoxLayout.X_AXIS)); 155 156 JButton createButton = new JButton("create"); 157 createButton.addActionListener(createVehicleActionListener); 158 buttonPanel.add(createButton); 159 160 editBookmarkButton = new JButton("edit"); 161 editBookmarkButton.addActionListener(editVehicleActionListener); 162 buttonPanel.add(editBookmarkButton); 163 164 deleteBookmarkButton = new JButton("delete"); 165 deleteBookmarkButton.addActionListener(deleteVehicleActionListener); 166 buttonPanel.add(deleteBookmarkButton); 167 168 JButton restoreDefaultsButton = new JButton("restore defaults"); 169 restoreDefaultsButton.addActionListener(restoreVehicleDefaultsActionListener); 170 buttonPanel.add(restoreDefaultsButton); 171 172 vehiclePanel.add(buttonPanel); 173 174 return vehiclePanel; 175 } 176 177 private JPanel createVisualizationPanel() { 178 179 JPanel visualizationPanel = new JPanel(); 180 visualizationPanel.setBorder(BorderFactory.createTitledBorder("visualization")); 181 visualizationPanel.setLayout(new BoxLayout(visualizationPanel, BoxLayout.Y_AXIS)); 182 183 separateDirectionsCheckBox = new JCheckBox("draw directions separately"); 184 separateDirectionsCheckBox.setSelected(GraphViewPreferences.getInstance().getSeparateDirections()); 185 visualizationPanel.add(separateDirectionsCheckBox); 186 187 return visualizationPanel; 188 } 189 190 public boolean ok() { 191 192 GraphViewPreferences preferences = GraphViewPreferences.getInstance(); 193 194 preferences.setUseInternalRulesets(internalRulesetCheckBox.isSelected()); 195 preferences.setRulesetFolder(rulesetFolder); 196 197 preferences.setParameterBookmarks(parameterBookmarks); 198 199 String selectedBookmarkName = (String)bookmarkComboBox.getSelectedItem(); 200 preferences.setCurrentParameterBookmarkName(selectedBookmarkName); 201 202 preferences.setSeparateDirections(separateDirectionsCheckBox.isSelected()); 203 204 preferences.distributeChanges(); 205 206 return false; 207 } 208 209 private final ActionListener internalRulesetActionListener = new ActionListener() { 210 public void actionPerformed(ActionEvent e) { 211 updateRulesetPanel(); 212 } 213 }; 214 215 private final ActionListener selectRulesetFolderActionListener = new ActionListener() { 216 public void actionPerformed(ActionEvent e) { 217 218 File initialFCDirectory = rulesetFolder; 219 if (rulesetFolder.getParentFile() != null) { 220 initialFCDirectory = rulesetFolder.getParentFile(); 221 } 222 223 final JFileChooser fc = new JFileChooser(); 224 fc.setFileSelectionMode(JFileChooser.DIRECTORIES_ONLY); 225 fc.setCurrentDirectory(initialFCDirectory); 226 227 int returnVal = fc.showOpenDialog(preferencePanel); 228 229 if (returnVal == JFileChooser.APPROVE_OPTION) { 230 rulesetFolder = fc.getSelectedFile(); 231 rulesetFolderTextField.setText(rulesetFolder.getPath()); 232 } 233 234 } 235 }; 236 237 private final ActionListener createVehicleActionListener = new ActionListener() { 238 public void actionPerformed(ActionEvent e) { 239 240 PreferenceAccessParameters defaultBookmarkParameters = 241 GraphViewPreferenceDefaults.createDefaultBookmarkAccessParameters(); 242 243 AccessParameterDialog apd = new AccessParameterDialog( 244 null, 245 false, 246 "new bookmark", 247 parameterBookmarks.keySet(), 248 defaultBookmarkParameters, 249 new BookmarkAction() { 250 public void execute(String name, PreferenceAccessParameters parameters) { 251 parameterBookmarks.put(name, parameters); 252 updateVehiclePanel(name); 253 } 254 }); 255 256 apd.setVisible(true); 257 } 258 }; 259 260 private final ActionListener editVehicleActionListener = new ActionListener() { 261 public void actionPerformed(ActionEvent e) { 262 if (bookmarkComboBox.getSelectedItem() != null) { 263 264 final String selectedBookmarkName = (String)bookmarkComboBox.getSelectedItem(); 265 PreferenceAccessParameters parameters = 266 parameterBookmarks.get(selectedBookmarkName); 267 268 Collection<String> otherBookmarkNames = new LinkedList<String>(); 269 for (String bookmarkName : parameterBookmarks.keySet()) { 270 if (!bookmarkName.equals(selectedBookmarkName)) { 271 otherBookmarkNames.add(bookmarkName); 272 } 273 } 274 275 AccessParameterDialog apd = new AccessParameterDialog( 276 null, 277 true, 278 selectedBookmarkName, 279 otherBookmarkNames, 280 parameters, 281 new BookmarkAction() { 282 public void execute(String name, PreferenceAccessParameters parameters) { 283 parameterBookmarks.remove(selectedBookmarkName); 284 parameterBookmarks.put(name, parameters); 285 updateVehiclePanel(name); 286 } 287 }); 288 289 apd.setVisible(true); 290 } 291 292 } 293 }; 294 295 private final ActionListener deleteVehicleActionListener = new ActionListener() { 296 public void actionPerformed(ActionEvent e) { 297 if (bookmarkComboBox.getSelectedItem() != null) { 298 299 String selectedBookmarkName = (String)bookmarkComboBox.getSelectedItem(); 300 301 int userChoice = JOptionPane.showConfirmDialog( 302 preferencePanel, 303 "Really delete \"" + selectedBookmarkName + "\"?", 304 "Bookmark deletion", 305 JOptionPane.YES_NO_OPTION, JOptionPane.WARNING_MESSAGE); 306 307 if (userChoice == JOptionPane.YES_OPTION) { 308 parameterBookmarks.remove(selectedBookmarkName); 309 updateVehiclePanel(null); 310 } 311 312 } 313 } 314 }; 315 316 private final ActionListener restoreVehicleDefaultsActionListener = new ActionListener() { 317 public void actionPerformed(ActionEvent e) { 318 319 int userChoice = JOptionPane.showConfirmDialog( 320 preferencePanel, 321 "Really restore default bookmarks?\n" 322 + "All manually added or edited bookmarks will be lost!", 323 "Bookmark reset", 324 JOptionPane.YES_NO_OPTION, JOptionPane.WARNING_MESSAGE); 325 326 if (userChoice == JOptionPane.YES_OPTION) { 327 parameterBookmarks.clear(); 328 parameterBookmarks.putAll( 329 GraphViewPreferenceDefaults.createDefaultAccessParameterBookmarks()); 330 updateVehiclePanel(null); 331 } 332 333 } 334 }; 335 336 private void updateRulesetPanel() { 337 338 rulesetFolderLabel.setEnabled(!internalRulesetCheckBox.isSelected()); 339 rulesetFolderTextField.setEnabled(!internalRulesetCheckBox.isSelected()); 340 selectRulesetFolderButton.setEnabled(!internalRulesetCheckBox.isSelected()); 341 342 } 343 344 private void updateVehiclePanel(String selectedBookmarkName) { 345 346 bookmarkComboBox.removeAllItems(); 347 for (String bookmarkName : parameterBookmarks.keySet()) { 348 bookmarkComboBox.addItem(bookmarkName); 349 } 350 351 if (selectedBookmarkName == null) { 352 if (bookmarkComboBox.getItemCount() > 0) { 353 bookmarkComboBox.setSelectedIndex(0); 354 } 355 } else { 356 bookmarkComboBox.setSelectedItem(selectedBookmarkName); 357 } 358 359 editBookmarkButton.setEnabled(parameterBookmarks.size() > 0); 360 deleteBookmarkButton.setEnabled(parameterBookmarks.size() > 0); 361 362 } 363 363 364 364 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/layer/GraphViewLayer.java
r22547 r23189 46 46 public class GraphViewLayer extends Layer implements LayerChangeListener, WayGraphObserver { 47 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 48 private static final int NODE_RADIUS = 5; 49 50 /** 51 * offset from real position if {@link GraphViewPreferences#getSeparateDirections()} is active, 52 * causes the graph nodes for the two directions to be visually distinguishable. 53 * Positive values move "forward" direction to the right. 54 */ 55 private static final double DIRECTIONAL_OFFSET = 20; 56 57 private static final double OFFSET_ANGLE = 0.5 * Math.PI; 58 59 private static final double MIN_QUAD_DISTANCE_FOR_OFFSET = 4; 60 61 private static final boolean CONNECT_ALL_NODE_PAIRS = false; 62 63 /** an arrow head that points along the x-axis to (0,0) */ 64 private static final Shape ARROW_HEAD; 65 66 static { 67 68 Polygon head = new Polygon(); 69 70 head.addPoint( 0, 0); 71 head.addPoint(-15, +4); 72 head.addPoint(-15, -4); 73 74 ARROW_HEAD = head; 75 } 76 77 private WayGraph wayGraph = null; 78 79 private ColorScheme colorScheme = null; 80 private NodePositioner nodePositioner = new NonMovingNodePositioner(); 81 82 public GraphViewLayer() { 83 super("Graph view"); 84 MapView.addLayerChangeListener(this); 85 } 86 87 /** sets the WayGraph that is to be displayed by this layer, may be null */ 88 public void setWayGraph(WayGraph wayGraph) { 89 if (this.wayGraph != null) { 90 this.wayGraph.deleteObserver(this); 91 } 92 this.wayGraph = wayGraph; 93 if (wayGraph != null) { 94 wayGraph.addObserver(this); 95 } 96 } 97 98 /** sets the ColorScheme that is to be used for choosing colors, may be null */ 99 public void setColorScheme(ColorScheme colorScheme) { 100 this.colorScheme = colorScheme; 101 Main.panel.repaint(); 102 } 103 104 /** 105 * sets the NodePositioner that is to be used for determining node placement, 106 * null will cause a {@link NonMovingNodePositioner} to be used. 107 */ 108 public void setNodePositioner(NodePositioner nodePositioner) { 109 this.nodePositioner = nodePositioner; 110 if (nodePositioner == null) { 111 this.nodePositioner = new NonMovingNodePositioner(); 112 } 113 Main.panel.repaint(); 114 } 115 116 @Override 117 public Icon getIcon() { 118 return ImageProvider.get("layer", "graphview"); 119 } 120 121 private void paintGraphNode(final GraphNode node, final Graphics g, final MapView mv) { 122 123 Color color = colorScheme != null ? colorScheme.getNodeColor(node) : Color.LIGHT_GRAY; 124 125 Point p = getNodePoint(node, mv); 126 g.setColor(color); 127 g.fillOval(p.x - NODE_RADIUS, p.y - NODE_RADIUS, 2 * NODE_RADIUS, 2 * NODE_RADIUS); 128 129 } 130 131 private void paintGraphEdge(final GraphEdge e, final Graphics2D g2D, final MapView mv) { 132 133 if (!CONNECT_ALL_NODE_PAIRS && GraphViewPreferences.getInstance().getSeparateDirections()) { 134 135 //don't paint edges between nodes from the same SegmentNode and simply inverted Segment 136 if (e.getStartNode().getSegmentNode() == e.getTargetNode().getSegmentNode() 137 && e.getStartNode().getSegment().getNode2() == e.getTargetNode().getSegment().getNode1() 138 && e.getStartNode().getSegment().getNode1() == e.getTargetNode().getSegment().getNode2()) { 139 return; 140 } 141 142 } 143 144 /* draw line(s) */ 145 146 g2D.setStroke(new BasicStroke(3, BasicStroke.CAP_ROUND, BasicStroke.JOIN_ROUND)); 147 148 List<Segment> edgeSegments = e.getPropertyValue(GraphEdgeSegments.PROPERTY); 149 150 if (edgeSegments.size() > 0) { 151 152 Segment firstSegment = edgeSegments.get(0); 153 Segment lastSegment = edgeSegments.get(edgeSegments.size() - 1); 154 155 //draw segments 156 157 for (Segment segment : edgeSegments) { 158 159 Color color = Color.WHITE; 160 if (colorScheme != null) { 161 color = colorScheme.getSegmentColor(segment); 162 } 163 g2D.setColor(color); 164 165 Point p1 = getNodePoint(segment.getNode1(), mv); 166 Point p2 = getNodePoint(segment.getNode2(), mv); 167 168 if (segment == firstSegment) { 169 p1 = getNodePoint(e.getStartNode(), mv); 170 } 171 if (segment == lastSegment) { 172 p2 = getNodePoint(e.getTargetNode(), mv); 173 } 174 175 g2D.draw(new Line2D.Float(p1.x, p1.y, p2.x, p2.y)); 176 177 } 178 179 } else { 180 181 g2D.setColor(Color.WHITE); 182 183 Point p1 = getNodePoint(e.getStartNode(), mv); 184 Point p2 = getNodePoint(e.getTargetNode(), mv); 185 186 g2D.draw(new Line2D.Float(p1.x, p1.y, p2.x, p2.y)); 187 188 } 189 190 /* draw arrow head (note: color of last segment is still set) */ 191 192 { 193 Point p1 = getNodePoint(e.getStartNode(), mv); 194 Point p2 = getNodePoint(e.getTargetNode(), mv); 195 196 if (edgeSegments.size() > 0) { 197 Segment lastSegment = edgeSegments.get(edgeSegments.size() - 1); 198 p1 = getNodePoint(lastSegment.getNode1(), mv); 199 } 200 201 double angle = angleFromXAxis(p1, p2); // angle between x-axis and [p1,p2] 202 Shape head = ARROW_HEAD; 203 head = AffineTransform.getRotateInstance(angle).createTransformedShape(head); 204 head = AffineTransform.getTranslateInstance(p2.x, p2.y).createTransformedShape(head); 205 206 g2D.fill(head); 207 208 } 209 } 210 211 private Point getNodePoint(GraphNode node, MapView mv) { 212 213 Point nodePoint = getNodePoint(nodePositioner.getPosition(node), mv); 214 215 if (GraphViewPreferences.getInstance().getSeparateDirections() 216 && !GraphUtil.isEndNode(node)) { 217 218 SegmentNode node1 = node.getSegment().getNode1(); 219 SegmentNode node2 = node.getSegment().getNode2(); 220 221 Point node1Point = getNodePoint(node1, mv); 222 Point node2Point = getNodePoint(node2, mv); 223 224 double segmentX = node2Point.getX() - node1Point.getX(); 225 double segmentY = node2Point.getY() - node1Point.getY(); 226 227 if (segmentX*segmentX + segmentY*segmentY >= MIN_QUAD_DISTANCE_FOR_OFFSET) { 228 229 double rotatedX = Math.cos(OFFSET_ANGLE) * segmentX - Math.sin(OFFSET_ANGLE) * segmentY; 230 double rotatedY = Math.sin(OFFSET_ANGLE) * segmentX + Math.cos(OFFSET_ANGLE) * segmentY; 231 232 double segmentLength = Math.sqrt(rotatedX * rotatedX + rotatedY * rotatedY); 233 234 double normalizedX = rotatedX / segmentLength; 235 double normalizedY = rotatedY / segmentLength; 236 237 nodePoint.x += DIRECTIONAL_OFFSET * normalizedX; 238 nodePoint.y += DIRECTIONAL_OFFSET * normalizedY; 239 240 } 241 242 } 243 244 return nodePoint; 245 } 246 private Point getNodePoint(SegmentNode node, MapView mv) { 247 LatLonCoords coords = new LatLonCoords(node.getLat(), node.getLon()); 248 return getNodePoint(coords, mv); 249 } 250 private Point getNodePoint(LatLonCoords coords, MapView mv) { 251 LatLon latLon = new LatLon(coords.getLat(), coords.getLon()); 252 EastNorth eastNorth = Main.proj.latlon2eastNorth(latLon); 253 return mv.getPoint(eastNorth); 254 } 255 256 /** 257 * calculates the angle between the x axis and a vector given by two points 258 * @param p1 first point for vector; != null 259 * @param p2 second point for vector; != null 260 * @return angle in radians, in range [-Pi .. +Pi] 261 */ 262 private double angleFromXAxis(Point p1, Point p2) { 263 assert p1 != null && p2 != null; 264 265 final float vecX = p2.x - p1.x; 266 final float vecY = p2.y - p1.y; 267 268 final float vecLength = (float)Math.sqrt(vecX*vecX + vecY*vecY); 269 270 final float dotProductVecAxis = vecX; 271 272 float angle = (float)Math.acos(dotProductVecAxis / vecLength); 273 274 if (p2.y < p1.y) { 275 angle = -angle; 276 } 277 278 assert -Math.PI*0.5 < angle && angle <= Math.PI*0.5; 279 280 return angle; 281 } 282 283 @Override 284 public void paint(final Graphics2D g, final MapView mv, Bounds bounds) { 285 if (wayGraph != null) { 286 287 for (GraphNode n : wayGraph.getNodes()) { 288 paintGraphNode(n, g, mv); 289 } 290 291 for (GraphEdge e : wayGraph.getEdges()) { 292 paintGraphEdge(e, g, mv); 293 } 294 295 } 296 297 } 298 299 @Override 300 public String getToolTipText() { 301 return "routing graph calculated by the GraphView plugin"; 302 } 303 304 @Override 305 public void mergeFrom(Layer from) { 306 throw new AssertionError("GraphView layer is not mergable"); 307 } 308 309 @Override 310 public boolean isMergable(Layer other) { 311 return false; 312 } 313 314 @Override 315 public void visitBoundingBox(BoundingXYVisitor v) { 316 } 317 318 @Override 319 public Object getInfoComponent() { 320 return getToolTipText(); 321 } 322 323 @Override 324 public Action[] getMenuEntries() { 325 return new Action[] { 326 LayerListDialog.getInstance().createShowHideLayerAction(), 327 LayerListDialog.getInstance().createDeleteLayerAction(), 328 SeparatorLayerAction.INSTANCE, 329 new RenameLayerAction(null, this), 330 SeparatorLayerAction.INSTANCE, 331 new LayerListPopup.InfoAction(this)}; 332 } 333 334 public void update(WayGraph wayGraph) { 335 assert wayGraph == this.wayGraph; 336 Main.panel.repaint(); 337 } 338 339 public void activeLayerChange(Layer oldLayer, Layer newLayer) { 340 //do nothing 341 } 342 public void layerAdded(Layer newLayer) { 343 //do nothing 344 } 345 public void layerRemoved(Layer oldLayer) { 346 if (oldLayer == this) { 347 MapView.removeLayerChangeListener(this); 348 } 349 } 350 350 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/layer/PreferencesColorScheme.java
r16520 r23189 13 13 public class PreferencesColorScheme implements ColorScheme { 14 14 15 15 private final GraphViewPreferences preferences; 16 16 17 18 19 17 public PreferencesColorScheme(GraphViewPreferences preferences) { 18 this.preferences = preferences; 19 } 20 20 21 22 23 21 public Color getNodeColor(GraphNode node) { 22 return preferences.getNodeColor(); 23 } 24 24 25 26 27 25 public Color getSegmentColor(Segment segment) { 26 return preferences.getSegmentColor(); 27 } 28 28 29 29 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/preferences/GraphViewPreferenceDefaults.java
r16520 r23189 22 22 public final class GraphViewPreferenceDefaults { 23 23 24 25 24 /** prevents instantiation */ 25 private GraphViewPreferenceDefaults() { } 26 26 27 28 27 /** creates a default "empty" bookmark */ 28 public static PreferenceAccessParameters createDefaultBookmarkAccessParameters() { 29 29 30 31 30 Collection<AccessType> accessTypes = 31 Arrays.asList(UNDEFINED, YES, PERMISSIVE, DESIGNATED); 32 32 33 34 33 Map<VehiclePropertyType<?>, String> propertyStringMap = 34 new HashMap<VehiclePropertyType<?>, String>(); 35 35 36 37 38 39 40 36 try { 37 return new PreferenceAccessParameters("", accessTypes, propertyStringMap); 38 } catch (PropertyValueSyntaxException e) { 39 throw new AssertionError(e); 40 } 41 41 42 42 } 43 43 44 45 44 /** creates the default map of access parameter bookmarks */ 45 public static Map<String, PreferenceAccessParameters> createDefaultAccessParameterBookmarks() { 46 46 47 47 try { 48 48 49 50 49 Map<String, PreferenceAccessParameters> result = 50 new HashMap<String, PreferenceAccessParameters>(); 51 51 52 53 52 Collection<AccessType> accessTypes = 53 Arrays.asList(UNDEFINED, YES, PERMISSIVE, DESIGNATED); 54 54 55 56 57 58 55 /* create motorcar bookmark */ 56 { 57 Map<VehiclePropertyType<?>, String> propertyMap = 58 new HashMap<VehiclePropertyType<?>, String>(); 59 59 60 61 60 PreferenceAccessParameters accessParameters = 61 new PreferenceAccessParameters("motorcar", accessTypes, propertyMap); 62 62 63 64 63 result.put("motorcar", accessParameters); 64 } 65 65 66 67 68 69 70 66 /* create hgv bookmark */ 67 { 68 Map<VehiclePropertyType<?>, String> propertyMap = 69 new HashMap<VehiclePropertyType<?>, String>(); 70 propertyMap.put(VehiclePropertyTypes.WEIGHT, "3.5"); 71 71 72 73 72 PreferenceAccessParameters accessParameters = 73 new PreferenceAccessParameters("hgv", accessTypes, propertyMap); 74 74 75 76 75 result.put("hgv (3.5 t)", accessParameters); 76 } 77 77 78 79 80 81 78 /* create bicycle bookmark */ 79 { 80 Map<VehiclePropertyType<?>, String> propertyMap = 81 new HashMap<VehiclePropertyType<?>, String>(); 82 82 83 84 83 PreferenceAccessParameters accessParameters = 84 new PreferenceAccessParameters("bicycle", accessTypes, propertyMap); 85 85 86 87 86 result.put("bicycle", accessParameters); 87 } 88 88 89 90 91 92 89 /* create pedestrian bookmark */ 90 { 91 Map<VehiclePropertyType<?>, String> propertyMap = 92 new HashMap<VehiclePropertyType<?>, String>(); 93 93 94 95 94 PreferenceAccessParameters accessParameters = 95 new PreferenceAccessParameters("foot", accessTypes, propertyMap); 96 96 97 98 97 result.put("pedestrian", accessParameters); 98 } 99 99 100 100 return result; 101 101 102 103 104 105 102 } catch (PropertyValueSyntaxException e) { 103 throw new AssertionError(e); 104 } 105 } 106 106 107 108 109 107 public static File getDefaultRulesetFolder() { 108 return new File(System.getProperty("user.home")); 109 } 110 110 111 111 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/preferences/GraphViewPreferences.java
r16520 r23189 39 39 public class GraphViewPreferences extends Observable { 40 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 41 private static GraphViewPreferences instance; 42 43 /** 44 * returns the single instance of GraphViewPreferences. 45 * @param ignoreSyntaxErrors 46 * @return 47 */ 48 public static GraphViewPreferences getInstance() { 49 if (instance == null) { 50 instance = new GraphViewPreferences(); 51 } 52 return instance; 53 } 54 55 private boolean useInternalRulesets; 56 private File rulesetFolder; 57 private File currentRulesetFile; 58 private InternalRuleset currentInternalRuleset; 59 60 private String currentParameterBookmarkName; 61 private Map<String, PreferenceAccessParameters> parameterBookmarks; 62 63 private ColorScheme currentColorScheme; 64 private Color nodeColor; 65 private Color segmentColor; 66 67 private boolean separateDirections; 68 69 70 public synchronized boolean getUseInternalRulesets() { 71 return useInternalRulesets; 72 } 73 public synchronized void setUseInternalRulesets(boolean useInternalRulesets) { 74 this.useInternalRulesets = useInternalRulesets; 75 } 76 77 public synchronized File getRulesetFolder() { 78 return rulesetFolder; 79 } 80 public synchronized void setRulesetFolder(File rulesetFolder) { 81 this.rulesetFolder = rulesetFolder; 82 } 83 84 public synchronized File getCurrentRulesetFile() { 85 return currentRulesetFile; 86 } 87 public synchronized void setCurrentRulesetFile(File currentRulesetFile) { 88 this.currentRulesetFile = currentRulesetFile; 89 } 90 91 public synchronized InternalRuleset getCurrentInternalRuleset() { 92 return currentInternalRuleset; 93 } 94 public synchronized void setCurrentInternalRuleset(InternalRuleset internalRuleset) { 95 this.currentInternalRuleset = internalRuleset; 96 } 97 98 /** 99 * returns the name (map key) of the currently selected parameter bookmark 100 * or null if none is selected. 101 * If a name is returned, is has to be a key of the map returned by 102 * {@link #getParameterBookmarks()}. 103 */ 104 public synchronized String getCurrentParameterBookmarkName() { 105 assert parameterBookmarks.containsKey(currentParameterBookmarkName); 106 return currentParameterBookmarkName; 107 } 108 109 /** 110 * returns the access parameters of the currently selected parameter bookmark 111 * or null if none is selected. 112 */ 113 public synchronized AccessParameters getCurrentParameterBookmark() { 114 if (currentParameterBookmarkName == null) { 115 return null; 116 } else { 117 assert parameterBookmarks.containsKey(currentParameterBookmarkName); 118 return parameterBookmarks.get(currentParameterBookmarkName); 119 } 120 } 121 122 /** 123 * sets the active parameter bookmark using its name as an identifier 124 * @param currentParameters name of bookmark to set or null (no active bookmark). 125 * Non-null values must be keys of the map returned by 126 * {@link #getParameterBookmarks()}. 127 */ 128 public synchronized void setCurrentParameterBookmarkName(String parameterBookmarkName) { 129 assert parameterBookmarks.containsKey(parameterBookmarkName); 130 this.currentParameterBookmarkName = parameterBookmarkName; 131 } 132 133 public synchronized Map<String, PreferenceAccessParameters> getParameterBookmarks() { 134 return Collections.unmodifiableMap(parameterBookmarks); 135 } 136 public synchronized void setParameterBookmarks( 137 Map<String, PreferenceAccessParameters> parameterBookmarks) { 138 assert parameterBookmarks != null; 139 140 this.parameterBookmarks = 141 new HashMap<String, PreferenceAccessParameters>(parameterBookmarks); 142 } 143 144 public synchronized ColorScheme getCurrentColorScheme() { 145 return currentColorScheme; 146 } 147 public synchronized void setCurrentColorScheme(ColorScheme currentColorScheme) { 148 this.currentColorScheme = currentColorScheme; 149 } 150 151 public synchronized Color getNodeColor() { 152 return nodeColor; 153 } 154 public synchronized void setNodeColor(Color nodeColor) { 155 this.nodeColor = nodeColor; 156 } 157 158 public synchronized Color getSegmentColor() { 159 return segmentColor; 160 } 161 public synchronized void setSegmentColor(Color segmentColor) { 162 this.segmentColor = segmentColor; 163 } 164 165 public synchronized boolean getSeparateDirections() { 166 return separateDirections; 167 } 168 public synchronized void setSeparateDirections(boolean separateDirections) { 169 this.separateDirections = separateDirections; 170 } 171 172 /** 173 * writes changes to JOSM's preferences and notifies observers. 174 * Must be called explicitly after setters (to prevent distributing incomplete changes). 175 */ 176 public void distributeChanges() { 177 writePreferences(); 178 setChanged(); 179 notifyObservers(); 180 } 181 182 private GraphViewPreferences() { 183 184 /* set defaults first (in case preferences are incomplete) */ 185 186 fillDefaults(); 187 188 /* read preferences and overwrite defaults */ 189 190 readPreferences(); 191 192 /* write preferences 193 * (this will restore missing/defect preferences, 194 * but will simply rewrite valid preferences) */ 195 196 writePreferences(); 197 198 } 199 200 private void fillDefaults() { 201 202 parameterBookmarks = GraphViewPreferenceDefaults.createDefaultAccessParameterBookmarks(); 203 204 if (parameterBookmarks.size() > 0) { 205 currentParameterBookmarkName = parameterBookmarks.keySet().iterator().next(); 206 } else { 207 currentParameterBookmarkName = null; 208 } 209 210 useInternalRulesets = true; 211 rulesetFolder = GraphViewPreferenceDefaults.getDefaultRulesetFolder(); 212 currentRulesetFile = null; 213 currentInternalRuleset = null; 214 215 nodeColor = Color.WHITE; 216 segmentColor = Color.WHITE; 217 currentColorScheme = new PreferencesColorScheme(this); 218 219 separateDirections = false; 220 221 } 222 223 private void writePreferences() { 224 225 Main.pref.put("graphview.parameterBookmarks", 226 createAccessParameterBookmarksString(parameterBookmarks)); 227 228 if (currentParameterBookmarkName != null) { 229 Main.pref.put("graphview.activeBookmark", currentParameterBookmarkName); 230 } 231 232 Main.pref.put("graphview.useInternalRulesets", useInternalRulesets); 233 234 Main.pref.put("graphview.rulesetFolder", rulesetFolder.getPath()); 235 236 if (currentRulesetFile != null) { 237 Main.pref.put("graphview.rulesetFile", currentRulesetFile.getPath()); 238 } 239 if (currentInternalRuleset != null) { 240 Main.pref.put("graphview.rulesetResource", currentInternalRuleset.toString()); 241 } 242 243 Main.pref.put("graphview.defaultNodeColor", createColorString(nodeColor)); 244 Main.pref.put("graphview.defaultSegmentColor", createColorString(segmentColor)); 245 246 Main.pref.put("graphview.separateDirections", separateDirections); 247 248 } 249 250 private void readPreferences() { 251 252 if (Main.pref.hasKey("graphview.parameterBookmarks")) { 253 String bookmarksString = Main.pref.get("graphview.parameterBookmarks"); 254 parameterBookmarks = parseAccessParameterBookmarksString(bookmarksString); 255 } 256 257 if (Main.pref.hasKey("graphview.activeBookmark")) { 258 currentParameterBookmarkName = Main.pref.get("graphview.activeBookmark"); 259 } 260 if (!parameterBookmarks.containsKey(currentParameterBookmarkName)) { 261 currentParameterBookmarkName = null; 262 } 263 264 265 useInternalRulesets = Main.pref.getBoolean("graphview.useInternalRulesets", true); 266 267 if (Main.pref.hasKey("graphview.rulesetFolder")) { 268 String dirString = Main.pref.get("graphview.rulesetFolder"); 269 rulesetFolder = new File(dirString); 270 } 271 if (Main.pref.hasKey("graphview.rulesetFile")) { 272 String fileString = Main.pref.get("graphview.rulesetFile"); 273 currentRulesetFile = new File(fileString); 274 } 275 276 if (Main.pref.hasKey("graphview.rulesetResource")) { 277 String rulesetString = Main.pref.get("graphview.rulesetResource"); 278 //get the enum value for the string 279 //(InternalRuleset.valueOf cannot be used because it cannot handle invalid strings well) 280 for (InternalRuleset ruleset : InternalRuleset.values()) { 281 if (ruleset.toString().equals(rulesetString)) { 282 currentInternalRuleset = ruleset; 283 break; 284 } 285 } 286 } 287 288 if (Main.pref.hasKey("graphview.defaultNodeColor")) { 289 Color color = parseColorString(Main.pref.get("graphview.defaultNodeColor")); 290 if (color != null) { 291 nodeColor = color; 292 } 293 } 294 if (Main.pref.hasKey("graphview.defaultSegmentColor")) { 295 Color color = parseColorString(Main.pref.get("graphview.defaultSegmentColor")); 296 if (color != null) { 297 segmentColor = color; 298 } 299 } 300 301 separateDirections = Main.pref.getBoolean("graphview.separateDirections", false); 302 303 } 304 305 private static final Pattern ACCESS_PARAM_PATTERN = Pattern.compile("^([^;]*);([^;]*);types=\\{([^\\}]*)\\};properties=\\{([^\\}]*)\\}$"); 306 307 private static final Pattern PROPERTY_MAP_ENTRY_PATTERN = Pattern.compile("^([^=]*)=(.*)$"); 308 309 private static final Map<VehiclePropertyType<?>, String> VEHICLE_PROPERTY_TYPE_NAME_MAP = 310 new HashMap<VehiclePropertyType<?>, String>(); 311 312 313 static { 314 VEHICLE_PROPERTY_TYPE_NAME_MAP.put(AXLELOAD, "AXLELOAD"); 315 VEHICLE_PROPERTY_TYPE_NAME_MAP.put(HEIGHT, "HEIGHT"); 316 VEHICLE_PROPERTY_TYPE_NAME_MAP.put(LENGTH, "LENGTH"); 317 VEHICLE_PROPERTY_TYPE_NAME_MAP.put(MAX_INCLINE_DOWN, "MAX_INCLINE_DOWN"); 318 VEHICLE_PROPERTY_TYPE_NAME_MAP.put(MAX_INCLINE_UP, "MAX_INCLINE_UP"); 319 VEHICLE_PROPERTY_TYPE_NAME_MAP.put(MAX_TRACKTYPE, "MAX_TRACKTYPE"); 320 VEHICLE_PROPERTY_TYPE_NAME_MAP.put(SPEED, "SPEED"); 321 VEHICLE_PROPERTY_TYPE_NAME_MAP.put(SURFACE_BLACKLIST, "SURFACE_BLACKLIST"); 322 VEHICLE_PROPERTY_TYPE_NAME_MAP.put(WEIGHT, "WEIGHT"); 323 VEHICLE_PROPERTY_TYPE_NAME_MAP.put(WIDTH, "WIDTH"); 324 } 325 326 private static String createAccessParameterBookmarksString( 327 Map<String, PreferenceAccessParameters> parameterBookmarks) { 328 329 StringBuilder stringBuilder = new StringBuilder(); 330 331 boolean firstEntry = true; 332 333 for (String bookmarkName : parameterBookmarks.keySet()) { 334 335 if (!firstEntry) { 336 stringBuilder.append("|"); 337 } else { 338 firstEntry = false; 339 } 340 341 stringBuilder.append(createAccessParameterBookmarkString( 342 bookmarkName, 343 parameterBookmarks.get(bookmarkName))); 344 345 } 346 347 return stringBuilder.toString(); 348 } 349 350 private static String createAccessParameterBookmarkString( 351 String bookmarkName, PreferenceAccessParameters parameters) { 352 353 StringBuilder stringBuilder = new StringBuilder(); 354 355 stringBuilder.append(bookmarkName).append(";"); 356 357 stringBuilder.append(parameters.getAccessClass()); 358 359 stringBuilder.append(";types={"); 360 for (AccessType accessType : AccessType.values()) { 361 if (parameters.getAccessTypeUsable(accessType)) { 362 stringBuilder.append(accessType).append(","); 363 } 364 } 365 366 if(stringBuilder.charAt(stringBuilder.length()-1) == ',') { 367 stringBuilder.deleteCharAt(stringBuilder.length()-1); 368 } 369 stringBuilder.append("}"); 370 371 stringBuilder.append(";properties={"); 372 373 for (VehiclePropertyType<?> vehiclePropertyType : VEHICLE_PROPERTY_TYPE_NAME_MAP.keySet()) { 374 String propertyString = parameters.getVehiclePropertyString(vehiclePropertyType); 375 if (propertyString != null) { 376 stringBuilder.append(VEHICLE_PROPERTY_TYPE_NAME_MAP.get(vehiclePropertyType)); 377 stringBuilder.append("="); 378 stringBuilder.append(propertyString); 379 stringBuilder.append(","); 380 } 381 } 382 383 if(stringBuilder.charAt(stringBuilder.length()-1) == ',') { 384 stringBuilder.deleteCharAt(stringBuilder.length()-1); 385 } 386 stringBuilder.append("}"); 387 388 assert ACCESS_PARAM_PATTERN.matcher(stringBuilder.toString()).matches(); 389 390 return stringBuilder.toString(); 391 } 392 393 private static Map<String, PreferenceAccessParameters> parseAccessParameterBookmarksString( 394 String string) { 395 396 Map<String, PreferenceAccessParameters> resultMap = 397 new HashMap<String, PreferenceAccessParameters>(); 398 399 String[] bookmarkStrings = string.split("\\|"); 400 401 for (String bookmarkString : bookmarkStrings) { 402 parseAccessParameterBookmarkString(bookmarkString, resultMap); 403 } 404 405 return resultMap; 406 } 407 408 private static void parseAccessParameterBookmarkString(String bookmarkString, 409 Map<String, PreferenceAccessParameters> resultMap) { 410 411 Matcher matcher = ACCESS_PARAM_PATTERN.matcher(bookmarkString); 412 413 if (matcher.matches()) { 414 415 String bookmarkName = matcher.group(1); 416 417 String accessClass = matcher.group(2); 418 419 String[] accessTypeStrings = matcher.group(3).split(","); 420 Collection<AccessType> accessTypes = new LinkedList<AccessType>(); 421 for (String accessTypeString : accessTypeStrings) { 422 AccessType accessType = AccessType.valueOf(accessTypeString); 423 if (accessType != null) { 424 accessTypes.add(accessType); 425 } 426 } 427 428 429 String[] vehiclePropertyStrings = matcher.group(4).split(","); 430 Map<VehiclePropertyType<?>, String> vehiclePropertyMap = 431 new HashMap<VehiclePropertyType<?>, String>(); 432 433 for (String vehiclePropertyString : vehiclePropertyStrings) { 434 435 Matcher entryMatcher = PROPERTY_MAP_ENTRY_PATTERN.matcher(vehiclePropertyString); 436 if (entryMatcher.matches()) { 437 438 String propertyTypeString = entryMatcher.group(1); 439 String propertyValueString = entryMatcher.group(2); 440 441 for (VehiclePropertyType<?> propertyType : 442 VEHICLE_PROPERTY_TYPE_NAME_MAP.keySet()) { 443 444 if (propertyTypeString.equals( 445 VEHICLE_PROPERTY_TYPE_NAME_MAP.get(propertyType))) { 446 447 vehiclePropertyMap.put(propertyType, propertyValueString); 448 449 } 450 451 } 452 453 } 454 455 } 456 457 try { 458 459 PreferenceAccessParameters accessParameters = 460 new PreferenceAccessParameters(accessClass, accessTypes, vehiclePropertyMap); 461 462 resultMap.put(bookmarkName, accessParameters); 463 464 } catch (PropertyValueSyntaxException e) { 465 //don't add bookmark 466 } 467 468 } 469 } 470 471 private static final Pattern COLOR_PATTERN = 472 Pattern.compile("^(\\d{1,3}),\\s*(\\d{1,3}),\\s*(\\d{1,3})$"); 473 private String createColorString(Color color) { 474 return color.getRed() + ", " + color.getGreen() + ", " + color.getBlue(); 475 } 476 477 private Color parseColorString(String string) { 478 Matcher matcher = COLOR_PATTERN.matcher(string); 479 if (!matcher.matches()) { 480 return null; 481 } else { 482 int r = Integer.parseInt(matcher.group(1)); 483 int g = Integer.parseInt(matcher.group(2)); 484 int b = Integer.parseInt(matcher.group(3)); 485 return new Color(r, g, b); 486 } 487 } 488 488 489 489 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/preferences/InternalRuleset.java
r16520 r23189 3 3 public enum InternalRuleset { 4 4 5 6 5 DEFAULT("files/accessRuleset.xml"), 6 GERMANY("files/accessRuleset_de.xml"); 7 7 8 9 10 11 12 13 14 8 private String resourceName; 9 private InternalRuleset(String resourceName) { 10 this.resourceName = resourceName; 11 } 12 public String getResourceName() { 13 return resourceName; 14 } 15 15 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/preferences/PreferenceAccessParameters.java
r16520 r23189 17 17 public class PreferenceAccessParameters implements AccessParameters { 18 18 19 20 21 22 19 private final String accessClass; 20 private final Map<AccessType, Boolean> accessTypeUsableMap; 21 private final Map<VehiclePropertyType<?>, String> vehiclePropertyStrings; 22 private final Map<VehiclePropertyType<?>, Object> vehiclePropertyValues; 23 23 24 25 26 24 public String getAccessClass() { 25 return accessClass; 26 } 27 27 28 29 30 31 28 public boolean getAccessTypeUsable(AccessType accessType) { 29 assert accessType != null; 30 return accessTypeUsableMap.get(accessType); 31 } 32 32 33 34 35 33 public Collection<VehiclePropertyType<?>> getAvailableVehicleProperties() { 34 return vehiclePropertyValues.keySet(); 35 } 36 36 37 38 39 40 41 42 43 44 45 46 47 37 /** 38 * returns the value for a vehicle property. 39 * 40 * @param <D> type of property value 41 * @param vehicleProperty property to get value for; != null 42 * @return value for vehicleProperty, null if no value is available. 43 * Guaranteed to be valid according to vehicleProperty's 44 * {@link VehiclePropertyType#isValidValue(Object)} method. 45 */ 46 public <D> D getVehiclePropertyValue(VehiclePropertyType<D> vehicleProperty) { 47 assert vehicleProperty != null; 48 48 49 50 51 52 49 @SuppressWarnings("unchecked") 50 D value = (D)vehiclePropertyValues.get(vehicleProperty); 51 return value; 52 } 53 53 54 55 56 57 58 59 60 61 54 /** 55 * returns the unparsed String for a vehicle property. 56 * 57 * @param vehicleProperty property to get String for; != null 58 * @return unparsed String, null if no value is available. 59 */ 60 public String getVehiclePropertyString(VehiclePropertyType<?> vehicleProperty) { 61 assert vehicleProperty != null; 62 62 63 64 63 return vehiclePropertyStrings.get(vehicleProperty); 64 } 65 65 66 67 68 69 70 71 72 73 74 75 76 77 66 /** 67 * @param vehiclePropertyStrings map from vehicle properties to string representations 68 * that will be parsed using {@link VehiclePropertyStringParser} 69 * to get the property values; != null 70 * 71 * @throws VehiclePropertyStringParser.PropertyValueSyntaxException 72 * if a String from vehiclePropertyStrings contains a syntax error 73 */ 74 public PreferenceAccessParameters(String accessClass, 75 Collection<AccessType> usableAccessTypes, 76 Map<VehiclePropertyType<?>, String> vehiclePropertyStrings) 77 throws VehiclePropertyStringParser.PropertyValueSyntaxException { 78 78 79 79 this.accessClass = accessClass; 80 80 81 82 83 84 81 accessTypeUsableMap = new EnumMap<AccessType, Boolean>(AccessType.class); 82 for (AccessType accessType : AccessType.values()) { 83 accessTypeUsableMap.put(accessType, usableAccessTypes.contains(accessType)); 84 } 85 85 86 86 /* check and use vehicle properties */ 87 87 88 89 88 this.vehiclePropertyStrings = Collections.unmodifiableMap( 89 new HashMap<VehiclePropertyType<?>, String>(vehiclePropertyStrings)); 90 90 91 92 93 94 95 96 97 91 this.vehiclePropertyValues = new HashMap<VehiclePropertyType<?>, Object>(); 92 for (VehiclePropertyType<?> vehiclePropertyType : vehiclePropertyStrings.keySet()) { 93 String propertyValueString = vehiclePropertyStrings.get(vehiclePropertyType); 94 Object propertyValue = VehiclePropertyStringParser.parsePropertyValue( 95 vehiclePropertyType, propertyValueString); 96 this.vehiclePropertyValues.put(vehiclePropertyType, propertyValue); 97 } 98 98 99 99 } 100 100 101 101 } -
applications/editors/josm/plugins/graphview/src/org/openstreetmap/josm/plugins/graphview/plugin/preferences/VehiclePropertyStringParser.java
r20243 r23189 16 16 public final class VehiclePropertyStringParser { 17 17 18 19 18 /** prevents instantiation */ 19 private VehiclePropertyStringParser() { } 20 20 21 22 23 24 25 26 27 28 29 30 21 /** 22 * Exception class for syntax errors in property value Strings, 23 * the message contains the reason using one of this utility class' public String constants. 24 */ 25 public static class PropertyValueSyntaxException extends Exception { 26 private static final long serialVersionUID = 1L; 27 public PropertyValueSyntaxException(String message) { 28 super(message); 29 } 30 } 31 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 32 public static final String ERROR_WEIGHT = 33 "Weights must be given as positive decimal numbers with unit \"t\" or without unit."; 34 public static final String ERROR_LENGTH = 35 "Lengths must be given as positive decimal numbers with unit \"m\", \"km\", \"mi\"" + 36 " or without unit.\nAlternatively, the format FEET' INCHES\" can be used."; 37 public static final String ERROR_SPEED = 38 "Speeds should be given as numbers without unit or " 39 + "as numbers followed by \"mph\"."; 40 public static final String ERROR_INCLINE = 41 "Inclines must be given as positive decimal numbers with followed by \"%\"."; 42 public static final String ERROR_TRACKTYPE = 43 "Tracktype grades must be given as integers between 0 and 5."; 44 public static final String ERROR_SURFACE = 45 "Surface values must not contain any of the following characters: ',' '{' '}' '=' '|"; 46 46 47 48 47 private static final List<Character> FORBIDDEN_SURFACE_CHARS = 48 Arrays.asList(',', '{', '}', '=', '|'); 49 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 50 /** 51 * returns the value represented by the propertyValueString 52 * 53 * @throws PropertyValueSyntaxException if the string has syntax errors that prevent parsing 54 * @throws InvalidParameterException if an unknown property type was passed 55 * 56 * @param propertyType type of the property; != null 57 * @param propertyValueString string to parse; != null 58 * @return property value; != null. 59 * Guaranteed to be valid according to propertyType's 60 * {@link VehiclePropertyType#isValidValue(Object)} method. 61 */ 62 public static final <V> V parsePropertyValue( 63 VehiclePropertyType<V> propertyType, String propertyValueString) 64 throws PropertyValueSyntaxException { 65 65 66 66 assert propertyType != null && propertyValueString != null; 67 67 68 69 68 if (propertyType == VehiclePropertyTypes.AXLELOAD 69 || propertyType == VehiclePropertyTypes.WEIGHT) { 70 70 71 72 73 74 75 76 77 78 71 Float value = ValueStringParser.parseWeight(propertyValueString); 72 if (value != null && propertyType.isValidValue(value)) { 73 @SuppressWarnings("unchecked") //V must be float because of propertyType condition 74 V result = (V)value; 75 return result; 76 } else { 77 throw new PropertyValueSyntaxException(ERROR_WEIGHT); 78 } 79 79 80 81 82 80 } else if (propertyType == VehiclePropertyTypes.HEIGHT 81 || propertyType == VehiclePropertyTypes.LENGTH 82 || propertyType == VehiclePropertyTypes.WIDTH) { 83 83 84 85 86 87 88 89 90 91 84 Float value = ValueStringParser.parseMeasure(propertyValueString); 85 if (value != null && propertyType.isValidValue(value)) { 86 @SuppressWarnings("unchecked") //V must be float because of propertyType condition 87 V result = (V)value; 88 return result; 89 } else { 90 throw new PropertyValueSyntaxException(ERROR_LENGTH); 91 } 92 92 93 93 } else if (propertyType == VehiclePropertyTypes.SPEED) { 94 94 95 96 97 98 99 100 101 102 95 Float value = ValueStringParser.parseSpeed(propertyValueString); 96 if (value != null && propertyType.isValidValue(value)) { 97 @SuppressWarnings("unchecked") //V must be float because of propertyType condition 98 V result = (V)value; 99 return result; 100 } else { 101 throw new PropertyValueSyntaxException(ERROR_SPEED); 102 } 103 103 104 105 104 } else if (propertyType == VehiclePropertyTypes.MAX_INCLINE_DOWN 105 || propertyType == VehiclePropertyTypes.MAX_INCLINE_UP) { 106 106 107 108 109 110 111 112 113 114 107 Float value = ValueStringParser.parseIncline(propertyValueString); 108 if (value != null && propertyType.isValidValue(value)) { 109 @SuppressWarnings("unchecked") //V must be float because of propertyType condition 110 V result = (V)value; 111 return result; 112 } else { 113 throw new PropertyValueSyntaxException(ERROR_INCLINE); 114 } 115 115 116 116 } else if (propertyType == VehiclePropertyTypes.MAX_TRACKTYPE) { 117 117 118 119 120 121 122 123 124 125 118 try { 119 int value = Integer.parseInt(propertyValueString); 120 if (0 <= value && value <= 5) { 121 @SuppressWarnings("unchecked") //V must be int because of propertyType condition 122 V result = (V)(Integer)value; 123 return result; 124 } 125 } catch (NumberFormatException e) {} 126 126 127 127 throw new PropertyValueSyntaxException(ERROR_TRACKTYPE); 128 128 129 129 } else if (propertyType == VehiclePropertyTypes.SURFACE_BLACKLIST) { 130 130 131 132 133 134 135 136 137 138 139 140 131 String[] surfaces = propertyValueString.split(";\\s*"); 132 Collection<String> surfaceBlacklist = new ArrayList<String>(surfaces.length); 133 for (String surface : surfaces) { 134 for (char nameChar : surface.toCharArray()) { 135 if (FORBIDDEN_SURFACE_CHARS.contains(nameChar)) { 136 throw new PropertyValueSyntaxException(ERROR_SURFACE); 137 } 138 } 139 surfaceBlacklist.add(surface); 140 } 141 141 142 143 144 142 @SuppressWarnings("unchecked") //V must be Collection because of propertyType condition 143 V result = (V)surfaceBlacklist; 144 return result; 145 145 146 147 148 146 } else { 147 throw new InvalidParameterException("unknown property type: " + propertyType); 148 } 149 149 150 150 } 151 151 152 152 } -
applications/editors/josm/plugins/graphview/test/org/openstreetmap/josm/plugins/graphview/core/FullGraphCreationTest.java
r19216 r23189 32 32 public class FullGraphCreationTest { 33 33 34 35 36 37 38 34 private static final AccessParameters ACCESS_PARAMS; 35 static { 36 Map<VehiclePropertyType<?>, String> vehiclePropertyValues = 37 new HashMap<VehiclePropertyType<?>, String>(); 38 vehiclePropertyValues.put(VehiclePropertyTypes.WIDTH, "3.0"); 39 39 40 41 42 43 44 45 46 47 48 40 try { 41 ACCESS_PARAMS = new PreferenceAccessParameters( 42 "test_vehicle", 43 Arrays.asList(AccessType.UNDEFINED), 44 vehiclePropertyValues); 45 } catch (PropertyValueSyntaxException e) { 46 throw new Error(e); 47 } 48 } 49 49 50 51 52 53 54 55 56 57 58 59 60 50 private static final AccessRuleset TEST_RULESET = new AccessRuleset() { 51 public java.util.List<String> getAccessHierarchyAncestors(String transportMode) { 52 return Arrays.asList(transportMode); 53 } 54 public java.util.Collection<Tag> getBaseTags() { 55 return Arrays.asList(new Tag("highway", "test")); 56 } 57 public java.util.List<Implication> getImplications() { 58 return new LinkedList<Implication>(); 59 } 60 }; 61 61 62 63 62 @Test 63 public void testTJunction() { 64 64 65 65 TestDataSource ds = new TestDataSource(); 66 66 67 68 69 70 67 TestNode nodeN = new TestNode(2, 1); 68 TestNode nodeW = new TestNode(1, 0); 69 TestNode nodeS = new TestNode(0, 1); 70 TestNode nodeC = new TestNode(1, 1); 71 71 72 72 ds.nodes.addAll(Arrays.asList(nodeN, nodeW, nodeS, nodeC)); 73 73 74 75 76 77 78 79 80 81 82 74 TestWay wayNC = new TestWay(); 75 wayNC.tags.put("highway", "test"); 76 wayNC.nodes.addAll(Arrays.asList(nodeN, nodeC)); 77 TestWay wayCS = new TestWay(); 78 wayCS.tags.put("highway", "test"); 79 wayCS.nodes.addAll(Arrays.asList(nodeC, nodeS)); 80 TestWay wayCW = new TestWay(); 81 wayCW.tags.put("highway", "test"); 82 wayCW.nodes.addAll(Arrays.asList(nodeC, nodeW)); 83 83 84 85 86 84 ds.ways.add(wayNC); 85 ds.ways.add(wayCS); 86 ds.ways.add(wayCW); 87 87 88 89 90 88 /* variant 1: no restrictions */ 89 { 90 TransitionStructure ts1 = createTestTransitionStructure(ds); 91 91 92 93 94 92 assertSame(4, size(ts1.getNodes())); 93 assertSame(6, size(ts1.getSegments())); 94 assertSame(0, size(ts1.getRestrictions())); 95 95 96 96 WayGraph graph1 = new TSBasedWayGraph(ts1); 97 97 98 99 100 101 102 103 104 98 assertSame(12, graph1.getNodes().size()); 99 assertSame(24, graph1.getEdges().size()); 100 } 101 /* variant 2: no left turn from S to W */ 102 { 103 ds.relations.add(createTurnRestrictionRelation(wayCS, nodeC, wayCW, "no_left_turn")); 104 TransitionStructure ts2 = createTestTransitionStructure(ds); 105 105 106 107 108 106 assertSame(4, size(ts2.getNodes())); 107 assertSame(6, size(ts2.getSegments())); 108 assertSame(1, size(ts2.getRestrictions())); 109 109 110 110 WayGraph graph2 = new TSBasedWayGraph(ts2); 111 111 112 113 114 112 assertSame(12, graph2.getNodes().size()); 113 assertSame(23, graph2.getEdges().size()); 114 } 115 115 116 116 } 117 117 118 119 118 @Test 119 public void testBarrier() { 120 120 121 121 TestDataSource ds = new TestDataSource(); 122 122 123 124 125 126 123 TestNode node1 = new TestNode(0, 1); 124 TestNode nodeB = new TestNode(0, 2); 125 nodeB.tags.put("width", "1"); 126 TestNode node2 = new TestNode(0, 3); 127 127 128 128 ds.nodes.addAll(Arrays.asList(node1, nodeB, node2)); 129 129 130 131 132 133 134 130 TestWay way = new TestWay(); 131 way.tags.put("highway", "test"); 132 way.tags.put("oneway", "yes"); 133 way.nodes.addAll(Arrays.asList(node1, nodeB, node2)); 134 ds.ways.add(way); 135 135 136 136 /* variant 1: no restrictions */ 137 137 138 138 TransitionStructure ts = createTestTransitionStructure(ds); 139 139 140 141 142 140 assertSame(3, size(ts.getNodes())); 141 assertSame(2, size(ts.getSegments())); 142 assertSame(1, size(ts.getRestrictions())); 143 143 144 144 WayGraph graph = new TSBasedWayGraph(ts); 145 145 146 147 146 assertSame(4, graph.getNodes().size()); 147 assertSame(2, graph.getEdges().size()); 148 148 149 149 } 150 150 151 152 153 154 155 156 157 158 159 160 151 private TestRelation createTurnRestrictionRelation( 152 TestWay from, TestNode via, TestWay to, String restriction) { 153 TestRelation resultRelation = new TestRelation(); 154 resultRelation.tags.put("type", "restriction"); 155 resultRelation.tags.put("restriction", restriction); 156 resultRelation.members.add(new TestRelationMember("from", from)); 157 resultRelation.members.add(new TestRelationMember("via", via)); 158 resultRelation.members.add(new TestRelationMember("to", to)); 159 return resultRelation; 160 } 161 161 162 162 private TransitionStructure createTestTransitionStructure(TestDataSource dataSource) { 163 163 164 165 164 LinkedList<RoadPropertyType<?>> properties = new LinkedList<RoadPropertyType<?>>(); 165 properties.add(new RoadWidth()); 166 166 167 168 169 167 return new GenericTransitionStructure<TestNode, TestWay, TestRelation, TestRelationMember>( 168 ACCESS_PARAMS, TEST_RULESET, dataSource, properties); 169 } 170 170 171 172 173 174 175 176 177 178 179 171 private static int size(Iterable<?> iterable) { 172 Iterator<?> iterator = iterable.iterator(); 173 int size = 0; 174 while (iterator.hasNext()) { 175 iterator.next(); 176 size ++; 177 } 178 return size; 179 } 180 180 181 181 } -
applications/editors/josm/plugins/graphview/test/org/openstreetmap/josm/plugins/graphview/core/TestDataSource.java
r19216 r23189 14 14 public class TestDataSource implements DataSource<TestDataSource.TestNode, TestDataSource.TestWay, TestDataSource.TestRelation, TestDataSource.TestRelationMember> { 15 15 16 17 18 16 public static class TestPrimitive { 17 public final Map<String, String> tags = new HashMap<String, String>(); 18 }; 19 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 20 public static class TestNode extends TestPrimitive { 21 public final double lat; 22 public final double lon; 23 public TestNode() { 24 this(0, 0); 25 } 26 public TestNode(double lat, double lon) { 27 this.lat = lat; 28 this.lon = lon; 29 } 30 @Override 31 public String toString() { 32 return "(" + lat + ", " + lon + "); " + tags; 33 } 34 } 35 35 36 37 38 39 40 41 42 36 public static class TestWay extends TestPrimitive { 37 public final List<TestNode> nodes = new LinkedList<TestNode>(); 38 @Override 39 public String toString() { 40 return nodes + "; " + tags; 41 } 42 } 43 43 44 45 46 47 48 49 50 44 public static class TestRelation extends TestPrimitive { 45 public final Collection<TestRelationMember> members = new LinkedList<TestRelationMember>(); 46 @Override 47 public String toString() { 48 return members + "; " + tags; 49 } 50 } 51 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 52 public static class TestRelationMember { 53 public final String role; 54 public final TestPrimitive member; 55 public TestRelationMember(String role, TestPrimitive member) { 56 this.role = role; 57 this.member = member; 58 } 59 public TestPrimitive getMember() { 60 return member; 61 } 62 public String getRole() { 63 return role; 64 } 65 @Override 66 public String toString() { 67 return role + "=" + member; 68 } 69 } 70 70 71 71 72 73 74 72 public final Collection<TestNode> nodes = new LinkedList<TestNode>(); 73 public final Collection<TestWay> ways = new LinkedList<TestWay>(); 74 public final Collection<TestRelation> relations = new LinkedList<TestRelation>(); 75 75 76 76 77 78 79 80 81 82 77 public double getLat(TestNode node) { 78 return node.lat; 79 } 80 public double getLon(TestNode node) { 81 return node.lon; 82 } 83 83 84 85 86 84 public Iterable<TestRelationMember> getMembers(TestRelation relation) { 85 return relation.members; 86 } 87 87 88 89 90 88 public Iterable<TestNode> getNodes() { 89 return nodes; 90 } 91 91 92 93 94 92 public Iterable<TestNode> getNodes(TestWay way) { 93 return way.nodes; 94 } 95 95 96 97 98 96 public Iterable<TestWay> getWays() { 97 return ways; 98 } 99 99 100 101 102 100 public Iterable<TestRelation> getRelations() { 101 return relations; 102 } 103 103 104 105 106 104 public TagGroup getTagsN(TestNode node) { 105 return new MapBasedTagGroup(node.tags); 106 } 107 107 108 109 110 108 public TagGroup getTagsW(TestWay way) { 109 return new MapBasedTagGroup(way.tags); 110 } 111 111 112 public TagGroup getTagsR(TestRelation relation) { 113 return new MapBasedTagGroup(relation.tags); 114 } 115 116 public Object getMember(TestRelationMember member) { 117 return member.getMember(); 118 } 119 120 public String getRole(TestRelationMember member) { 121 return member.getRole(); 122 } 123 124 public boolean isNMember(TestRelationMember member) { 125 return member.getMember() instanceof TestNode; 126 } 127 128 public boolean isWMember(TestRelationMember member) { 129 return member.getMember() instanceof TestWay; 130 } 131 132 public boolean isRMember(TestRelationMember member) { 133 return member.getMember() instanceof TestRelation; 134 } 112 public TagGroup getTagsR(TestRelation relation) { 113 return new MapBasedTagGroup(relation.tags); 114 } 135 115 136 void addObserver(DataSourceObserver observer) {137 // not needed for test 138 116 public Object getMember(TestRelationMember member) { 117 return member.getMember(); 118 } 139 119 140 public void deleteObserver(DataSourceObserver observer) { 141 // not needed for test 142 } 120 public String getRole(TestRelationMember member) { 121 return member.getRole(); 122 } 123 124 public boolean isNMember(TestRelationMember member) { 125 return member.getMember() instanceof TestNode; 126 } 127 128 public boolean isWMember(TestRelationMember member) { 129 return member.getMember() instanceof TestWay; 130 } 131 132 public boolean isRMember(TestRelationMember member) { 133 return member.getMember() instanceof TestRelation; 134 } 135 136 public void addObserver(DataSourceObserver observer) { 137 // not needed for test 138 } 139 140 public void deleteObserver(DataSourceObserver observer) { 141 // not needed for test 142 } 143 143 144 144 } -
applications/editors/josm/plugins/graphview/test/org/openstreetmap/josm/plugins/graphview/core/access/AccessRulesetReaderTest.java
r16520 r23189 22 22 public class AccessRulesetReaderTest { 23 23 24 25 24 @Test 25 public void testReadAccessRuleset_valid_classes() throws IOException { 26 26 27 28 29 27 InputStream is = new FileInputStream("plugins/graphview/test/files/accessRuleset_valid.xml"); 28 AccessRuleset ruleset = AccessRulesetReader.readAccessRuleset(is); 29 assertNotNull(ruleset); 30 30 31 31 32 32 assertEquals("vehicle", ruleset.getAccessHierarchyAncestors("vehicle").get(0)); 33 33 34 35 34 assertEquals("motor_vehicle", ruleset.getAccessHierarchyAncestors("motor_vehicle").get(0)); 35 assertEquals("vehicle", ruleset.getAccessHierarchyAncestors("motor_vehicle").get(1)); 36 36 37 38 39 37 assertEquals("bus", ruleset.getAccessHierarchyAncestors("bus").get(0)); 38 assertEquals("motor_vehicle", ruleset.getAccessHierarchyAncestors("bus").get(1)); 39 assertEquals("vehicle", ruleset.getAccessHierarchyAncestors("bus").get(2)); 40 40 41 42 41 assertEquals("bicycle", ruleset.getAccessHierarchyAncestors("bicycle").get(0)); 42 assertEquals("vehicle", ruleset.getAccessHierarchyAncestors("bicycle").get(1)); 43 43 44 44 assertFalse(ruleset.getAccessHierarchyAncestors("bus").contains("bicycle")); 45 45 46 46 assertSame(ruleset.getAccessHierarchyAncestors("boat").size(), 0); 47 47 48 48 } 49 49 50 51 50 @Test 51 public void testReadAccessRuleset_valid_basetags() throws IOException { 52 52 53 54 55 53 InputStream is = new FileInputStream("plugins/graphview/test/files/accessRuleset_valid.xml"); 54 AccessRuleset ruleset = AccessRulesetReader.readAccessRuleset(is); 55 assertNotNull(ruleset); 56 56 57 57 assertSame(2, ruleset.getBaseTags().size()); 58 58 59 60 61 62 59 assertTrue(ruleset.getBaseTags().contains(new Tag("highway", "residential"))); 60 assertTrue(ruleset.getBaseTags().contains(new Tag("highway", "cycleway"))); 61 assertFalse(ruleset.getBaseTags().contains(new Tag("building", "residential"))); 62 assertFalse(ruleset.getBaseTags().contains(new Tag("highway", "stop"))); 63 63 64 64 } 65 65 66 67 66 @Test 67 public void testReadAccessRuleset_valid_implications() throws IOException { 68 68 69 70 71 69 InputStream is = new FileInputStream("plugins/graphview/test/files/accessRuleset_valid.xml"); 70 AccessRuleset ruleset = AccessRulesetReader.readAccessRuleset(is); 71 assertNotNull(ruleset); 72 72 73 73 List<Implication> implications = ruleset.getImplications(); 74 74 75 75 assertSame(3, implications.size()); 76 76 77 78 79 80 81 77 TagGroup[] tagGroups = new TagGroup[4]; 78 tagGroups[0] = createTagGroup(new Tag("highway", "cycleway")); 79 tagGroups[1] = createTagGroup(new Tag("highway", "steps")); 80 tagGroups[2] = createTagGroup(new Tag("highway", "steps"), new Tag("escalator", "yes")); 81 tagGroups[3] = createTagGroup(new Tag("disused", "yes"), new Tag("construction", "no")); 82 82 83 84 85 86 87 83 for (Implication implication : implications) { 84 for (int i = 0; i < tagGroups.length; i++) { 85 tagGroups[i] = implication.apply(tagGroups[i]); 86 } 87 } 88 88 89 90 89 assertSame(2, tagGroups[0].size()); 90 assertTrue(tagGroups[0].contains(new Tag("bicycle", "designated"))); 91 91 92 93 92 assertSame(2, tagGroups[1].size()); 93 assertTrue(tagGroups[1].contains(new Tag("normal_steps", "yes"))); 94 94 95 96 95 assertSame(2, tagGroups[2].size()); 96 assertFalse(tagGroups[2].contains(new Tag("normal_steps", "yes"))); 97 97 98 99 100 98 assertSame(3, tagGroups[3].size()); 99 assertTrue(tagGroups[3].contains(new Tag("usable", "no"))); 100 } 101 101 102 103 104 105 106 107 108 102 private static TagGroup createTagGroup(Tag... tags) { 103 Map<String, String> tagMap = new HashMap<String, String>(); 104 for (Tag tag : tags) { 105 tagMap.put(tag.key, tag.value); 106 } 107 return new MapBasedTagGroup(tagMap); 108 } 109 109 110 110 } -
applications/editors/josm/plugins/graphview/test/org/openstreetmap/josm/plugins/graphview/core/property/RoadInclineTest.java
r16520 r23189 6 6 public class RoadInclineTest extends RoadPropertyTest { 7 7 8 9 8 private static void testIncline(Float expectedInclineForward, Float expectedInclineBackward, 9 String inclineString) { 10 10 11 12 13 14 11 testEvaluateW(new RoadIncline(), 12 expectedInclineForward, expectedInclineBackward, 13 new Tag("incline", inclineString)); 14 } 15 15 16 17 18 19 20 21 22 16 @Test 17 public void testEvaluate() { 18 testIncline(5f, -5f, "5 %"); 19 testIncline(9.5f, -9.5f, "9.5 %"); 20 testIncline(-2.5f, 2.5f, "-2.5%"); 21 testIncline(null, null, "steep"); 22 } 23 23 24 24 } -
applications/editors/josm/plugins/graphview/test/org/openstreetmap/josm/plugins/graphview/core/property/RoadMaxspeedTest.java
r21609 r23189 6 6 public class RoadMaxspeedTest extends RoadPropertyTest { 7 7 8 9 10 8 private static void testMaxspeed(float expectedMaxspeed, String maxspeedString) { 9 testEvaluateBoth(new RoadMaxspeed(), expectedMaxspeed, new Tag("maxspeed", maxspeedString)); 10 } 11 11 12 13 14 15 16 12 @Test 13 public void testEvaluate_numeric() { 14 testMaxspeed(30, "30"); 15 testMaxspeed(48.3f, "48.3"); 16 } 17 17 18 19 20 21 22 23 18 @Test 19 public void testEvaluate_kmh() { 20 testMaxspeed(50, "50 km/h"); 21 testMaxspeed(120, "120km/h"); 22 testMaxspeed(30, "30 km/h"); 23 } 24 24 25 26 27 28 29 30 25 @Test 26 public void testEvaluate_mph() { 27 testMaxspeed(72.42048f, "45 mph"); 28 testMaxspeed(64.373764f, "40mph"); 29 testMaxspeed(24.14016f, "15 mph"); 30 } 31 31 32 32 } -
applications/editors/josm/plugins/graphview/test/org/openstreetmap/josm/plugins/graphview/core/property/RoadPropertyTest.java
r16520 r23189 8 8 abstract public class RoadPropertyTest { 9 9 10 10 protected static <P> void testEvaluateW(RoadPropertyType<P> property, P expectedForward, P expectedBackward, Tag... wayTags) { 11 11 12 13 14 15 16 17 12 TestDataSource ds = new TestDataSource(); 13 TestDataSource.TestWay testWay = new TestDataSource.TestWay(); 14 for (Tag tag : wayTags) { 15 testWay.tags.put(tag.key, tag.value); 16 } 17 ds.ways.add(testWay); 18 18 19 20 19 assertEquals(expectedForward, property.evaluateW(testWay, true, null, ds)); 20 assertEquals(expectedBackward, property.evaluateW(testWay, false, null, ds)); 21 21 22 22 } 23 23 24 24 protected static <P> void testEvaluateN(RoadPropertyType<P> property, P expected, Tag... nodeTags) { 25 25 26 27 28 29 30 31 26 TestDataSource ds = new TestDataSource(); 27 TestDataSource.TestNode testNode = new TestDataSource.TestNode(); 28 for (Tag tag : nodeTags) { 29 testNode.tags.put(tag.key, tag.value); 30 } 31 ds.nodes.add(testNode); 32 32 33 33 RoadMaxspeed m = new RoadMaxspeed(); 34 34 35 35 assertEquals(expected, m.evaluateN(testNode, null, ds)); 36 36 37 37 } 38 38 39 40 41 42 39 protected static <P> void testEvaluateBoth(RoadPropertyType<P> property, P expected, Tag... nodeTags) { 40 testEvaluateW(property, expected, expected, nodeTags); 41 testEvaluateN(property, expected, nodeTags); 42 } 43 43 44 44 } -
applications/editors/josm/plugins/graphview/test/org/openstreetmap/josm/plugins/graphview/core/util/TagConditionLogicTest.java
r16520 r23189 15 15 public class TagConditionLogicTest { 16 16 17 18 17 TagGroup groupA; 18 TagGroup groupB; 19 19 20 21 22 23 24 25 26 20 @Before 21 public void setUp() { 22 Map<String, String> mapA = new HashMap<String, String>(); 23 mapA.put("key1", "value1"); 24 mapA.put("key2", "value2"); 25 mapA.put("key3", "value1"); 26 groupA = new MapBasedTagGroup(mapA); 27 27 28 29 30 31 32 28 Map<String, String> mapB = new HashMap<String, String>(); 29 mapB.put("key1", "value1"); 30 mapB.put("key4", "value4"); 31 groupB = new MapBasedTagGroup(mapB); 32 } 33 33 34 35 36 37 38 39 34 @Test 35 public void testTag() { 36 TagCondition condition = TagConditionLogic.tag(new Tag("key3", "value1")); 37 assertTrue(condition.matches(groupA)); 38 assertFalse(condition.matches(groupB)); 39 } 40 40 41 42 43 44 45 46 41 @Test 42 public void testKey() { 43 TagCondition condition = TagConditionLogic.key("key3"); 44 assertTrue(condition.matches(groupA)); 45 assertFalse(condition.matches(groupB)); 46 } 47 47 48 49 50 51 52 48 @Test 49 public void testAnd() { 50 TagCondition condition1 = TagConditionLogic.tag(new Tag("key2", "value2")); 51 TagCondition conditionAnd1a = TagConditionLogic.and(condition1); 52 TagCondition conditionAnd1b = TagConditionLogic.and(Arrays.asList(condition1)); 53 53 54 55 56 57 54 assertTrue(conditionAnd1a.matches(groupA)); 55 assertTrue(conditionAnd1b.matches(groupA)); 56 assertFalse(conditionAnd1a.matches(groupB)); 57 assertFalse(conditionAnd1b.matches(groupB)); 58 58 59 60 61 59 TagCondition condition2 = TagConditionLogic.tag(new Tag("key1", "value1")); 60 TagCondition conditionAnd2a = TagConditionLogic.and(condition1, condition2); 61 TagCondition conditionAnd2b = TagConditionLogic.and(Arrays.asList(condition1, condition2)); 62 62 63 64 65 66 63 assertTrue(conditionAnd2a.matches(groupA)); 64 assertTrue(conditionAnd2b.matches(groupA)); 65 assertFalse(conditionAnd2a.matches(groupB)); 66 assertFalse(conditionAnd2b.matches(groupB)); 67 67 68 69 70 68 TagCondition condition3 = TagConditionLogic.tag(new Tag("key4", "value4")); 69 TagCondition conditionAnd3a = TagConditionLogic.and(condition1, condition2, condition3); 70 TagCondition conditionAnd3b = TagConditionLogic.and(Arrays.asList(condition1, condition2, condition3)); 71 71 72 73 74 75 76 72 assertFalse(conditionAnd3a.matches(groupA)); 73 assertFalse(conditionAnd3b.matches(groupA)); 74 assertFalse(conditionAnd3a.matches(groupB)); 75 assertFalse(conditionAnd3b.matches(groupB)); 76 } 77 77 78 79 80 81 82 78 @Test 79 public void testOr() { 80 TagCondition condition1 = TagConditionLogic.tag(new Tag("key42", "value42")); 81 TagCondition conditionOr1a = TagConditionLogic.or(condition1); 82 TagCondition conditionOr1b = TagConditionLogic.or(Arrays.asList(condition1)); 83 83 84 85 86 87 84 assertFalse(conditionOr1a.matches(groupA)); 85 assertFalse(conditionOr1b.matches(groupA)); 86 assertFalse(conditionOr1a.matches(groupB)); 87 assertFalse(conditionOr1b.matches(groupB)); 88 88 89 90 91 89 TagCondition condition2 = TagConditionLogic.tag(new Tag("key3", "value1")); 90 TagCondition conditionOr2a = TagConditionLogic.or(condition1, condition2); 91 TagCondition conditionOr2b = TagConditionLogic.or(Arrays.asList(condition1, condition2)); 92 92 93 94 95 96 93 assertTrue(conditionOr2a.matches(groupA)); 94 assertTrue(conditionOr2b.matches(groupA)); 95 assertFalse(conditionOr2a.matches(groupB)); 96 assertFalse(conditionOr2b.matches(groupB)); 97 97 98 99 100 98 TagCondition condition3 = TagConditionLogic.tag(new Tag("key1", "value1")); 99 TagCondition conditionOr3a = TagConditionLogic.or(condition1, condition2, condition3); 100 TagCondition conditionOr3b = TagConditionLogic.or(Arrays.asList(condition1, condition2, condition3)); 101 101 102 103 104 105 106 102 assertTrue(conditionOr3a.matches(groupA)); 103 assertTrue(conditionOr3b.matches(groupA)); 104 assertTrue(conditionOr3a.matches(groupB)); 105 assertTrue(conditionOr3b.matches(groupB)); 106 } 107 107 108 109 110 111 112 113 108 @Test 109 public void testNot() { 110 TagCondition condition = TagConditionLogic.not(TagConditionLogic.key("key3")); 111 assertFalse(condition.matches(groupA)); 112 assertTrue(condition.matches(groupB)); 113 } 114 114 115 115 } -
applications/editors/josm/plugins/graphview/test/org/openstreetmap/josm/plugins/graphview/core/util/ValueStringParserTest.java
r20244 r23189 10 10 public class ValueStringParserTest { 11 11 12 12 /* speed */ 13 13 14 15 16 17 14 @Test 15 public void testParseSpeedDefault() { 16 assertClose(50, parseSpeed("50")); 17 } 18 18 19 20 21 22 23 19 @Test 20 public void testParseSpeedKmh() { 21 assertClose(30, parseSpeed("30 km/h")); 22 assertClose(100, parseSpeed("100km/h")); 23 } 24 24 25 26 27 28 29 25 @Test 26 public void testParseSpeedMph() { 27 assertClose(40.234f, parseSpeed("25mph")); 28 assertClose(40.234f, parseSpeed("25 mph")); 29 } 30 30 31 32 33 34 31 @Test 32 public void testParseSpeedInvalid() { 33 assertNull(parseSpeed("lightspeed")); 34 } 35 35 36 36 /* measure */ 37 37 38 39 40 41 38 @Test 39 public void testParseMeasureDefault() { 40 assertClose(3.5f, parseMeasure("3.5")); 41 } 42 42 43 44 45 46 47 43 @Test 44 public void testParseMeasureM() { 45 assertClose(2, parseMeasure("2m")); 46 assertClose(5.5f, parseMeasure("5.5 m")); 47 } 48 48 49 50 51 52 53 49 @Test 50 public void testParseMeasureKm() { 51 assertClose(1000, parseMeasure("1 km")); 52 assertClose(7200, parseMeasure("7.2km")); 53 } 54 54 55 56 57 58 55 @Test 56 public void testParseMeasureMi() { 57 assertClose(1609.344f, parseMeasure("1 mi")); 58 } 59 59 60 61 62 63 64 60 @Test 61 public void testParseMeasureFeetInches() { 62 assertClose(3.6576f, parseMeasure("12'0\"")); 63 assertClose(1.9812f, parseMeasure("6' 6\"")); 64 } 65 65 66 67 68 69 70 66 @Test 67 public void testParseMeasureInvalid() { 68 assertNull(parseMeasure("very long")); 69 assertNull(parseMeasure("6' 16\"")); 70 } 71 71 72 72 /* weight */ 73 73 74 75 76 77 74 @Test 75 public void testParseWeightDefault() { 76 assertClose(3.6f, parseWeight("3.6")); 77 } 78 78 79 80 81 82 83 79 @Test 80 public void testParseWeightT() { 81 assertClose(30, parseWeight("30t")); 82 assertClose(3.5f, parseWeight("3.5 t")); 83 } 84 84 85 86 87 88 85 @Test 86 public void testParseWeightInvalid() { 87 assertNull(parseWeight("heavy")); 88 } 89 89 90 91 92 93 94 90 private static final void assertClose(float expected, float actual) { 91 if (Math.abs(expected - actual) > 0.001) { 92 throw new AssertionError("expected " + expected + ", was " + actual); 93 } 94 } 95 95 96 96 } -
applications/editors/josm/plugins/graphview/test/org/openstreetmap/josm/plugins/graphview/core/visualisation/FloatPropertyColorSchemeTest.java
r16520 r23189 13 13 public class FloatPropertyColorSchemeTest { 14 14 15 15 private FloatPropertyColorScheme subject; 16 16 17 18 17 @Before 18 public void setUp() { 19 19 20 21 22 23 20 Map<Float, Color> colorMap = new HashMap<Float, Color>(); 21 colorMap.put( 5f, new Color( 42, 42, 42)); 22 colorMap.put(10f, new Color(100, 100, 100)); 23 colorMap.put(20f, new Color(200, 200, 200)); 24 24 25 26 25 subject = new FloatPropertyColorScheme(RoadMaxweight.class, colorMap, Color.RED); 26 } 27 27 28 29 30 31 32 28 @Test 29 public void testGetColorForValue_below() { 30 assertEquals(new Color(42, 42, 42), subject.getColorForValue(1f)); 31 assertEquals(new Color(42, 42, 42), subject.getColorForValue(5f)); 32 } 33 33 34 35 36 37 34 @Test 35 public void testGetColorForValue_above() { 36 assertEquals(new Color(200, 200, 200), subject.getColorForValue(25f)); 37 } 38 38 39 40 41 42 39 @Test 40 public void testGetColorForValue_value() { 41 assertEquals(new Color(100, 100, 100), subject.getColorForValue(10f)); 42 } 43 43 44 45 46 47 44 @Test 45 public void testGetColorForValue_interpolate() { 46 assertEquals(new Color(150, 150, 150), subject.getColorForValue(15f)); 47 } 48 48 49 49 }
Note:
See TracChangeset
for help on using the changeset viewer.