jdk/src/share/classes/sun/awt/AppContext.java
author anthony
Tue, 20 Dec 2011 12:48:01 +0300
changeset 11272 d890af1d95b7
parent 11264 54f2f4c6bd30
child 11902 a94ba35d9c4a
permissions -rw-r--r--
7122796: SunToolkit constructor should create the EventQueue for the Main AppContext Summary: Always create an EQ for the main AppContext in SunToolkit constructor Reviewed-by: art
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     1
/*
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 4365
diff changeset
     2
 * Copyright (c) 1998, 2008, Oracle and/or its affiliates. All rights reserved.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
90ce3da70b43 Initial load
duke
parents:
diff changeset
     4
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
90ce3da70b43 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 4365
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 4365
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    10
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
90ce3da70b43 Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
90ce3da70b43 Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
90ce3da70b43 Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    16
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
90ce3da70b43 Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    20
 *
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 4365
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 4365
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 4365
diff changeset
    23
 * questions.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    24
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    25
90ce3da70b43 Initial load
duke
parents:
diff changeset
    26
package sun.awt;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    27
90ce3da70b43 Initial load
duke
parents:
diff changeset
    28
import java.awt.EventQueue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    29
import java.awt.Window;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    30
import java.awt.SystemTray;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    31
import java.awt.TrayIcon;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    32
import java.awt.Toolkit;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    33
import java.awt.GraphicsEnvironment;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    34
import java.awt.event.InvocationEvent;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    35
import java.security.AccessController;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    36
import java.security.PrivilegedAction;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    37
import java.util.Collections;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    38
import java.util.HashMap;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    39
import java.util.IdentityHashMap;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    40
import java.util.Map;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    41
import java.util.Set;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    42
import java.util.HashSet;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    43
import java.beans.PropertyChangeSupport;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    44
import java.beans.PropertyChangeListener;
3938
ef327bd847c0 6879044: Eliminate the dependency on logging from the AWT/2D/Swing classes
mchung
parents: 113
diff changeset
    45
import sun.util.logging.PlatformLogger;
4365
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
    46
import java.util.concurrent.locks.Condition;
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
    47
import java.util.concurrent.locks.Lock;
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
    48
import java.util.concurrent.locks.ReentrantLock;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    49
90ce3da70b43 Initial load
duke
parents:
diff changeset
    50
/**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    51
 * The AppContext is a table referenced by ThreadGroup which stores
90ce3da70b43 Initial load
duke
parents:
diff changeset
    52
 * application service instances.  (If you are not writing an application
90ce3da70b43 Initial load
duke
parents:
diff changeset
    53
 * service, or don't know what one is, please do not use this class.)
90ce3da70b43 Initial load
duke
parents:
diff changeset
    54
 * The AppContext allows applet access to what would otherwise be
90ce3da70b43 Initial load
duke
parents:
diff changeset
    55
 * potentially dangerous services, such as the ability to peek at
90ce3da70b43 Initial load
duke
parents:
diff changeset
    56
 * EventQueues or change the look-and-feel of a Swing application.<p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    57
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    58
 * Most application services use a singleton object to provide their
90ce3da70b43 Initial load
duke
parents:
diff changeset
    59
 * services, either as a default (such as getSystemEventQueue or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    60
 * getDefaultToolkit) or as static methods with class data (System).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    61
 * The AppContext works with the former method by extending the concept
90ce3da70b43 Initial load
duke
parents:
diff changeset
    62
 * of "default" to be ThreadGroup-specific.  Application services
90ce3da70b43 Initial load
duke
parents:
diff changeset
    63
 * lookup their singleton in the AppContext.<p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    64
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    65
 * For example, here we have a Foo service, with its pre-AppContext
90ce3da70b43 Initial load
duke
parents:
diff changeset
    66
 * code:<p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    67
 * <code><pre>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    68
 *    public class Foo {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    69
 *        private static Foo defaultFoo = new Foo();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    70
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    71
 *        public static Foo getDefaultFoo() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    72
 *            return defaultFoo;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    73
 *        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    74
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    75
 *    ... Foo service methods
90ce3da70b43 Initial load
duke
parents:
diff changeset
    76
 *    }</pre></code><p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    77
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    78
 * The problem with the above is that the Foo service is global in scope,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    79
 * so that applets and other untrusted code can execute methods on the
90ce3da70b43 Initial load
duke
parents:
diff changeset
    80
 * single, shared Foo instance.  The Foo service therefore either needs
90ce3da70b43 Initial load
duke
parents:
diff changeset
    81
 * to block its use by untrusted code using a SecurityManager test, or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    82
 * restrict its capabilities so that it doesn't matter if untrusted code
90ce3da70b43 Initial load
duke
parents:
diff changeset
    83
 * executes it.<p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    84
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    85
 * Here's the Foo class written to use the AppContext:<p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    86
 * <code><pre>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    87
 *    public class Foo {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    88
 *        public static Foo getDefaultFoo() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    89
 *            Foo foo = (Foo)AppContext.getAppContext().get(Foo.class);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    90
 *            if (foo == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    91
 *                foo = new Foo();
90ce3da70b43 Initial load
duke
parents:
diff changeset
    92
 *                getAppContext().put(Foo.class, foo);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    93
 *            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    94
 *            return foo;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    95
 *        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    96
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    97
 *    ... Foo service methods
90ce3da70b43 Initial load
duke
parents:
diff changeset
    98
 *    }</pre></code><p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
    99
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   100
 * Since a separate AppContext can exist for each ThreadGroup, trusted
90ce3da70b43 Initial load
duke
parents:
diff changeset
   101
 * and untrusted code have access to different Foo instances.  This allows
90ce3da70b43 Initial load
duke
parents:
diff changeset
   102
 * untrusted code access to "system-wide" services -- the service remains
90ce3da70b43 Initial load
duke
parents:
diff changeset
   103
 * within the AppContext "sandbox".  For example, say a malicious applet
90ce3da70b43 Initial load
duke
parents:
diff changeset
   104
 * wants to peek all of the key events on the EventQueue to listen for
90ce3da70b43 Initial load
duke
parents:
diff changeset
   105
 * passwords; if separate EventQueues are used for each ThreadGroup
90ce3da70b43 Initial load
duke
parents:
diff changeset
   106
 * using AppContexts, the only key events that applet will be able to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   107
 * listen to are its own.  A more reasonable applet request would be to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   108
 * change the Swing default look-and-feel; with that default stored in
90ce3da70b43 Initial load
duke
parents:
diff changeset
   109
 * an AppContext, the applet's look-and-feel will change without
90ce3da70b43 Initial load
duke
parents:
diff changeset
   110
 * disrupting other applets or potentially the browser itself.<p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   111
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   112
 * Because the AppContext is a facility for safely extending application
90ce3da70b43 Initial load
duke
parents:
diff changeset
   113
 * service support to applets, none of its methods may be blocked by a
90ce3da70b43 Initial load
duke
parents:
diff changeset
   114
 * a SecurityManager check in a valid Java implementation.  Applets may
90ce3da70b43 Initial load
duke
parents:
diff changeset
   115
 * therefore safely invoke any of its methods without worry of being
90ce3da70b43 Initial load
duke
parents:
diff changeset
   116
 * blocked.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   117
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   118
 * Note: If a SecurityManager is installed which derives from
90ce3da70b43 Initial load
duke
parents:
diff changeset
   119
 * sun.awt.AWTSecurityManager, it may override the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   120
 * AWTSecurityManager.getAppContext() method to return the proper
90ce3da70b43 Initial load
duke
parents:
diff changeset
   121
 * AppContext based on the execution context, in the case where
90ce3da70b43 Initial load
duke
parents:
diff changeset
   122
 * the default ThreadGroup-based AppContext indexing would return
90ce3da70b43 Initial load
duke
parents:
diff changeset
   123
 * the main "system" AppContext.  For example, in an applet situation,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   124
 * if a system thread calls into an applet, rather than returning the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   125
 * main "system" AppContext (the one corresponding to the system thread),
90ce3da70b43 Initial load
duke
parents:
diff changeset
   126
 * an installed AWTSecurityManager may return the applet's AppContext
90ce3da70b43 Initial load
duke
parents:
diff changeset
   127
 * based on the execution context.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   128
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   129
 * @author  Thomas Ball
90ce3da70b43 Initial load
duke
parents:
diff changeset
   130
 * @author  Fred Ecks
90ce3da70b43 Initial load
duke
parents:
diff changeset
   131
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   132
public final class AppContext {
3938
ef327bd847c0 6879044: Eliminate the dependency on logging from the AWT/2D/Swing classes
mchung
parents: 113
diff changeset
   133
    private static final PlatformLogger log = PlatformLogger.getLogger("sun.awt.AppContext");
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   134
90ce3da70b43 Initial load
duke
parents:
diff changeset
   135
    /* Since the contents of an AppContext are unique to each Java
90ce3da70b43 Initial load
duke
parents:
diff changeset
   136
     * session, this class should never be serialized. */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   137
4365
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   138
    /*
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   139
     * The key to put()/get() the Java EventQueue into/from the AppContext.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   140
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   141
    public static final Object EVENT_QUEUE_KEY = new StringBuffer("EventQueue");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   142
4365
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   143
    /*
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   144
     * The keys to store EventQueue push/pop lock and condition.
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   145
     */
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   146
    public final static Object EVENT_QUEUE_LOCK_KEY = new StringBuilder("EventQueue.Lock");
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   147
    public final static Object EVENT_QUEUE_COND_KEY = new StringBuilder("EventQueue.Condition");
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   148
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   149
    /* A map of AppContexts, referenced by ThreadGroup.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   150
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   151
    private static final Map<ThreadGroup, AppContext> threadGroup2appContext =
90ce3da70b43 Initial load
duke
parents:
diff changeset
   152
            Collections.synchronizedMap(new IdentityHashMap<ThreadGroup, AppContext>());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   153
90ce3da70b43 Initial load
duke
parents:
diff changeset
   154
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   155
     * Returns a set containing all <code>AppContext</code>s.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   156
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   157
    public static Set<AppContext> getAppContexts() {
112
c7ed16a1bef2 6636331: ConcurrentModificationException in AppContext code
son
parents: 108
diff changeset
   158
        synchronized (threadGroup2appContext) {
c7ed16a1bef2 6636331: ConcurrentModificationException in AppContext code
son
parents: 108
diff changeset
   159
            return new HashSet<AppContext>(threadGroup2appContext.values());
c7ed16a1bef2 6636331: ConcurrentModificationException in AppContext code
son
parents: 108
diff changeset
   160
        }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   161
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   162
90ce3da70b43 Initial load
duke
parents:
diff changeset
   163
    /* The main "system" AppContext, used by everything not otherwise
90ce3da70b43 Initial load
duke
parents:
diff changeset
   164
       contained in another AppContext.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   165
     */
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   166
    private static volatile AppContext mainAppContext = null;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   167
90ce3da70b43 Initial load
duke
parents:
diff changeset
   168
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   169
     * The hash map associated with this AppContext.  A private delegate
90ce3da70b43 Initial load
duke
parents:
diff changeset
   170
     * is used instead of subclassing HashMap so as to avoid all of
90ce3da70b43 Initial load
duke
parents:
diff changeset
   171
     * HashMap's potentially risky methods, such as clear(), elements(),
90ce3da70b43 Initial load
duke
parents:
diff changeset
   172
     * putAll(), etc.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   173
     */
11264
54f2f4c6bd30 7117008: Warnings cleanup day: reduce number of javac warnings in the sun.awt package
art
parents: 5506
diff changeset
   174
    private final Map<Object, Object> table = new HashMap<>();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   175
90ce3da70b43 Initial load
duke
parents:
diff changeset
   176
    private final ThreadGroup threadGroup;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   177
90ce3da70b43 Initial load
duke
parents:
diff changeset
   178
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   179
     * If any <code>PropertyChangeListeners</code> have been registered,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   180
     * the <code>changeSupport</code> field describes them.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   181
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   182
     * @see #addPropertyChangeListener
90ce3da70b43 Initial load
duke
parents:
diff changeset
   183
     * @see #removePropertyChangeListener
90ce3da70b43 Initial load
duke
parents:
diff changeset
   184
     * @see #firePropertyChange
90ce3da70b43 Initial load
duke
parents:
diff changeset
   185
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   186
    private PropertyChangeSupport changeSupport = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   187
90ce3da70b43 Initial load
duke
parents:
diff changeset
   188
    public static final String DISPOSED_PROPERTY_NAME = "disposed";
90ce3da70b43 Initial load
duke
parents:
diff changeset
   189
    public static final String GUI_DISPOSED = "guidisposed";
90ce3da70b43 Initial load
duke
parents:
diff changeset
   190
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   191
    private volatile boolean isDisposed = false; // true if AppContext is disposed
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   192
90ce3da70b43 Initial load
duke
parents:
diff changeset
   193
    public boolean isDisposed() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   194
        return isDisposed;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   195
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   196
90ce3da70b43 Initial load
duke
parents:
diff changeset
   197
    static {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   198
        // On the main Thread, we get the ThreadGroup, make a corresponding
90ce3da70b43 Initial load
duke
parents:
diff changeset
   199
        // AppContext, and instantiate the Java EventQueue.  This way, legacy
90ce3da70b43 Initial load
duke
parents:
diff changeset
   200
        // code is unaffected by the move to multiple AppContext ability.
11264
54f2f4c6bd30 7117008: Warnings cleanup day: reduce number of javac warnings in the sun.awt package
art
parents: 5506
diff changeset
   201
        AccessController.doPrivileged(new PrivilegedAction<Void>() {
54f2f4c6bd30 7117008: Warnings cleanup day: reduce number of javac warnings in the sun.awt package
art
parents: 5506
diff changeset
   202
            public Void run() {
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   203
                ThreadGroup currentThreadGroup =
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   204
                        Thread.currentThread().getThreadGroup();
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   205
                ThreadGroup parentThreadGroup = currentThreadGroup.getParent();
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   206
                while (parentThreadGroup != null) {
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   207
                    // Find the root ThreadGroup to construct our main AppContext
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   208
                    currentThreadGroup = parentThreadGroup;
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   209
                    parentThreadGroup = currentThreadGroup.getParent();
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   210
                }
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   211
                mainAppContext = new AppContext(currentThreadGroup);
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   212
                numAppContexts = 1;
11264
54f2f4c6bd30 7117008: Warnings cleanup day: reduce number of javac warnings in the sun.awt package
art
parents: 5506
diff changeset
   213
                return null;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   214
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   215
        });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   216
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   217
90ce3da70b43 Initial load
duke
parents:
diff changeset
   218
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   219
     * The total number of AppContexts, system-wide.  This number is
90ce3da70b43 Initial load
duke
parents:
diff changeset
   220
     * incremented at the beginning of the constructor, and decremented
90ce3da70b43 Initial load
duke
parents:
diff changeset
   221
     * at the end of dispose().  getAppContext() checks to see if this
90ce3da70b43 Initial load
duke
parents:
diff changeset
   222
     * number is 1.  If so, it returns the sole AppContext without
90ce3da70b43 Initial load
duke
parents:
diff changeset
   223
     * checking Thread.currentThread().
90ce3da70b43 Initial load
duke
parents:
diff changeset
   224
     */
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   225
    private static volatile int numAppContexts;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   226
90ce3da70b43 Initial load
duke
parents:
diff changeset
   227
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   228
     * The context ClassLoader that was used to create this AppContext.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   229
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   230
    private final ClassLoader contextClassLoader;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   231
90ce3da70b43 Initial load
duke
parents:
diff changeset
   232
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   233
     * Constructor for AppContext.  This method is <i>not</i> public,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   234
     * nor should it ever be used as such.  The proper way to construct
90ce3da70b43 Initial load
duke
parents:
diff changeset
   235
     * an AppContext is through the use of SunToolkit.createNewAppContext.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   236
     * A ThreadGroup is created for the new AppContext, a Thread is
90ce3da70b43 Initial load
duke
parents:
diff changeset
   237
     * created within that ThreadGroup, and that Thread calls
90ce3da70b43 Initial load
duke
parents:
diff changeset
   238
     * SunToolkit.createNewAppContext before calling anything else.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   239
     * That creates both the new AppContext and its EventQueue.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   240
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   241
     * @param   threadGroup     The ThreadGroup for the new AppContext
90ce3da70b43 Initial load
duke
parents:
diff changeset
   242
     * @see     sun.awt.SunToolkit
90ce3da70b43 Initial load
duke
parents:
diff changeset
   243
     * @since   1.2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   244
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   245
    AppContext(ThreadGroup threadGroup) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   246
        numAppContexts++;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   247
90ce3da70b43 Initial load
duke
parents:
diff changeset
   248
        this.threadGroup = threadGroup;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   249
        threadGroup2appContext.put(threadGroup, this);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   250
90ce3da70b43 Initial load
duke
parents:
diff changeset
   251
        this.contextClassLoader =
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   252
             AccessController.doPrivileged(new PrivilegedAction<ClassLoader>() {
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   253
                    public ClassLoader run() {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   254
                        return Thread.currentThread().getContextClassLoader();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   255
                    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   256
                });
4365
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   257
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   258
        // Initialize push/pop lock and its condition to be used by all the
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   259
        // EventQueues within this AppContext
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   260
        Lock eventQueuePushPopLock = new ReentrantLock();
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   261
        put(EVENT_QUEUE_LOCK_KEY, eventQueuePushPopLock);
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   262
        Condition eventQueuePushPopCond = eventQueuePushPopLock.newCondition();
4ac67034e98b 4913324: Deadlock when using two event queues
art
parents: 3938
diff changeset
   263
        put(EVENT_QUEUE_COND_KEY, eventQueuePushPopCond);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   264
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   265
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   266
    private static final ThreadLocal<AppContext> threadAppContext =
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   267
            new ThreadLocal<AppContext>();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   268
90ce3da70b43 Initial load
duke
parents:
diff changeset
   269
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   270
     * Returns the appropriate AppContext for the caller,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   271
     * as determined by its ThreadGroup.  If the main "system" AppContext
90ce3da70b43 Initial load
duke
parents:
diff changeset
   272
     * would be returned and there's an AWTSecurityManager installed, it
90ce3da70b43 Initial load
duke
parents:
diff changeset
   273
     * is called to get the proper AppContext based on the execution
90ce3da70b43 Initial load
duke
parents:
diff changeset
   274
     * context.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   275
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   276
     * @return  the AppContext for the caller.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   277
     * @see     java.lang.ThreadGroup
90ce3da70b43 Initial load
duke
parents:
diff changeset
   278
     * @since   1.2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   279
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   280
    public final static AppContext getAppContext() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   281
        if (numAppContexts == 1)   // If there's only one system-wide,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   282
            return mainAppContext; // return the main system AppContext.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   283
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   284
        AppContext appContext = threadAppContext.get();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   285
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   286
        if (null == appContext) {
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   287
            appContext = AccessController.doPrivileged(new PrivilegedAction<AppContext>()
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   288
            {
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   289
                public AppContext run() {
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   290
                    // Get the current ThreadGroup, and look for it and its
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   291
                    // parents in the hash from ThreadGroup to AppContext --
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   292
                    // it should be found, because we use createNewContext()
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   293
                    // when new AppContext objects are created.
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   294
                    ThreadGroup currentThreadGroup = Thread.currentThread().getThreadGroup();
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   295
                    ThreadGroup threadGroup = currentThreadGroup;
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   296
                    AppContext context = threadGroup2appContext.get(threadGroup);
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   297
                    while (context == null) {
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   298
                        threadGroup = threadGroup.getParent();
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   299
                        if (threadGroup == null) {
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   300
                            // If we get here, we're running under a ThreadGroup that
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   301
                            // has no AppContext associated with it.  This should never
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   302
                            // happen, because createNewContext() should be used by the
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   303
                            // toolkit to create the ThreadGroup that everything runs
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   304
                            // under.
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   305
                            throw new RuntimeException("Invalid ThreadGroup");
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   306
                        }
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   307
                        context = threadGroup2appContext.get(threadGroup);
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   308
                    }
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   309
                    // In case we did anything in the above while loop, we add
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   310
                    // all the intermediate ThreadGroups to threadGroup2appContext
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   311
                    // so we won't spin again.
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   312
                    for (ThreadGroup tg = currentThreadGroup; tg != threadGroup; tg = tg.getParent()) {
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   313
                        threadGroup2appContext.put(tg, context);
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   314
                    }
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   315
                    // Now we're done, so we cache the latest key/value pair.
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   316
                    // (we do this before checking with any AWTSecurityManager, so if
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   317
                    // this Thread equates with the main AppContext in the cache, it
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   318
                    // still will)
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   319
                    threadAppContext.set(context);
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   320
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   321
                    return context;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   322
                }
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   323
            });
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   324
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   325
90ce3da70b43 Initial load
duke
parents:
diff changeset
   326
        if (appContext == mainAppContext)  {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   327
            // Before we return the main "system" AppContext, check to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   328
            // see if there's an AWTSecurityManager installed.  If so,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   329
            // allow it to choose the AppContext to return.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   330
            SecurityManager securityManager = System.getSecurityManager();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   331
            if ((securityManager != null) &&
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   332
                (securityManager instanceof AWTSecurityManager))
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   333
            {
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   334
                AWTSecurityManager awtSecMgr = (AWTSecurityManager)securityManager;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   335
                AppContext secAppContext = awtSecMgr.getAppContext();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   336
                if (secAppContext != null)  {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   337
                    appContext = secAppContext; // Return what we're told
90ce3da70b43 Initial load
duke
parents:
diff changeset
   338
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   339
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   340
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   341
90ce3da70b43 Initial load
duke
parents:
diff changeset
   342
        return appContext;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   343
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   344
11272
d890af1d95b7 7122796: SunToolkit constructor should create the EventQueue for the Main AppContext
anthony
parents: 11264
diff changeset
   345
    /**
d890af1d95b7 7122796: SunToolkit constructor should create the EventQueue for the Main AppContext
anthony
parents: 11264
diff changeset
   346
     * Returns the main ("system") AppContext.
d890af1d95b7 7122796: SunToolkit constructor should create the EventQueue for the Main AppContext
anthony
parents: 11264
diff changeset
   347
     *
d890af1d95b7 7122796: SunToolkit constructor should create the EventQueue for the Main AppContext
anthony
parents: 11264
diff changeset
   348
     * @return  the main AppContext
d890af1d95b7 7122796: SunToolkit constructor should create the EventQueue for the Main AppContext
anthony
parents: 11264
diff changeset
   349
     * @since   1.8
d890af1d95b7 7122796: SunToolkit constructor should create the EventQueue for the Main AppContext
anthony
parents: 11264
diff changeset
   350
     */
d890af1d95b7 7122796: SunToolkit constructor should create the EventQueue for the Main AppContext
anthony
parents: 11264
diff changeset
   351
    final static AppContext getMainAppContext() {
d890af1d95b7 7122796: SunToolkit constructor should create the EventQueue for the Main AppContext
anthony
parents: 11264
diff changeset
   352
        return mainAppContext;
d890af1d95b7 7122796: SunToolkit constructor should create the EventQueue for the Main AppContext
anthony
parents: 11264
diff changeset
   353
    }
d890af1d95b7 7122796: SunToolkit constructor should create the EventQueue for the Main AppContext
anthony
parents: 11264
diff changeset
   354
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   355
    private long DISPOSAL_TIMEOUT = 5000;  // Default to 5-second timeout
90ce3da70b43 Initial load
duke
parents:
diff changeset
   356
                                           // for disposal of all Frames
90ce3da70b43 Initial load
duke
parents:
diff changeset
   357
                                           // (we wait for this time twice,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   358
                                           // once for dispose(), and once
90ce3da70b43 Initial load
duke
parents:
diff changeset
   359
                                           // to clear the EventQueue).
90ce3da70b43 Initial load
duke
parents:
diff changeset
   360
90ce3da70b43 Initial load
duke
parents:
diff changeset
   361
    private long THREAD_INTERRUPT_TIMEOUT = 1000;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   362
                            // Default to 1-second timeout for all
90ce3da70b43 Initial load
duke
parents:
diff changeset
   363
                            // interrupted Threads to exit, and another
90ce3da70b43 Initial load
duke
parents:
diff changeset
   364
                            // 1 second for all stopped Threads to die.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   365
90ce3da70b43 Initial load
duke
parents:
diff changeset
   366
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   367
     * Disposes of this AppContext, all of its top-level Frames, and
90ce3da70b43 Initial load
duke
parents:
diff changeset
   368
     * all Threads and ThreadGroups contained within it.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   369
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   370
     * This method must be called from a Thread which is not contained
90ce3da70b43 Initial load
duke
parents:
diff changeset
   371
     * within this AppContext.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   372
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   373
     * @exception  IllegalThreadStateException  if the current thread is
90ce3da70b43 Initial load
duke
parents:
diff changeset
   374
     *                                    contained within this AppContext
90ce3da70b43 Initial load
duke
parents:
diff changeset
   375
     * @since      1.2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   376
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   377
    public void dispose() throws IllegalThreadStateException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   378
        // Check to be sure that the current Thread isn't in this AppContext
90ce3da70b43 Initial load
duke
parents:
diff changeset
   379
        if (this.threadGroup.parentOf(Thread.currentThread().getThreadGroup())) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   380
            throw new IllegalThreadStateException(
90ce3da70b43 Initial load
duke
parents:
diff changeset
   381
                "Current Thread is contained within AppContext to be disposed."
90ce3da70b43 Initial load
duke
parents:
diff changeset
   382
              );
90ce3da70b43 Initial load
duke
parents:
diff changeset
   383
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   384
90ce3da70b43 Initial load
duke
parents:
diff changeset
   385
        synchronized(this) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   386
            if (this.isDisposed) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   387
                return; // If already disposed, bail.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   388
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   389
            this.isDisposed = true;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   390
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   391
90ce3da70b43 Initial load
duke
parents:
diff changeset
   392
        final PropertyChangeSupport changeSupport = this.changeSupport;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   393
        if (changeSupport != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   394
            changeSupport.firePropertyChange(DISPOSED_PROPERTY_NAME, false, true);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   395
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   396
90ce3da70b43 Initial load
duke
parents:
diff changeset
   397
        // First, we post an InvocationEvent to be run on the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   398
        // EventDispatchThread which disposes of all top-level Frames and TrayIcons
90ce3da70b43 Initial load
duke
parents:
diff changeset
   399
90ce3da70b43 Initial load
duke
parents:
diff changeset
   400
        final Object notificationLock = new Object();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   401
90ce3da70b43 Initial load
duke
parents:
diff changeset
   402
        Runnable runnable = new Runnable() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   403
            public void run() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   404
                Window[] windowsToDispose = Window.getOwnerlessWindows();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   405
                for (Window w : windowsToDispose) {
108
deaaed5cedb7 6592751: EmbeddedFrame disposal is fragile and breaks clean AppContext termination
son
parents: 2
diff changeset
   406
                    try {
deaaed5cedb7 6592751: EmbeddedFrame disposal is fragile and breaks clean AppContext termination
son
parents: 2
diff changeset
   407
                        w.dispose();
deaaed5cedb7 6592751: EmbeddedFrame disposal is fragile and breaks clean AppContext termination
son
parents: 2
diff changeset
   408
                    } catch (Throwable t) {
3938
ef327bd847c0 6879044: Eliminate the dependency on logging from the AWT/2D/Swing classes
mchung
parents: 113
diff changeset
   409
                        log.finer("exception occured while disposing app context", t);
108
deaaed5cedb7 6592751: EmbeddedFrame disposal is fragile and breaks clean AppContext termination
son
parents: 2
diff changeset
   410
                    }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   411
                }
11264
54f2f4c6bd30 7117008: Warnings cleanup day: reduce number of javac warnings in the sun.awt package
art
parents: 5506
diff changeset
   412
                AccessController.doPrivileged(new PrivilegedAction<Void>() {
54f2f4c6bd30 7117008: Warnings cleanup day: reduce number of javac warnings in the sun.awt package
art
parents: 5506
diff changeset
   413
                        public Void run() {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   414
                            if (!GraphicsEnvironment.isHeadless() && SystemTray.isSupported())
90ce3da70b43 Initial load
duke
parents:
diff changeset
   415
                            {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   416
                                SystemTray systemTray = SystemTray.getSystemTray();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   417
                                TrayIcon[] trayIconsToDispose = systemTray.getTrayIcons();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   418
                                for (TrayIcon ti : trayIconsToDispose) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   419
                                    systemTray.remove(ti);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   420
                                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   421
                            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   422
                            return null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   423
                        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   424
                    });
90ce3da70b43 Initial load
duke
parents:
diff changeset
   425
                // Alert PropertyChangeListeners that the GUI has been disposed.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   426
                if (changeSupport != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   427
                    changeSupport.firePropertyChange(GUI_DISPOSED, false, true);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   428
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   429
                synchronized(notificationLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   430
                    notificationLock.notifyAll(); // Notify caller that we're done
90ce3da70b43 Initial load
duke
parents:
diff changeset
   431
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   432
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   433
        };
90ce3da70b43 Initial load
duke
parents:
diff changeset
   434
        synchronized(notificationLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   435
            SunToolkit.postEvent(this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   436
                new InvocationEvent(Toolkit.getDefaultToolkit(), runnable));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   437
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   438
                notificationLock.wait(DISPOSAL_TIMEOUT);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   439
            } catch (InterruptedException e) { }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   440
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   441
90ce3da70b43 Initial load
duke
parents:
diff changeset
   442
        // Next, we post another InvocationEvent to the end of the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   443
        // EventQueue.  When it's executed, we know we've executed all
90ce3da70b43 Initial load
duke
parents:
diff changeset
   444
        // events in the queue.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   445
90ce3da70b43 Initial load
duke
parents:
diff changeset
   446
        runnable = new Runnable() { public void run() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   447
            synchronized(notificationLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   448
                notificationLock.notifyAll(); // Notify caller that we're done
90ce3da70b43 Initial load
duke
parents:
diff changeset
   449
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   450
        } };
90ce3da70b43 Initial load
duke
parents:
diff changeset
   451
        synchronized(notificationLock) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   452
            SunToolkit.postEvent(this,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   453
                new InvocationEvent(Toolkit.getDefaultToolkit(), runnable));
90ce3da70b43 Initial load
duke
parents:
diff changeset
   454
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   455
                notificationLock.wait(DISPOSAL_TIMEOUT);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   456
            } catch (InterruptedException e) { }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   457
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   458
90ce3da70b43 Initial load
duke
parents:
diff changeset
   459
        // Next, we interrupt all Threads in the ThreadGroup
90ce3da70b43 Initial load
duke
parents:
diff changeset
   460
        this.threadGroup.interrupt();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   461
            // Note, the EventDispatchThread we've interrupted may dump an
90ce3da70b43 Initial load
duke
parents:
diff changeset
   462
            // InterruptedException to the console here.  This needs to be
90ce3da70b43 Initial load
duke
parents:
diff changeset
   463
            // fixed in the EventDispatchThread, not here.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   464
90ce3da70b43 Initial load
duke
parents:
diff changeset
   465
        // Next, we sleep 10ms at a time, waiting for all of the active
90ce3da70b43 Initial load
duke
parents:
diff changeset
   466
        // Threads in the ThreadGroup to exit.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   467
90ce3da70b43 Initial load
duke
parents:
diff changeset
   468
        long startTime = System.currentTimeMillis();
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   469
        long endTime = startTime + THREAD_INTERRUPT_TIMEOUT;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   470
        while ((this.threadGroup.activeCount() > 0) &&
90ce3da70b43 Initial load
duke
parents:
diff changeset
   471
               (System.currentTimeMillis() < endTime)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   472
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   473
                Thread.sleep(10);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   474
            } catch (InterruptedException e) { }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   475
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   476
90ce3da70b43 Initial load
duke
parents:
diff changeset
   477
        // Then, we stop any remaining Threads
90ce3da70b43 Initial load
duke
parents:
diff changeset
   478
        this.threadGroup.stop();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   479
90ce3da70b43 Initial load
duke
parents:
diff changeset
   480
        // Next, we sleep 10ms at a time, waiting for all of the active
90ce3da70b43 Initial load
duke
parents:
diff changeset
   481
        // Threads in the ThreadGroup to die.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   482
90ce3da70b43 Initial load
duke
parents:
diff changeset
   483
        startTime = System.currentTimeMillis();
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   484
        endTime = startTime + THREAD_INTERRUPT_TIMEOUT;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   485
        while ((this.threadGroup.activeCount() > 0) &&
90ce3da70b43 Initial load
duke
parents:
diff changeset
   486
               (System.currentTimeMillis() < endTime)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   487
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   488
                Thread.sleep(10);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   489
            } catch (InterruptedException e) { }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   490
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   491
90ce3da70b43 Initial load
duke
parents:
diff changeset
   492
        // Next, we remove this and all subThreadGroups from threadGroup2appContext
90ce3da70b43 Initial load
duke
parents:
diff changeset
   493
        int numSubGroups = this.threadGroup.activeGroupCount();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   494
        if (numSubGroups > 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   495
            ThreadGroup [] subGroups = new ThreadGroup[numSubGroups];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   496
            numSubGroups = this.threadGroup.enumerate(subGroups);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   497
            for (int subGroup = 0; subGroup < numSubGroups; subGroup++) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   498
                threadGroup2appContext.remove(subGroups[subGroup]);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   499
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   500
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   501
        threadGroup2appContext.remove(this.threadGroup);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   502
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   503
        threadAppContext.set(null);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   504
90ce3da70b43 Initial load
duke
parents:
diff changeset
   505
        // Finally, we destroy the ThreadGroup entirely.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   506
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   507
            this.threadGroup.destroy();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   508
        } catch (IllegalThreadStateException e) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   509
            // Fired if not all the Threads died, ignore it and proceed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   510
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   511
90ce3da70b43 Initial load
duke
parents:
diff changeset
   512
        synchronized (table) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   513
            this.table.clear(); // Clear out the Hashtable to ease garbage collection
90ce3da70b43 Initial load
duke
parents:
diff changeset
   514
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   515
90ce3da70b43 Initial load
duke
parents:
diff changeset
   516
        numAppContexts--;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   517
90ce3da70b43 Initial load
duke
parents:
diff changeset
   518
        mostRecentKeyValue = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   519
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   520
90ce3da70b43 Initial load
duke
parents:
diff changeset
   521
    static final class PostShutdownEventRunnable implements Runnable {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   522
        private final AppContext appContext;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   523
90ce3da70b43 Initial load
duke
parents:
diff changeset
   524
        public PostShutdownEventRunnable(AppContext ac) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   525
            appContext = ac;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   526
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   527
90ce3da70b43 Initial load
duke
parents:
diff changeset
   528
        public void run() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   529
            final EventQueue eq = (EventQueue)appContext.get(EVENT_QUEUE_KEY);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   530
            if (eq != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   531
                eq.postEvent(AWTAutoShutdown.getShutdownEvent());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   532
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   533
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   534
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   535
11264
54f2f4c6bd30 7117008: Warnings cleanup day: reduce number of javac warnings in the sun.awt package
art
parents: 5506
diff changeset
   536
    static final class CreateThreadAction implements PrivilegedAction<Thread> {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   537
        private final AppContext appContext;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   538
        private final Runnable runnable;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   539
90ce3da70b43 Initial load
duke
parents:
diff changeset
   540
        public CreateThreadAction(AppContext ac, Runnable r) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   541
            appContext = ac;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   542
            runnable = r;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   543
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   544
11264
54f2f4c6bd30 7117008: Warnings cleanup day: reduce number of javac warnings in the sun.awt package
art
parents: 5506
diff changeset
   545
        public Thread run() {
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   546
            Thread t = new Thread(appContext.getThreadGroup(), runnable);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   547
            t.setContextClassLoader(appContext.getContextClassLoader());
90ce3da70b43 Initial load
duke
parents:
diff changeset
   548
            t.setPriority(Thread.NORM_PRIORITY + 1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   549
            t.setDaemon(true);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   550
            return t;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   551
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   552
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   553
90ce3da70b43 Initial load
duke
parents:
diff changeset
   554
    static void stopEventDispatchThreads() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   555
        for (AppContext appContext: getAppContexts()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   556
            if (appContext.isDisposed()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   557
                continue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   558
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   559
            Runnable r = new PostShutdownEventRunnable(appContext);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   560
            // For security reasons EventQueue.postEvent should only be called
90ce3da70b43 Initial load
duke
parents:
diff changeset
   561
            // on a thread that belongs to the corresponding thread group.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   562
            if (appContext != AppContext.getAppContext()) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   563
                // Create a thread that belongs to the thread group associated
90ce3da70b43 Initial load
duke
parents:
diff changeset
   564
                // with the AppContext and invokes EventQueue.postEvent.
11264
54f2f4c6bd30 7117008: Warnings cleanup day: reduce number of javac warnings in the sun.awt package
art
parents: 5506
diff changeset
   565
                PrivilegedAction<Thread> action = new CreateThreadAction(appContext, r);
54f2f4c6bd30 7117008: Warnings cleanup day: reduce number of javac warnings in the sun.awt package
art
parents: 5506
diff changeset
   566
                Thread thread = AccessController.doPrivileged(action);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   567
                thread.start();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   568
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   569
                r.run();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   570
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   571
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   572
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   573
90ce3da70b43 Initial load
duke
parents:
diff changeset
   574
    private MostRecentKeyValue mostRecentKeyValue = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   575
    private MostRecentKeyValue shadowMostRecentKeyValue = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   576
90ce3da70b43 Initial load
duke
parents:
diff changeset
   577
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   578
     * Returns the value to which the specified key is mapped in this context.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   579
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   580
     * @param   key   a key in the AppContext.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   581
     * @return  the value to which the key is mapped in this AppContext;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   582
     *          <code>null</code> if the key is not mapped to any value.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   583
     * @see     #put(Object, Object)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   584
     * @since   1.2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   585
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   586
    public Object get(Object key) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   587
        /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   588
         * The most recent reference should be updated inside a synchronized
90ce3da70b43 Initial load
duke
parents:
diff changeset
   589
         * block to avoid a race when put() and get() are executed in
90ce3da70b43 Initial load
duke
parents:
diff changeset
   590
         * parallel on different threads.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   591
         */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   592
        synchronized (table) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   593
            // Note: this most recent key/value caching is thread-hot.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   594
            // A simple test using SwingSet found that 72% of lookups
90ce3da70b43 Initial load
duke
parents:
diff changeset
   595
            // were matched using the most recent key/value.  By instantiating
90ce3da70b43 Initial load
duke
parents:
diff changeset
   596
            // a simple MostRecentKeyValue object on cache misses, the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   597
            // cache hits can be processed without synchronization.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   598
90ce3da70b43 Initial load
duke
parents:
diff changeset
   599
            MostRecentKeyValue recent = mostRecentKeyValue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   600
            if ((recent != null) && (recent.key == key)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   601
                return recent.value;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   602
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   603
90ce3da70b43 Initial load
duke
parents:
diff changeset
   604
            Object value = table.get(key);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   605
            if(mostRecentKeyValue == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   606
                mostRecentKeyValue = new MostRecentKeyValue(key, value);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   607
                shadowMostRecentKeyValue = new MostRecentKeyValue(key, value);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   608
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   609
                MostRecentKeyValue auxKeyValue = mostRecentKeyValue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   610
                shadowMostRecentKeyValue.setPair(key, value);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   611
                mostRecentKeyValue = shadowMostRecentKeyValue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   612
                shadowMostRecentKeyValue = auxKeyValue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   613
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   614
            return value;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   615
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   616
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   617
90ce3da70b43 Initial load
duke
parents:
diff changeset
   618
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   619
     * Maps the specified <code>key</code> to the specified
90ce3da70b43 Initial load
duke
parents:
diff changeset
   620
     * <code>value</code> in this AppContext.  Neither the key nor the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   621
     * value can be <code>null</code>.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   622
     * <p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   623
     * The value can be retrieved by calling the <code>get</code> method
90ce3da70b43 Initial load
duke
parents:
diff changeset
   624
     * with a key that is equal to the original key.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   625
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   626
     * @param      key     the AppContext key.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   627
     * @param      value   the value.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   628
     * @return     the previous value of the specified key in this
90ce3da70b43 Initial load
duke
parents:
diff changeset
   629
     *             AppContext, or <code>null</code> if it did not have one.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   630
     * @exception  NullPointerException  if the key or value is
90ce3da70b43 Initial load
duke
parents:
diff changeset
   631
     *               <code>null</code>.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   632
     * @see     #get(Object)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   633
     * @since   1.2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   634
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   635
    public Object put(Object key, Object value) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   636
        synchronized (table) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   637
            MostRecentKeyValue recent = mostRecentKeyValue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   638
            if ((recent != null) && (recent.key == key))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   639
                recent.value = value;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   640
            return table.put(key, value);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   641
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   642
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   643
90ce3da70b43 Initial load
duke
parents:
diff changeset
   644
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   645
     * Removes the key (and its corresponding value) from this
90ce3da70b43 Initial load
duke
parents:
diff changeset
   646
     * AppContext. This method does nothing if the key is not in the
90ce3da70b43 Initial load
duke
parents:
diff changeset
   647
     * AppContext.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   648
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   649
     * @param   key   the key that needs to be removed.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   650
     * @return  the value to which the key had been mapped in this AppContext,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   651
     *          or <code>null</code> if the key did not have a mapping.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   652
     * @since   1.2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   653
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   654
    public Object remove(Object key) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   655
        synchronized (table) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   656
            MostRecentKeyValue recent = mostRecentKeyValue;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   657
            if ((recent != null) && (recent.key == key))
90ce3da70b43 Initial load
duke
parents:
diff changeset
   658
                recent.value = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   659
            return table.remove(key);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   660
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   661
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   662
90ce3da70b43 Initial load
duke
parents:
diff changeset
   663
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   664
     * Returns the root ThreadGroup for all Threads contained within
90ce3da70b43 Initial load
duke
parents:
diff changeset
   665
     * this AppContext.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   666
     * @since   1.2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   667
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   668
    public ThreadGroup getThreadGroup() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   669
        return threadGroup;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   670
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   671
90ce3da70b43 Initial load
duke
parents:
diff changeset
   672
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   673
     * Returns the context ClassLoader that was used to create this
90ce3da70b43 Initial load
duke
parents:
diff changeset
   674
     * AppContext.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   675
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   676
     * @see java.lang.Thread#getContextClassLoader
90ce3da70b43 Initial load
duke
parents:
diff changeset
   677
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   678
    public ClassLoader getContextClassLoader() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   679
        return contextClassLoader;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   680
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   681
90ce3da70b43 Initial load
duke
parents:
diff changeset
   682
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   683
     * Returns a string representation of this AppContext.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   684
     * @since   1.2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   685
     */
113
d35e2fc341c7 6636370: minor corrections and simplification of code in AppContext
son
parents: 112
diff changeset
   686
    @Override
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   687
    public String toString() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   688
        return getClass().getName() + "[threadGroup=" + threadGroup.getName() + "]";
90ce3da70b43 Initial load
duke
parents:
diff changeset
   689
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   690
90ce3da70b43 Initial load
duke
parents:
diff changeset
   691
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   692
     * Returns an array of all the property change listeners
90ce3da70b43 Initial load
duke
parents:
diff changeset
   693
     * registered on this component.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   694
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   695
     * @return all of this component's <code>PropertyChangeListener</code>s
90ce3da70b43 Initial load
duke
parents:
diff changeset
   696
     *         or an empty array if no property change
90ce3da70b43 Initial load
duke
parents:
diff changeset
   697
     *         listeners are currently registered
90ce3da70b43 Initial load
duke
parents:
diff changeset
   698
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   699
     * @see      #addPropertyChangeListener
90ce3da70b43 Initial load
duke
parents:
diff changeset
   700
     * @see      #removePropertyChangeListener
90ce3da70b43 Initial load
duke
parents:
diff changeset
   701
     * @see      #getPropertyChangeListeners(java.lang.String)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   702
     * @see      java.beans.PropertyChangeSupport#getPropertyChangeListeners
90ce3da70b43 Initial load
duke
parents:
diff changeset
   703
     * @since    1.4
90ce3da70b43 Initial load
duke
parents:
diff changeset
   704
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   705
    public synchronized PropertyChangeListener[] getPropertyChangeListeners() {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   706
        if (changeSupport == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   707
            return new PropertyChangeListener[0];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   708
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   709
        return changeSupport.getPropertyChangeListeners();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   710
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   711
90ce3da70b43 Initial load
duke
parents:
diff changeset
   712
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   713
     * Adds a PropertyChangeListener to the listener list for a specific
90ce3da70b43 Initial load
duke
parents:
diff changeset
   714
     * property. The specified property may be one of the following:
90ce3da70b43 Initial load
duke
parents:
diff changeset
   715
     * <ul>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   716
     *    <li>if this AppContext is disposed ("disposed")</li>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   717
     * </ul>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   718
     * <ul>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   719
     *    <li>if this AppContext's unowned Windows have been disposed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   720
     *    ("guidisposed").  Code to cleanup after the GUI is disposed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   721
     *    (such as LookAndFeel.uninitialize()) should execute in response to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   722
     *    this property being fired.  Notifications for the "guidisposed"
90ce3da70b43 Initial load
duke
parents:
diff changeset
   723
     *    property are sent on the event dispatch thread.</li>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   724
     * </ul>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   725
     * <p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   726
     * If listener is null, no exception is thrown and no action is performed.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   727
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   728
     * @param propertyName one of the property names listed above
90ce3da70b43 Initial load
duke
parents:
diff changeset
   729
     * @param listener the PropertyChangeListener to be added
90ce3da70b43 Initial load
duke
parents:
diff changeset
   730
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   731
     * @see #removePropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   732
     * @see #getPropertyChangeListeners(java.lang.String)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   733
     * @see #addPropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   734
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   735
    public synchronized void addPropertyChangeListener(
90ce3da70b43 Initial load
duke
parents:
diff changeset
   736
                             String propertyName,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   737
                             PropertyChangeListener listener) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   738
        if (listener == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   739
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   740
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   741
        if (changeSupport == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   742
            changeSupport = new PropertyChangeSupport(this);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   743
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   744
        changeSupport.addPropertyChangeListener(propertyName, listener);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   745
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   746
90ce3da70b43 Initial load
duke
parents:
diff changeset
   747
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   748
     * Removes a PropertyChangeListener from the listener list for a specific
90ce3da70b43 Initial load
duke
parents:
diff changeset
   749
     * property. This method should be used to remove PropertyChangeListeners
90ce3da70b43 Initial load
duke
parents:
diff changeset
   750
     * that were registered for a specific bound property.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   751
     * <p>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   752
     * If listener is null, no exception is thrown and no action is performed.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   753
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   754
     * @param propertyName a valid property name
90ce3da70b43 Initial load
duke
parents:
diff changeset
   755
     * @param listener the PropertyChangeListener to be removed
90ce3da70b43 Initial load
duke
parents:
diff changeset
   756
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   757
     * @see #addPropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   758
     * @see #getPropertyChangeListeners(java.lang.String)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   759
     * @see #removePropertyChangeListener(java.beans.PropertyChangeListener)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   760
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   761
    public synchronized void removePropertyChangeListener(
90ce3da70b43 Initial load
duke
parents:
diff changeset
   762
                             String propertyName,
90ce3da70b43 Initial load
duke
parents:
diff changeset
   763
                             PropertyChangeListener listener) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   764
        if (listener == null || changeSupport == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   765
            return;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   766
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   767
        changeSupport.removePropertyChangeListener(propertyName, listener);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   768
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   769
90ce3da70b43 Initial load
duke
parents:
diff changeset
   770
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   771
     * Returns an array of all the listeners which have been associated
90ce3da70b43 Initial load
duke
parents:
diff changeset
   772
     * with the named property.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   773
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   774
     * @return all of the <code>PropertyChangeListeners</code> associated with
90ce3da70b43 Initial load
duke
parents:
diff changeset
   775
     *         the named property or an empty array if no listeners have
90ce3da70b43 Initial load
duke
parents:
diff changeset
   776
     *         been added
90ce3da70b43 Initial load
duke
parents:
diff changeset
   777
     *
90ce3da70b43 Initial load
duke
parents:
diff changeset
   778
     * @see #addPropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   779
     * @see #removePropertyChangeListener(java.lang.String, java.beans.PropertyChangeListener)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   780
     * @see #getPropertyChangeListeners
90ce3da70b43 Initial load
duke
parents:
diff changeset
   781
     * @since 1.4
90ce3da70b43 Initial load
duke
parents:
diff changeset
   782
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   783
    public synchronized PropertyChangeListener[] getPropertyChangeListeners(
90ce3da70b43 Initial load
duke
parents:
diff changeset
   784
                                                        String propertyName) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   785
        if (changeSupport == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   786
            return new PropertyChangeListener[0];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   787
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   788
        return changeSupport.getPropertyChangeListeners(propertyName);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   789
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   790
}
90ce3da70b43 Initial load
duke
parents:
diff changeset
   791
90ce3da70b43 Initial load
duke
parents:
diff changeset
   792
final class MostRecentKeyValue {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   793
    Object key;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   794
    Object value;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   795
    MostRecentKeyValue(Object k, Object v) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   796
        key = k;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   797
        value = v;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   798
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   799
    void setPair(Object k, Object v) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   800
        key = k;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   801
        value = v;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   802
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   803
}