001/******************************************************************************* 002 * This software is provided as a supplement to the authors' textbooks on digital 003 * image processing published by Springer-Verlag in various languages and editions. 004 * Permission to use and distribute this software is granted under the BSD 2-Clause 005 * "Simplified" License (see http://opensource.org/licenses/BSD-2-Clause). 006 * Copyright (c) 2006-2023 Wilhelm Burger, Mark J. Burge. All rights reserved. 007 * Visit https://imagingbook.com for additional details. 008 ******************************************************************************/ 009package imagingbook.core.resource; 010 011import java.io.InputStream; 012import java.net.URL; 013 014/** 015 * <p> 016 * This interface is supposed to be implemented by some enum class that specifies the root of a resource tree. The 017 * following example defines such an enum type that refers to three images. The path relative to the location of the 018 * enum class itself must be specified and images must be stored there. The enum type must define an appropriate 019 * constructor and implement method {@link #getFileName()}: 020 * </p> 021 * <pre> 022 * public enum MyImages implements NamedResource { 023 * image1("image1.tif"), 024 * image2("image2.png"), 025 * image3("jpegs/image3.jpg"); 026 * // field to associate the resource's file name 027 * private final String filename; 028 * @Override 029 * public String getFileName() { 030 * return filename; 031 * } 032 * // constructor 033 * MyImages(String filename) { 034 * this.filename = filename; 035 * } 036 * }</pre> 037 * <p> 038 * By default, resource files are assumed to be placed in a folder with the same name as the resource class (i.e., 039 * "MyImages") relative to the location of the class. Resources may then be simply referenced by their enum-name, for 040 * example, 041 * </p> 042 * <pre> 043 * URL url = MyImages.image1.getURL(); 044 * ImagePlus im = IJ.openImage(url.toString()); 045 * </pre> 046 * <p> 047 * The benefit of using a named resource (over specifying resources by path strings) is that the resource is guaranteed 048 * to exist at compile time (this is assured by validating the existence of all named resources in the test phase) and 049 * can be retrieved even if located inside a JAR file. Also, everything is defined in a single place and renaming a 050 * resource is easy. 051 * </p> 052 * 053 * @author WB 054 * @version 2022/08/23 055 * @see ImageResource 056 */ 057public interface NamedResource { 058 059 /** 060 * Specifies the name of the resource directory relative to some resource class. For example, given some resource 061 * class in 062 * <pre>src/main/java/.../foo/ZeClass.java</pre> 063 * the associated resource files are assumed to be in directory 064 * <pre>src/main/resources/.../foo/ZeClass-data/</pre> 065 */ 066 public static final String RelativeDirectorySuffix = "-data"; 067 068 /** 069 * Returns the resource directory relative to the implementing class. The default implementations assumes that this 070 * is a single-level directory with exactly the same name as the implementing class, extended by 071 * {@link #RelativeDirectorySuffix} ("-data"). Implementations may override this definition to return some other 072 * relative directory. Note that the directory should not have a legal Java package name to avoid confusion in the 073 * build process (e.g., by javadoc). 074 * 075 * @return the relative resource directory for the associated resource 076 */ 077 public default String getRelativeDirectory() { 078 // return getClass().getSimpleName() + RelativeDirectorySuffix; 079 return getRelativeDirectory(this.getClass()); 080 } 081 082 public static String getRelativeDirectory(Class<? extends NamedResource> clazz) { 083 return clazz.getSimpleName() + RelativeDirectorySuffix; 084 } 085 086 /** 087 * Returns the path to the associated resource relative to the location of the implementing class. This method is 088 * not supposed to be overridden. 089 * 090 * @return the relative path to the associated resource 091 */ 092 public default String getRelativePath() { 093 return getRelativeDirectory() + "/" + getFileName(); 094 } 095 096 /** 097 * Returns the file name for the associated resource (to be implemented by concrete classes). 098 * 099 * @return the name of the resource file 100 */ 101 public String getFileName(); 102 103 /** 104 * Returns the URL to the associated resource. This method is not supposed to be overridden. Throws an exception if 105 * the requested resource does not exist. 106 * 107 * @return the URL to the associated resource 108 */ 109 public default URL getURL() { 110 Class<?> clazz = this.getClass(); 111 String relPath = this.getRelativePath(); 112 URL url = clazz.getResource(relPath); 113 if (url == null) { 114 throw new RuntimeException("could not find resource " + 115 clazz.getResource("") + relPath); 116 } 117 return url; 118 } 119 120 /** 121 * Returns true if the associated resource (class) was loaded from a JAR file. 122 * 123 * @return true if inside a JAR file 124 */ 125 public default boolean isInsideJar() { 126 return ResourceUtils.isInsideJar(getClass()); 127// URL url = getClass().getProtectionDomain().getCodeSource().getLocation(); 128// String path = url.getPath(); 129// File file = new File(path); 130// return file.isFile(); 131 } 132 133 /** 134 * Returns an {@link InputStream} for reading from this resource. See also 135 * {@link Class#getResourceAsStream(String)}. This method is not supposed to be overridden. 136 * 137 * @return an {@link InputStream} for the associated resource 138 */ 139 public default InputStream getStream() { 140 return getClass().getResourceAsStream(getRelativePath()); 141 } 142 143 /** 144 * Returns the names of the actual files contained in the associated resource directory of the specified class, 145 * which must implement the {@link NamedResource} interface. This can be used to check if a given named resource has 146 * a matching file in a case-sensitive way. 147 * 148 * @param clazz the resource class 149 * @return an array of strings 150 */ 151 public static String[] getResourceFileNames(Class<? extends NamedResource> clazz) { 152 // return ResourceUtils.getResourceFileNames(clazz, clazz.getSimpleName() + RelativeDirectorySuffix); 153 return ResourceUtils.getResourceFileNames(clazz, getRelativeDirectory(clazz)); 154 } 155 156}