Changeset 35320 in osm for applications/viewer
- Timestamp:
- 2020-02-15T12:30:27+01:00 (5 years ago)
- Location:
- applications/viewer/jmapviewer
- Files:
-
- 3 edited
Legend:
- Unmodified
- Added
- Removed
-
applications/viewer/jmapviewer/src/org/openstreetmap/gui/jmapviewer/FeatureAdapter.java
r35019 r35320 22 22 public final class FeatureAdapter { 23 23 24 private static ApiKeyAdapter apiKeyAdapter = new DefaultApiKeyAdapter(); 24 25 private static BrowserAdapter browserAdapter = new DefaultBrowserAdapter(); 25 26 private static ImageAdapter imageAdapter = new DefaultImageAdapter(); … … 32 33 } 33 34 35 /** 36 * Provider of confidential API keys. 37 */ 38 @FunctionalInterface 39 public interface ApiKeyAdapter { 40 /** 41 * Retrieves the API key for the given imagery id. 42 * @param imageryId imagery id 43 * @return the API key for the given imagery id 44 */ 45 String retrieveApiKey(String imageryId); 46 } 47 48 /** 49 * Link browser. 50 */ 51 @FunctionalInterface 34 52 public interface BrowserAdapter { 53 /** 54 * Browses to a given link. 55 * @param url link 56 */ 35 57 void openLink(String url); 36 58 } 37 59 60 /** 61 * Translation support. 62 */ 38 63 public interface TranslationAdapter { 64 /** 65 * Translates some text for the current locale. 66 * <br> 67 * For example, <code>tr("JMapViewer''s default value is ''{0}''.", val)</code>. 68 * <br> 69 * @param text the text to translate. 70 * Must be a string literal. (No constants or local vars.) 71 * Can be broken over multiple lines. 72 * An apostrophe ' must be quoted by another apostrophe. 73 * @param objects the parameters for the string. 74 * Mark occurrences in {@code text} with <code>{0}</code>, <code>{1}</code>, ... 75 * @return the translated string. 76 */ 39 77 String tr(String text, Object... objects); 40 78 // TODO: more i18n functions 41 79 } 42 80 81 /** 82 * Logging support. 83 */ 84 @FunctionalInterface 43 85 public interface LoggingAdapter { 86 /** 87 * Retrieves a logger for the given name. 88 * @param name logger name 89 * @return logger for the given name 90 */ 44 91 Logger getLogger(String name); 45 92 } 46 93 94 /** 95 * Image provider. 96 */ 97 @FunctionalInterface 47 98 public interface ImageAdapter { 99 /** 100 * Returns a <code>BufferedImage</code> as the result of decoding a supplied <code>URL</code>. 101 * 102 * @param input a <code>URL</code> to read from. 103 * @param readMetadata if {@code true}, makes sure to read image metadata to detect transparency color for non translucent images, 104 * if any. 105 * Always considered {@code true} if {@code enforceTransparency} is also {@code true} 106 * @param enforceTransparency if {@code true}, makes sure to read image metadata and, if the image does not 107 * provide an alpha channel but defines a {@code TransparentColor} metadata node, that the resulting image 108 * has a transparency set to {@code TRANSLUCENT} and uses the correct transparent color. 109 * 110 * @return a <code>BufferedImage</code> containing the decoded contents of the input, or <code>null</code>. 111 * 112 * @throws IllegalArgumentException if <code>input</code> is <code>null</code>. 113 * @throws IOException if an error occurs during reading. 114 */ 48 115 BufferedImage read(URL input, boolean readMetadata, boolean enforceTransparency) throws IOException; 49 116 } … … 71 138 } 72 139 140 /** 141 * Registers API key adapter. 142 * @param apiKeyAdapter API key adapter 143 */ 144 public static void registerApiKeyAdapter(ApiKeyAdapter apiKeyAdapter) { 145 FeatureAdapter.apiKeyAdapter = Objects.requireNonNull(apiKeyAdapter); 146 } 147 148 /** 149 * Registers browser adapter. 150 * @param browserAdapter browser adapter 151 */ 73 152 public static void registerBrowserAdapter(BrowserAdapter browserAdapter) { 74 153 FeatureAdapter.browserAdapter = Objects.requireNonNull(browserAdapter); 75 154 } 76 155 156 /** 157 * Registers image adapter. 158 * @param imageAdapter image adapter 159 */ 77 160 public static void registerImageAdapter(ImageAdapter imageAdapter) { 78 161 FeatureAdapter.imageAdapter = Objects.requireNonNull(imageAdapter); 79 162 } 80 163 164 /** 165 * Registers translation adapter. 166 * @param translationAdapter translation adapter 167 */ 81 168 public static void registerTranslationAdapter(TranslationAdapter translationAdapter) { 82 169 FeatureAdapter.translationAdapter = Objects.requireNonNull(translationAdapter); 83 170 } 84 171 172 /** 173 * Registers logging adapter. 174 * @param loggingAdapter logging adapter 175 */ 85 176 public static void registerLoggingAdapter(LoggingAdapter loggingAdapter) { 86 177 FeatureAdapter.loggingAdapter = Objects.requireNonNull(loggingAdapter); … … 96 187 } 97 188 189 /** 190 * Retrieves the API key for the given imagery id using the current {@link ApiKeyAdapter}. 191 * @param imageryId imagery id 192 * @return the API key for the given imagery id 193 */ 194 public static String retrieveApiKey(String imageryId) { 195 return apiKeyAdapter.retrieveApiKey(imageryId); 196 } 197 198 /** 199 * Opens a link using the current {@link BrowserAdapter}. 200 * @param url link to open 201 */ 98 202 public static void openLink(String url) { 99 203 browserAdapter.openLink(url); 100 204 } 101 205 206 /** 207 * Reads an image using the current {@link ImageAdapter}. 208 * @param url image URL to read 209 * @return a <code>BufferedImage</code> containing the decoded contents of the input, or <code>null</code>. 210 * @throws IOException if an error occurs during reading. 211 */ 102 212 public static BufferedImage readImage(URL url) throws IOException { 103 213 return imageAdapter.read(url, false, false); 104 214 } 105 215 216 /** 217 * Translates a text using the current {@link TranslationAdapter}. 218 * @param text the text to translate. 219 * Must be a string literal. (No constants or local vars.) 220 * Can be broken over multiple lines. 221 * An apostrophe ' must be quoted by another apostrophe. 222 * @param objects the parameters for the string. 223 * Mark occurrences in {@code text} with <code>{0}</code>, <code>{1}</code>, ... 224 * @return the translated string. 225 */ 106 226 public static String tr(String text, Object... objects) { 107 227 return translationAdapter.tr(text, objects); 108 228 } 109 229 230 /** 231 * Returns a logger for the given name using the current {@link LoggingAdapter}. 232 * @param name logger name 233 * @return logger for the given name 234 */ 110 235 public static Logger getLogger(String name) { 111 236 return loggingAdapter.getLogger(name); 112 237 } 113 238 239 /** 240 * Returns a logger for the given class using the current {@link LoggingAdapter}. 241 * @param klass logger class 242 * @return logger for the given class 243 */ 114 244 public static Logger getLogger(Class<?> klass) { 115 245 return loggingAdapter.getLogger(klass.getSimpleName()); … … 148 278 } 149 279 280 /** 281 * Default API key support that relies on system property named {@code <imageryId>.api-key}. 282 */ 283 public static class DefaultApiKeyAdapter implements ApiKeyAdapter { 284 @Override 285 public String retrieveApiKey(String imageryId) { 286 return System.getProperty(imageryId + ".api-key"); 287 } 288 } 289 290 /** 291 * Default browser support that relies on Java Desktop API. 292 */ 150 293 public static class DefaultBrowserAdapter implements BrowserAdapter { 151 294 @Override … … 165 308 } 166 309 310 /** 311 * Default image support that relies on Java Image IO API. 312 */ 167 313 public static class DefaultImageAdapter implements ImageAdapter { 168 314 @Override … … 172 318 } 173 319 320 /** 321 * Default "translation" support that do not really translates strings, but only takes care of formatting arguments. 322 */ 174 323 public static class DefaultTranslationAdapter implements TranslationAdapter { 175 324 @Override … … 179 328 } 180 329 330 /** 331 * Default logging support that relies on Java Logging API. 332 */ 181 333 public static class DefaultLoggingAdapter implements LoggingAdapter { 182 334 @Override -
applications/viewer/jmapviewer/src/org/openstreetmap/gui/jmapviewer/tilesources/TemplatedTMSTileSource.java
r34715 r35320 5 5 import java.util.Map; 6 6 import java.util.Random; 7 import java.util.function.BiConsumer; 7 8 import java.util.regex.Matcher; 8 9 import java.util.regex.Pattern; 9 10 11 import org.openstreetmap.gui.jmapviewer.FeatureAdapter; 10 12 import org.openstreetmap.gui.jmapviewer.interfaces.TemplatedTileSource; 11 13 … … 25 27 * {!y} - substituted with Yahoo Y tile number 26 28 * {-y} - substituted with reversed Y tile number 29 * {apiKey} - substituted with API key retrieved for the imagery id 27 30 * {switch:VAL_A,VAL_B,VAL_C,...} - substituted with one of VAL_A, VAL_B, VAL_C. Usually 28 31 * used to specify many tile servers … … 46 49 private static final Pattern PATTERN_SWITCH = Pattern.compile("\\{switch:([^}]+)\\}"); 47 50 private static final Pattern PATTERN_HEADER = Pattern.compile("\\{header\\(([^,]+),([^}]+)\\)\\}"); 51 private static final Pattern PATTERN_API_KEY = Pattern.compile("\\{apiKey\\}"); 48 52 private static final Pattern PATTERN_PARAM = Pattern.compile("\\{((?:\\d+-)?z(?:oom)?(:?[+-]\\d+)?|x|y|!y|-y|switch:([^}]+))\\}"); 49 53 // CHECKSTYLE.ON: SingleSpaceSeparator 50 54 51 55 private static final Pattern[] ALL_PATTERNS = { 52 PATTERN_HEADER, PATTERN_ZOOM, PATTERN_X, PATTERN_Y, PATTERN_Y_YAHOO, PATTERN_NEG_Y, PATTERN_SWITCH 56 PATTERN_HEADER, PATTERN_ZOOM, PATTERN_X, PATTERN_Y, PATTERN_Y_YAHOO, PATTERN_NEG_Y, PATTERN_SWITCH, PATTERN_API_KEY 53 57 }; 54 58 … … 63 67 headers.put(COOKIE_HEADER, cookies); 64 68 } 65 handleTemplate(); 69 handleTemplate(info.getId()); 66 70 } 67 71 68 private void handleTemplate() { 72 private void replacePattern(Pattern p, BiConsumer<Matcher, StringBuffer> replaceAction) { 73 StringBuffer output = new StringBuffer(); 74 Matcher m = p.matcher(baseUrl); 75 while (m.find()) { 76 replaceAction.accept(m, output); 77 } 78 m.appendTail(output); 79 baseUrl = output.toString(); 80 } 81 82 private void handleTemplate(String imageryId) { 69 83 // Capturing group pattern on switch values 70 84 Matcher m = PATTERN_SWITCH.matcher(baseUrl); … … 73 87 randomParts = m.group(1).split(","); 74 88 } 75 StringBuffer output = new StringBuffer(); 76 Matcher matcher = PATTERN_HEADER.matcher(baseUrl); 77 while (matcher.find()) { 89 // Capturing group pattern on header values 90 replacePattern(PATTERN_HEADER, (matcher, output) -> { 78 91 headers.put(matcher.group(1), matcher.group(2)); 79 92 matcher.appendReplacement(output, ""); 80 } 81 matcher.appendTail(output); 82 baseUrl = output.toString(); 83 m = PATTERN_ZOOM.matcher(this.baseUrl); 93 }); 94 // Capturing group pattern on API key values 95 replacePattern(PATTERN_API_KEY, (matcher, output) -> 96 matcher.appendReplacement(output, FeatureAdapter.retrieveApiKey(imageryId)) 97 ); 98 // Capturing group pattern on zoom values 99 m = PATTERN_ZOOM.matcher(baseUrl); 84 100 if (m.find()) { 85 101 if (m.group(1) != null) { … … 94 110 } 95 111 } 96 97 112 } 98 113 -
applications/viewer/jmapviewer/test/org/openstreetmap/gui/jmapviewer/tilesources/TemplatedTMSTileSourceTest.java
r34716 r35320 1 // License: GPL. For details, see LICENSEfile.1 // License: GPL. For details, see Readme.txt file. 2 2 package org.openstreetmap.gui.jmapviewer.tilesources; 3 3 … … 13 13 import org.junit.Test; 14 14 15 16 15 /** 17 *18 16 * Tests for TemplaedTMSTileSource 19 17 */ 20 18 public class TemplatedTMSTileSourceTest { 21 19 22 private final staticCollection<String> TMS_IMAGERIES = Arrays.asList(new String[]{20 private static final Collection<String> TMS_IMAGERIES = Arrays.asList(new String[]{ 23 21 "http://imagico.de/map/osmim_tiles.php?layer=S2A_R136_N41_20150831T093006&z={zoom}&x={x}&y={-y}", 24 22 /* … … 36 34 * * expected tile url for zoom=3, x=2, y=1 37 35 */ 38 @SuppressWarnings("unchecked")39 36 private Collection<String[]> TEST_DATA = Arrays.asList(new String[][] { 40 37 /* 41 38 * generate with main method below once TMS_IMAGERIES is filled in 42 39 */ 43 new String[]{"http://imagico.de/map/osmim_tiles.php?layer=S2A_R136_N41_20150831T093006&z={zoom}&x={x}&y={-y}", 44 "http://imagico.de/map/osmim_tiles.php?layer=S2A_R136_N41_20150831T093006&z=1&x=2&y=-2", 45 "http://imagico.de/map/osmim_tiles.php?layer=S2A_R136_N41_20150831T093006&z=3&x=2&y=6" 46 } 40 new String[] { 41 "http://imagico.de/map/osmim_tiles.php?layer=S2A_R136_N41_20150831T093006&z={zoom}&x={x}&y={-y}", 42 "http://imagico.de/map/osmim_tiles.php?layer=S2A_R136_N41_20150831T093006&z=1&x=2&y=-2", 43 "http://imagico.de/map/osmim_tiles.php?layer=S2A_R136_N41_20150831T093006&z=3&x=2&y=6" 44 } 47 45 }); 48 46 … … 59 57 } 60 58 61 62 59 /** 63 60 * Check template with positive zoom index … … 106 103 "http://localhost/2/1/2" 107 104 ); 105 } 106 107 /** 108 * Test template with switch 109 */ 110 @Test 111 public void testGetTileUrl_apiKey() { 112 System.setProperty("id1.api-key", "wololo"); 113 TileSourceInfo testImageryTMS = new TileSourceInfo("test imagery", "http://localhost/{zoom}/{x}/{y}?token={apiKey}&foo=bar", "id1"); 114 TemplatedTMSTileSource ts = new TemplatedTMSTileSource(testImageryTMS); 115 assertEquals("http://localhost/1/2/3?token=wololo&foo=bar", ts.getTileUrl(1, 2, 3)); 108 116 } 109 117 … … 176 184 assertEquals(expected312, ts.getTileUrl(3, 1, 2)); 177 185 } 186 178 187 /** 179 188 * Tests all entries in TEST_DATA. This test will fail if {switch:...} template is used … … 181 190 @Test 182 191 public void testAllUrls() { 183 for(String[] test: TEST_DATA) { 192 for (String[] test: TEST_DATA) { 184 193 TileSourceInfo testImageryTMS = new TileSourceInfo("test imagery", test[0], "id1"); 185 194 TemplatedTMSTileSource ts = new TemplatedTMSTileSource(testImageryTMS); … … 190 199 191 200 public static void main(String[] args) { 192 for(String url: TMS_IMAGERIES) { 201 for (String url: TMS_IMAGERIES) { 193 202 TileSourceInfo testImageryTMS = new TileSourceInfo("test imagery", url, "id1"); 194 203 TemplatedTMSTileSource ts = new TemplatedTMSTileSource(testImageryTMS); 195 System.out.println(MessageFormat.format("new String[]{\"{0}\", \"{1}\", \"{2}\"},", url, ts.getTileUrl(1, 2, 3), ts.getTileUrl(3, 2, 1))); 204 System.out.println(MessageFormat.format("new String[]{\"{0}\", \"{1}\", \"{2}\"},", 205 url, ts.getTileUrl(1, 2, 3), ts.getTileUrl(3, 2, 1))); 196 206 } 197 207 } 198 199 208 }
Note:
See TracChangeset
for help on using the changeset viewer.