context/src/main/java/org/netbeans/html/context/spi/Contexts.java
author Jaroslav Tulach <jtulach@netbeans.org>
Thu, 04 Dec 2014 09:21:55 +0100
changeset 886 88d62267a0b5
parent 838 bdc3d696dd4a
child 904 6505c38a43b3
permissions -rw-r--r--
#248918: Introducing technology identifiers
jaroslav@110
     1
/**
jaroslav@358
     2
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
jaroslav@110
     3
 *
jaroslav@551
     4
 * Copyright 2013-2014 Oracle and/or its affiliates. All rights reserved.
jaroslav@110
     5
 *
jaroslav@358
     6
 * Oracle and Java are registered trademarks of Oracle and/or its affiliates.
jaroslav@358
     7
 * Other names may be trademarks of their respective owners.
jaroslav@110
     8
 *
jaroslav@358
     9
 * The contents of this file are subject to the terms of either the GNU
jaroslav@358
    10
 * General Public License Version 2 only ("GPL") or the Common
jaroslav@358
    11
 * Development and Distribution License("CDDL") (collectively, the
jaroslav@358
    12
 * "License"). You may not use this file except in compliance with the
jaroslav@358
    13
 * License. You can obtain a copy of the License at
jaroslav@358
    14
 * http://www.netbeans.org/cddl-gplv2.html
jaroslav@358
    15
 * or nbbuild/licenses/CDDL-GPL-2-CP. See the License for the
jaroslav@358
    16
 * specific language governing permissions and limitations under the
jaroslav@358
    17
 * License.  When distributing the software, include this License Header
jaroslav@358
    18
 * Notice in each file and include the License file at
jaroslav@358
    19
 * nbbuild/licenses/CDDL-GPL-2-CP.  Oracle designates this
jaroslav@358
    20
 * particular file as subject to the "Classpath" exception as provided
jaroslav@358
    21
 * by Oracle in the GPL Version 2 section of the License file that
jaroslav@358
    22
 * accompanied this code. If applicable, add the following below the
jaroslav@358
    23
 * License Header, with the fields enclosed by brackets [] replaced by
jaroslav@358
    24
 * your own identifying information:
jaroslav@358
    25
 * "Portions Copyrighted [year] [name of copyright owner]"
jaroslav@358
    26
 *
jaroslav@358
    27
 * Contributor(s):
jaroslav@358
    28
 *
jaroslav@358
    29
 * The Original Software is NetBeans. The Initial Developer of the Original
jaroslav@551
    30
 * Software is Oracle. Portions Copyright 2013-2014 Oracle. All Rights Reserved.
jaroslav@358
    31
 *
jaroslav@358
    32
 * If you wish your version of this file to be governed by only the CDDL
jaroslav@358
    33
 * or only the GPL Version 2, indicate your decision by adding
jaroslav@358
    34
 * "[Contributor] elects to include this software in this distribution
jaroslav@358
    35
 * under the [CDDL or GPL Version 2] license." If you do not indicate a
jaroslav@358
    36
 * single choice of license, a recipient has the option to distribute
jaroslav@358
    37
 * your version of this file under either the CDDL, the GPL Version 2 or
jaroslav@358
    38
 * to extend the choice of license to its licensees as provided above.
jaroslav@358
    39
 * However, if you add GPL Version 2 code and therefore, elected the GPL
jaroslav@358
    40
 * Version 2 license, then the option applies only if the new code is
jaroslav@358
    41
 * made subject to such option by the copyright holder.
jaroslav@110
    42
 */
jtulach@838
    43
package org.netbeans.html.context.spi;
jaroslav@110
    44
jtulach@886
    45
import java.lang.annotation.ElementType;
jtulach@886
    46
import java.lang.annotation.Retention;
jtulach@886
    47
import java.lang.annotation.RetentionPolicy;
jtulach@886
    48
import java.lang.annotation.Target;
jaroslav@574
    49
import java.util.ServiceLoader;
jaroslav@110
    50
import net.java.html.BrwsrCtx;
jaroslav@362
    51
import org.netbeans.html.context.impl.CtxImpl;
jaroslav@110
    52
jaroslav@538
    53
/** Factory class to assign various technologies 
jaroslav@538
    54
 * to a {@link BrwsrCtx browser context}. Start with {@link #newBuilder()}
jaroslav@538
    55
 * and then assign technologies with {@link Builder#register(java.lang.Class, java.lang.Object, int)}
jaroslav@538
    56
 * method.
jaroslav@110
    57
 *
jtulach@655
    58
 * @author Jaroslav Tulach
jaroslav@110
    59
 */
jaroslav@110
    60
public final class Contexts {
jaroslav@110
    61
    private Contexts() {
jaroslav@110
    62
    }
jaroslav@110
    63
jaroslav@110
    64
    /** Creates new, empty builder for creation of {@link BrwsrCtx}. At the
jaroslav@534
    65
     * end call the {@link Builder#build()} method to generate the context.
jtulach@886
    66
     * 
jtulach@886
    67
     * @param context instances of various classes or names of {@link Id technologies} 
jtulach@886
    68
     *    to be preferred and used in the built {@link BrwsrCtx context}.
jtulach@886
    69
     * @return new instance of the builder
jtulach@886
    70
     * @since 1.1
jtulach@886
    71
     */
jtulach@886
    72
    public static Builder newBuilder(Object... context) {
jtulach@886
    73
        return new Builder(context);
jtulach@886
    74
    }
jtulach@886
    75
    /** Creates new, empty builder for creation of {@link BrwsrCtx}. At the
jtulach@886
    76
     * end call the {@link Builder#build()} method to generate the context.
jtulach@886
    77
     * Simply calls {@link #newBuilder(java.lang.Object...) newBuilder(new Object[0])}.
jtulach@886
    78
     * 
jaroslav@110
    79
     * @return new instance of the builder
jaroslav@110
    80
     */
jaroslav@110
    81
    public static Builder newBuilder() {
jtulach@886
    82
        return newBuilder(new Object[0]);
jaroslav@110
    83
    }
jaroslav@110
    84
jaroslav@110
    85
    /** Seeks for the specified technology in the provided context.
jaroslav@110
    86
     * 
jaroslav@110
    87
     * @param <Tech> type of the technology
jaroslav@110
    88
     * @param context the context to seek in 
jaroslav@110
    89
     *    (previously filled with ({@link Builder#register(java.lang.Class, java.lang.Object, int)})
jaroslav@110
    90
     * @param technology class that identifies the technology
jaroslav@534
    91
     * @return the technology in the context or <code>null</code>
jaroslav@110
    92
     */
jaroslav@110
    93
    public static <Tech> Tech find(BrwsrCtx context, Class<Tech> technology) {
jaroslav@110
    94
        return CtxImpl.find(context, technology);
jaroslav@110
    95
    }
jaroslav@110
    96
jaroslav@574
    97
    /** Seeks {@link ServiceLoader} for all registered instances of
jtulach@838
    98
     * {@link Provider} and asks them to {@link Provider#fillContext(org.netbeans.html.context.spi.Contexts.Builder, java.lang.Class) fill
jaroslav@574
    99
     * the builder}.
jaroslav@574
   100
     * 
jaroslav@574
   101
     * @param requestor the application class for which to find the context
jaroslav@574
   102
     * @param cb the context builder to register technologies into
jaroslav@574
   103
     * @return <code>true</code>, if some instances of the provider were
jaroslav@574
   104
     *    found, <code>false</code> otherwise
jaroslav@574
   105
     * @since 0.7.6
jaroslav@574
   106
     */
jaroslav@574
   107
    public static boolean fillInByProviders(Class<?> requestor, Contexts.Builder cb) {
jaroslav@574
   108
        boolean found = false;
jaroslav@574
   109
        ClassLoader l;
jaroslav@574
   110
        try {
jaroslav@574
   111
            l = requestor.getClassLoader();
jaroslav@574
   112
        } catch (SecurityException ex) {
jaroslav@574
   113
            l = null;
jaroslav@574
   114
        }
jaroslav@574
   115
        for (Provider cp : ServiceLoader.load(Provider.class, l)) {
jaroslav@574
   116
            cp.fillContext(cb, requestor);
jaroslav@574
   117
            found = true;
jaroslav@574
   118
        }
jaroslav@574
   119
        try {
jaroslav@574
   120
            for (Provider cp : ServiceLoader.load(Provider.class, Provider.class.getClassLoader())) {
jaroslav@574
   121
                cp.fillContext(cb, requestor);
jaroslav@574
   122
                found = true;
jaroslav@574
   123
            }
jaroslav@574
   124
        } catch (SecurityException ex) {
jaroslav@574
   125
            if (!found) {
jaroslav@574
   126
                throw ex;
jaroslav@574
   127
            }
jaroslav@574
   128
        }
jaroslav@574
   129
        if (!found) {
jaroslav@574
   130
            for (Provider cp : ServiceLoader.load(Provider.class)) {
jaroslav@574
   131
                cp.fillContext(cb, requestor);
jaroslav@574
   132
                found = true;
jaroslav@574
   133
            }
jaroslav@574
   134
        }
jaroslav@574
   135
        return found;
jaroslav@574
   136
    }
jtulach@886
   137
    
jtulach@886
   138
    /** Identifies the technologies passed to {@link Builder context builder}
jtulach@886
   139
     * by a name. Each implementation of a technology 
jtulach@886
   140
     * {@link Builder#register(java.lang.Class, java.lang.Object, int) registered into a context}
jtulach@886
   141
     * can be annotated with a name (or multiple names). Such implementation
jtulach@886
   142
     * will later be 
jtulach@886
   143
     * {@link Contexts#fillInByProviders(java.lang.Class, org.netbeans.html.context.spi.Contexts.Builder)  preferred during lookup}
jtulach@886
   144
     * if their name(s) has been requested in when 
jtulach@886
   145
     * {@link Contexts#newBuilder(java.lang.Object...)  creating a context}.
jtulach@886
   146
     * @since 1.1
jtulach@886
   147
     */
jtulach@886
   148
    @Retention(RetentionPolicy.RUNTIME)
jtulach@886
   149
    @Target(ElementType.TYPE)
jtulach@886
   150
    public @interface Id {
jtulach@886
   151
        /** Identifier(s) for the implementation. */
jtulach@886
   152
        public String[] value();
jtulach@886
   153
    }
jaroslav@574
   154
jaroslav@110
   155
    /** Implementors of various HTML technologies should
jtulach@790
   156
     * register their implementation via <code>org.openide.util.lookup.ServiceProvider</code>, so
jtulach@790
   157
     * {@link ServiceLoader} can find them, when their JARs are included
jaroslav@110
   158
     * on the classpath of the running application.
jaroslav@110
   159
     *
jtulach@655
   160
     * @author Jaroslav Tulach
jaroslav@110
   161
     */
jaroslav@110
   162
    public static interface Provider {
jaroslav@110
   163
jaroslav@110
   164
        /** Register into the context if suitable technology is
jaroslav@110
   165
         * available for the requesting class.
jaroslav@110
   166
         * The provider should check if its own technology is available in current
jaroslav@110
   167
         * scope (e.g. proper JDK, proper browser, etc.). The provider
jaroslav@110
   168
         * can also find the right context depending on requestor's classloader, etc.
jaroslav@110
   169
         * <p>
jaroslav@110
   170
         * Providers should use {@link Builder} to enrich appropriately
jaroslav@110
   171
         * the context.
jaroslav@110
   172
         *
jaroslav@110
   173
         * @param context the context builder to fill with technologies
jaroslav@110
   174
         * @param requestor the application class requesting access the the HTML page
jaroslav@110
   175
         * @see BrwsrCtx#findDefault(java.lang.Class)
jaroslav@110
   176
         */
jaroslav@110
   177
        void fillContext(Builder context, Class<?> requestor);
jaroslav@110
   178
    }
jaroslav@110
   179
jaroslav@110
   180
    /** Support for providers of new {@link BrwsrCtx}. Providers of different
jaroslav@110
   181
     * technologies should be of particular interest in this class. End users
jaroslav@110
   182
     * designing their application with existing technologies should rather
jaroslav@110
   183
     * point their attention to {@link BrwsrCtx} and co.
jaroslav@110
   184
     *
jtulach@655
   185
     * @author Jaroslav Tulach
jaroslav@110
   186
     */
jaroslav@110
   187
    public static final class Builder {
jtulach@886
   188
        private final CtxImpl impl;
jaroslav@110
   189
jtulach@886
   190
        public Builder(Object[] context) {
jtulach@886
   191
            this.impl = new CtxImpl(context);
jaroslav@110
   192
        }
jaroslav@110
   193
        
jaroslav@110
   194
        /** Registers new technology into the context. Each technology is
jaroslav@110
   195
         * exactly identified by its implementation class and can be associated
jaroslav@110
   196
         * with (positive) priority. In case of presence of multiple technologies
jaroslav@110
   197
         * with the same class, the one with higher lower priority takes precedence.
jaroslav@256
   198
         * @param <Tech> type of technology to register
jaroslav@256
   199
         * @param type the real class of the technology type
jaroslav@256
   200
         * @param impl an instance of the technology class
jaroslav@256
   201
         * @param position the lower position, the more important implementation 
jaroslav@256
   202
         *    which will be consulted sooner when seeking for a {@link Contexts#find(net.java.html.BrwsrCtx, java.lang.Class)}
jaroslav@256
   203
         *    an implementation
jaroslav@256
   204
         * @return this builder
jaroslav@110
   205
         */
jaroslav@256
   206
        public <Tech> Builder register(Class<Tech> type, Tech impl, int position) {
jaroslav@589
   207
            if (impl == null) {
jaroslav@589
   208
                return this;
jaroslav@589
   209
            }
jaroslav@256
   210
            if (position <= 0) {
jaroslav@110
   211
                throw new IllegalStateException();
jaroslav@110
   212
            }
jaroslav@256
   213
            this.impl.register(type, impl, position);
jaroslav@110
   214
            return this;
jaroslav@110
   215
        }
jaroslav@110
   216
jaroslav@110
   217
        /** Generates context based on values previously inserted into
jaroslav@110
   218
         * this builder.
jaroslav@110
   219
         *
jaroslav@110
   220
         * @return new, immutable instance of {@link BrwsrCtx}
jaroslav@110
   221
         */
jaroslav@110
   222
        public BrwsrCtx build() {
jaroslav@110
   223
            return impl.build();
jaroslav@110
   224
        }
jaroslav@110
   225
    }
jaroslav@110
   226
}