context/src/main/java/org/netbeans/html/context/spi/Contexts.java
author Jaroslav Tulach <jtulach@netbeans.org>
Mon, 26 Jan 2015 19:28:18 +0100
changeset 927 3eb54c43c5c3
parent 904 6505c38a43b3
child 1057 b547f8f663f5
permissions -rw-r--r--
#249545: Only call fillContext once per registered implementation class
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;
jtulach@927
    49
import java.util.HashSet;
jaroslav@574
    50
import java.util.ServiceLoader;
jtulach@927
    51
import java.util.Set;
jaroslav@110
    52
import net.java.html.BrwsrCtx;
jaroslav@362
    53
import org.netbeans.html.context.impl.CtxImpl;
jaroslav@110
    54
jaroslav@538
    55
/** Factory class to assign various technologies 
jaroslav@538
    56
 * to a {@link BrwsrCtx browser context}. Start with {@link #newBuilder()}
jaroslav@538
    57
 * and then assign technologies with {@link Builder#register(java.lang.Class, java.lang.Object, int)}
jaroslav@538
    58
 * method.
jaroslav@110
    59
 *
jtulach@655
    60
 * @author Jaroslav Tulach
jaroslav@110
    61
 */
jaroslav@110
    62
public final class Contexts {
jaroslav@110
    63
    private Contexts() {
jaroslav@110
    64
    }
jaroslav@110
    65
jaroslav@110
    66
    /** Creates new, empty builder for creation of {@link BrwsrCtx}. At the
jaroslav@534
    67
     * end call the {@link Builder#build()} method to generate the context.
jtulach@886
    68
     * 
jtulach@886
    69
     * @param context instances of various classes or names of {@link Id technologies} 
jtulach@886
    70
     *    to be preferred and used in the built {@link BrwsrCtx context}.
jtulach@886
    71
     * @return new instance of the builder
jtulach@886
    72
     * @since 1.1
jtulach@886
    73
     */
jtulach@886
    74
    public static Builder newBuilder(Object... context) {
jtulach@886
    75
        return new Builder(context);
jtulach@886
    76
    }
jtulach@886
    77
    /** Creates new, empty builder for creation of {@link BrwsrCtx}. At the
jtulach@886
    78
     * end call the {@link Builder#build()} method to generate the context.
jtulach@886
    79
     * Simply calls {@link #newBuilder(java.lang.Object...) newBuilder(new Object[0])}.
jtulach@886
    80
     * 
jaroslav@110
    81
     * @return new instance of the builder
jaroslav@110
    82
     */
jaroslav@110
    83
    public static Builder newBuilder() {
jtulach@886
    84
        return newBuilder(new Object[0]);
jaroslav@110
    85
    }
jaroslav@110
    86
jaroslav@110
    87
    /** Seeks for the specified technology in the provided context.
jaroslav@110
    88
     * 
jaroslav@110
    89
     * @param <Tech> type of the technology
jaroslav@110
    90
     * @param context the context to seek in 
jaroslav@110
    91
     *    (previously filled with ({@link Builder#register(java.lang.Class, java.lang.Object, int)})
jaroslav@110
    92
     * @param technology class that identifies the technology
jaroslav@534
    93
     * @return the technology in the context or <code>null</code>
jaroslav@110
    94
     */
jaroslav@110
    95
    public static <Tech> Tech find(BrwsrCtx context, Class<Tech> technology) {
jaroslav@110
    96
        return CtxImpl.find(context, technology);
jaroslav@110
    97
    }
jaroslav@110
    98
jaroslav@574
    99
    /** Seeks {@link ServiceLoader} for all registered instances of
jtulach@838
   100
     * {@link Provider} and asks them to {@link Provider#fillContext(org.netbeans.html.context.spi.Contexts.Builder, java.lang.Class) fill
jaroslav@574
   101
     * the builder}.
jaroslav@574
   102
     * 
jaroslav@574
   103
     * @param requestor the application class for which to find the context
jaroslav@574
   104
     * @param cb the context builder to register technologies into
jaroslav@574
   105
     * @return <code>true</code>, if some instances of the provider were
jaroslav@574
   106
     *    found, <code>false</code> otherwise
jaroslav@574
   107
     * @since 0.7.6
jaroslav@574
   108
     */
jaroslav@574
   109
    public static boolean fillInByProviders(Class<?> requestor, Contexts.Builder cb) {
jaroslav@574
   110
        boolean found = false;
jaroslav@574
   111
        ClassLoader l;
jaroslav@574
   112
        try {
jaroslav@574
   113
            l = requestor.getClassLoader();
jaroslav@574
   114
        } catch (SecurityException ex) {
jaroslav@574
   115
            l = null;
jaroslav@574
   116
        }
jtulach@927
   117
        Set<Class<?>> classes = new HashSet<Class<?>>();
jaroslav@574
   118
        for (Provider cp : ServiceLoader.load(Provider.class, l)) {
jtulach@927
   119
            if (!classes.add(cp.getClass())) {
jtulach@927
   120
                continue;
jtulach@927
   121
            }
jaroslav@574
   122
            cp.fillContext(cb, requestor);
jaroslav@574
   123
            found = true;
jaroslav@574
   124
        }
jaroslav@574
   125
        try {
jaroslav@574
   126
            for (Provider cp : ServiceLoader.load(Provider.class, Provider.class.getClassLoader())) {
jtulach@927
   127
                if (!classes.add(cp.getClass())) {
jtulach@927
   128
                    continue;
jtulach@927
   129
                }
jaroslav@574
   130
                cp.fillContext(cb, requestor);
jaroslav@574
   131
                found = true;
jaroslav@574
   132
            }
jaroslav@574
   133
        } catch (SecurityException ex) {
jaroslav@574
   134
            if (!found) {
jaroslav@574
   135
                throw ex;
jaroslav@574
   136
            }
jaroslav@574
   137
        }
jaroslav@574
   138
        if (!found) {
jaroslav@574
   139
            for (Provider cp : ServiceLoader.load(Provider.class)) {
jtulach@927
   140
                if (!classes.add(cp.getClass())) {
jtulach@927
   141
                    continue;
jtulach@927
   142
                }
jaroslav@574
   143
                cp.fillContext(cb, requestor);
jaroslav@574
   144
                found = true;
jaroslav@574
   145
            }
jaroslav@574
   146
        }
jaroslav@574
   147
        return found;
jaroslav@574
   148
    }
jtulach@886
   149
    
jtulach@886
   150
    /** Identifies the technologies passed to {@link Builder context builder}
jtulach@886
   151
     * by a name. Each implementation of a technology 
jtulach@886
   152
     * {@link Builder#register(java.lang.Class, java.lang.Object, int) registered into a context}
jtulach@886
   153
     * can be annotated with a name (or multiple names). Such implementation
jtulach@886
   154
     * will later be 
jtulach@886
   155
     * {@link Contexts#fillInByProviders(java.lang.Class, org.netbeans.html.context.spi.Contexts.Builder)  preferred during lookup}
jtulach@886
   156
     * if their name(s) has been requested in when 
jtulach@886
   157
     * {@link Contexts#newBuilder(java.lang.Object...)  creating a context}.
jtulach@886
   158
     * @since 1.1
jtulach@886
   159
     */
jtulach@886
   160
    @Retention(RetentionPolicy.RUNTIME)
jtulach@886
   161
    @Target(ElementType.TYPE)
jtulach@886
   162
    public @interface Id {
jtulach@904
   163
        /** Identifier(s) for the implementation. 
jtulach@904
   164
         * 
jtulach@904
   165
         * @return one of more identifiers giving this implementation name(s)
jtulach@904
   166
         */
jtulach@886
   167
        public String[] value();
jtulach@886
   168
    }
jaroslav@574
   169
jaroslav@110
   170
    /** Implementors of various HTML technologies should
jtulach@790
   171
     * register their implementation via <code>org.openide.util.lookup.ServiceProvider</code>, so
jtulach@790
   172
     * {@link ServiceLoader} can find them, when their JARs are included
jaroslav@110
   173
     * on the classpath of the running application.
jaroslav@110
   174
     *
jtulach@655
   175
     * @author Jaroslav Tulach
jaroslav@110
   176
     */
jaroslav@110
   177
    public static interface Provider {
jaroslav@110
   178
jaroslav@110
   179
        /** Register into the context if suitable technology is
jaroslav@110
   180
         * available for the requesting class.
jaroslav@110
   181
         * The provider should check if its own technology is available in current
jaroslav@110
   182
         * scope (e.g. proper JDK, proper browser, etc.). The provider
jaroslav@110
   183
         * can also find the right context depending on requestor's classloader, etc.
jaroslav@110
   184
         * <p>
jaroslav@110
   185
         * Providers should use {@link Builder} to enrich appropriately
jaroslav@110
   186
         * the context.
jaroslav@110
   187
         *
jaroslav@110
   188
         * @param context the context builder to fill with technologies
jaroslav@110
   189
         * @param requestor the application class requesting access the the HTML page
jaroslav@110
   190
         * @see BrwsrCtx#findDefault(java.lang.Class)
jaroslav@110
   191
         */
jaroslav@110
   192
        void fillContext(Builder context, Class<?> requestor);
jaroslav@110
   193
    }
jaroslav@110
   194
jaroslav@110
   195
    /** Support for providers of new {@link BrwsrCtx}. Providers of different
jaroslav@110
   196
     * technologies should be of particular interest in this class. End users
jaroslav@110
   197
     * designing their application with existing technologies should rather
jaroslav@110
   198
     * point their attention to {@link BrwsrCtx} and co.
jaroslav@110
   199
     *
jtulach@655
   200
     * @author Jaroslav Tulach
jaroslav@110
   201
     */
jaroslav@110
   202
    public static final class Builder {
jtulach@886
   203
        private final CtxImpl impl;
jaroslav@110
   204
jtulach@886
   205
        public Builder(Object[] context) {
jtulach@886
   206
            this.impl = new CtxImpl(context);
jaroslav@110
   207
        }
jaroslav@110
   208
        
jaroslav@110
   209
        /** Registers new technology into the context. Each technology is
jaroslav@110
   210
         * exactly identified by its implementation class and can be associated
jaroslav@110
   211
         * with (positive) priority. In case of presence of multiple technologies
jaroslav@110
   212
         * with the same class, the one with higher lower priority takes precedence.
jaroslav@256
   213
         * @param <Tech> type of technology to register
jaroslav@256
   214
         * @param type the real class of the technology type
jaroslav@256
   215
         * @param impl an instance of the technology class
jaroslav@256
   216
         * @param position the lower position, the more important implementation 
jaroslav@256
   217
         *    which will be consulted sooner when seeking for a {@link Contexts#find(net.java.html.BrwsrCtx, java.lang.Class)}
jaroslav@256
   218
         *    an implementation
jaroslav@256
   219
         * @return this builder
jaroslav@110
   220
         */
jaroslav@256
   221
        public <Tech> Builder register(Class<Tech> type, Tech impl, int position) {
jaroslav@589
   222
            if (impl == null) {
jaroslav@589
   223
                return this;
jaroslav@589
   224
            }
jaroslav@256
   225
            if (position <= 0) {
jaroslav@110
   226
                throw new IllegalStateException();
jaroslav@110
   227
            }
jaroslav@256
   228
            this.impl.register(type, impl, position);
jaroslav@110
   229
            return this;
jaroslav@110
   230
        }
jaroslav@110
   231
jaroslav@110
   232
        /** Generates context based on values previously inserted into
jaroslav@110
   233
         * this builder.
jaroslav@110
   234
         *
jaroslav@110
   235
         * @return new, immutable instance of {@link BrwsrCtx}
jaroslav@110
   236
         */
jaroslav@110
   237
        public BrwsrCtx build() {
jaroslav@110
   238
            return impl.build();
jaroslav@110
   239
        }
jaroslav@110
   240
    }
jaroslav@110
   241
}