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 *      &#64;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}