All Downloads are FREE. Search and download functionalities are using the official Maven repository.

javafx.application.Platform Maven / Gradle / Ivy

There is a newer version: 24-ea+19
Show newest version
/*
 * Copyright (c) 2010, 2024, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package javafx.application;

import com.sun.javafx.application.PlatformImpl;
import com.sun.javafx.tk.Toolkit;
import java.util.Optional;
import javafx.beans.property.ReadOnlyBooleanProperty;
import javafx.beans.property.ReadOnlyBooleanWrapper;
import javafx.beans.property.ReadOnlyObjectProperty;
import javafx.collections.ObservableMap;
import javafx.scene.input.KeyCode;
import javafx.scene.paint.Color;

/**
 * Application platform support class.
 * @since JavaFX 2.0
 */
public final class Platform {

    // To prevent instantiation
    private Platform() {
    }

    /**
     * This method starts the JavaFX runtime. The specified Runnable will then be
     * called on the JavaFX Application Thread. In general it is not necessary to
     * explicitly call this method, since it is invoked as a consequence of
     * how most JavaFX applications are built. However there are valid use cases
     * for calling this method directly. Because this method starts the JavaFX
     * runtime, there is not yet any JavaFX Application Thread, so it is normal
     * that this method is called directly on the main thread of the application.
     *
     * 

* This method may or may not return to the caller before the run method * of the specified Runnable has been called. In any case, this * method is not blocked on the specified Runnable being executed. * Once this method * returns, you may call {@link #runLater(Runnable)} with additional Runnables. * Those Runnables will be called, also on the JavaFX Application Thread, * after the Runnable passed into this method has been called. *

* *

As noted, it is normally the case that the JavaFX Application Thread * is started automatically. It is important that this method only be called * when the JavaFX runtime has not yet been initialized. Situations where * the JavaFX runtime is started automatically include: *

* *
    *
  • For standard JavaFX applications that extend {@link Application}, and * use either the Java launcher or one of the launch methods in the * Application class to launch the application, the FX runtime is * initialized automatically by the launcher before the {@code Application} * class is loaded.
  • *
  • For Swing applications that use {@link javafx.embed.swing.JFXPanel} * to display FX content, the * FX runtime is initialized when the first {@code JFXPanel} instance is * constructed.
  • *
  • For SWT application that use {@code FXCanvas} to display FX content, * the FX runtime is initialized when the first {@code FXCanvas} instance is * constructed.
  • *
* *

When an application does not follow any of these common approaches, * then it becomes the responsibility of the developer to manually start the * JavaFX runtime by calling this startup method. *

* *

Calling this method when the JavaFX runtime is already running will result in an * {@link IllegalStateException} being thrown - it is only valid to request * that the JavaFX runtime be started once. *

* *

Note: The JavaFX classes must be loaded from a set of * named {@code javafx.*} modules on the module path. * Loading the JavaFX classes from the classpath is not supported. * A warning is logged when the JavaFX runtime is started if the JavaFX * classes are not loaded from the expected named module. * This warning is logged regardless of whether the JavaFX runtime was * started by calling this method or automatically as described above. * * @throws IllegalStateException if the JavaFX runtime is already running * * @param runnable the Runnable whose run method will be executed on the * JavaFX Application Thread once it has been started * * @see Application * * @since 9 */ public static void startup(Runnable runnable) { PlatformImpl.startup(runnable, true); } /** * Run the specified Runnable on the JavaFX Application Thread at some * unspecified * time in the future. This method, which may be called from any thread, * will post the Runnable to an event queue and then return immediately to * the caller. The Runnables are executed in the order they are posted. * A runnable passed into the runLater method will be * executed before any Runnable passed into a subsequent call to runLater. * If this method is called after the JavaFX runtime has been shutdown, the * call will be ignored: the Runnable will not be executed and no * exception will be thrown. * *

* NOTE: applications should avoid flooding JavaFX with too many * pending Runnables. Otherwise, the application may become unresponsive. * Applications are encouraged to batch up multiple operations into fewer * runLater calls. * Additionally, long-running operations should be done on a background * thread where possible, freeing up the JavaFX Application Thread for GUI * operations. *

* *

* This method must not be called before the FX runtime has been * initialized. For standard JavaFX applications that extend * {@link Application}, and use either the Java launcher or one of the * launch methods in the Application class to launch the application, * the FX runtime is initialized by the launcher before the Application * class is loaded. * For Swing applications that use JFXPanel to display FX content, the FX * runtime is initialized when the first JFXPanel instance is constructed. * For SWT application that use FXCanvas to display FX content, the FX * runtime is initialized when the first FXCanvas instance is constructed. * For applications that do not follow any of these approaches, then it is * necessary to manually start the JavaFX runtime by calling * {@link #startup(Runnable)} once. *

* *

* Memory consistency effects: Actions in a thread prior to submitting a * {@code runnable} to this method happen-before actions performed * by the runnable in the JavaFX Application Thread. *

* * @param runnable the Runnable whose run method will be executed on the * JavaFX Application Thread * * @throws IllegalStateException if the FX runtime has not been initialized * * @see Application */ public static void runLater(Runnable runnable) { PlatformImpl.runLater(runnable); } // NOTE: Add the following if we decide to expose it publicly // public static void runAndWait(Runnable runnable) { // PlatformImpl.runAndWait(runnable); // } /** * Requests the Java Runtime to perform a pulse. This will run a pulse * even if there are no animation timers, scene graph modifications, * or window events that would otherwise cause the pulse to run. * If no pulse is in progress, then one will be scheduled to * run the next time the pulse timer fires. * If there is already a pulse running, then * at least one more pulse after the current pulse will be scheduled. * This method may be called on any thread. * * @since 9 */ public static void requestNextPulse() { Toolkit.getToolkit().requestNextPulse(); } /** * Returns true if the calling thread is the JavaFX Application Thread. * Use this call to ensure that a given task is being executed * (or not being executed) on the JavaFX Application Thread. * * @return true if running on the JavaFX Application Thread */ public static boolean isFxApplicationThread() { return PlatformImpl.isFxApplicationThread(); } /** * Causes the JavaFX application to terminate. If this method is called * after the Application start method is called, then the JavaFX launcher * will call the Application stop method and terminate the JavaFX * application thread. The launcher thread will then shutdown. If there * are no other non-daemon threads that are running, the Java VM will exit. * If this method is called from the Preloader or the Application init * method, then the Application stop method may not be called. * *

This method may be called from any thread.

*/ public static void exit() { PlatformImpl.exit(); } /** * Sets the implicitExit attribute to the specified value. If this * attribute is true, the JavaFX runtime will implicitly shutdown * when the last window is closed; the JavaFX launcher will call the * {@link Application#stop} method and terminate the JavaFX * application thread. * If this attribute is false, the application will continue to * run normally even after the last window is closed, until the * application calls {@link #exit}. * The default value is true. * *

This method may be called from any thread.

* * @param implicitExit a flag indicating whether or not to implicitly exit * when the last window is closed. * @since JavaFX 2.2 */ public static void setImplicitExit(boolean implicitExit) { PlatformImpl.setImplicitExit(implicitExit); } /** * Gets the value of the implicitExit attribute. * *

This method may be called from any thread.

* * @return the implicitExit attribute * @since JavaFX 2.2 */ public static boolean isImplicitExit() { return PlatformImpl.isImplicitExit(); } /** * Queries whether a specific conditional feature is supported * by the platform. *

* For example: *

     * // Query whether filter effects are supported
     * if (Platform.isSupported(ConditionalFeature.EFFECT)) {
     *    // use effects
     * }
     * 
* * @param feature the conditional feature in question. * @return true if a specific conditional feature is supported by the * platform, otherwise false */ public static boolean isSupported(ConditionalFeature feature) { return PlatformImpl.isSupported(feature); } /** * Enter a nested event loop and block until the corresponding * exitNestedEventLoop call is made. * The key passed into this method is used to * uniquely identify the matched enter/exit pair. This method creates a * new nested event loop and blocks until the corresponding * exitNestedEventLoop method is called with the same key. * The return value of this method will be the {@code rval} * object supplied to the exitNestedEventLoop method call that unblocks it. * *

* This method must either be called from an input event handler or * from the run method of a Runnable passed to * {@link javafx.application.Platform#runLater Platform.runLater}. * It must not be called during animation or layout processing. *

* * @param key the Object that identifies the nested event loop, which * must not be null * * @throws IllegalArgumentException if the specified key is associated * with a nested event loop that has not yet returned * * @throws NullPointerException if the key is null * * @throws IllegalStateException if this method is called during * animation or layout processing. * * @throws IllegalStateException if this method is called on a thread * other than the JavaFX Application Thread. * * @return the value passed into the corresponding call to exitEventLoop * * @since 9 */ public static Object enterNestedEventLoop(Object key) { return Toolkit.getToolkit().enterNestedEventLoop(key); } /** * Exit a nested event loop and unblock the caller of the * corresponding enterNestedEventLoop. * The key passed into this method is used to * uniquely identify the matched enter/exit pair. This method causes the * nested event loop that was previously created with the key to exit and * return control to the caller. If the specified nested event loop is not * the inner-most loop then it will not return until all other inner loops * also exit. * * @param key the Object that identifies the nested event loop, which * must not be null * * @param rval an Object that is returned to the caller of the * corresponding enterNestedEventLoop. This may be null. * * @throws IllegalArgumentException if the specified key is not associated * with an active nested event loop * * @throws NullPointerException if the key is null * * @throws IllegalStateException if this method is called on a thread * other than the FX Application thread * * @since 9 */ public static void exitNestedEventLoop(Object key, Object rval) { Toolkit.getToolkit().exitNestedEventLoop(key, rval); } /** * Returns a flag indicating whether the key corresponding to {@code keyCode} * is in the locked (or "on") state. * {@code keyCode} must be one of: {@link KeyCode#CAPS} or * {@link KeyCode#NUM_LOCK}. * If the underlying system is not able to determine the state of the * specified {@code keyCode}, an empty {@code Optional} is returned. * If the keyboard attached to the system doesn't have the specified key, * an {@code Optional} containing {@code false} is returned. * This method must be called on the JavaFX Application thread. * * @param keyCode the {@code KeyCode} of the lock state to query * * @return the lock state of the key corresponding to {@code keyCode}, * or an empty {@code Optional} if the system cannot determine its state * * @throws IllegalArgumentException if {@code keyCode} is not one of the * valid {@code KeyCode} values * * @throws IllegalStateException if this method is called on a thread * other than the JavaFX Application Thread * * @since 17 */ public static Optional isKeyLocked(KeyCode keyCode) { Toolkit.getToolkit().checkFxUserThread(); switch (keyCode) { case CAPS: case NUM_LOCK: break; default: throw new IllegalArgumentException("Invalid KeyCode"); } return Toolkit.getToolkit().isKeyLocked(keyCode); } /** * Checks whether a nested event loop is running, returning true to indicate * that one is, and false if there are no nested event loops currently * running. * This method must be called on the JavaFX Application thread. * * @return true if there is a nested event loop running, and false otherwise. * * @throws IllegalStateException if this method is called on a thread * other than the JavaFX Application Thread. * * @since 9 */ public static boolean isNestedLoopRunning() { return Toolkit.getToolkit().isNestedLoopRunning(); } /** * Indicates whether a nested event loop can be started from the current thread in the current state. * A nested event loop can be started from an event handler or from a {@code Runnable} passed to * {@link #runLater(Runnable)}. * This method must be called on the JavaFX Application thread. * * @return {@code true} if a nested event loop can be started, and {@code false} otherwise. * * @throws IllegalStateException if this method is called on a thread other than the JavaFX Application Thread. * * @since 21 */ public static boolean canStartNestedEventLoop() { return Toolkit.getToolkit().canStartNestedEventLoop(); } private static ReadOnlyBooleanWrapper accessibilityActiveProperty; public static boolean isAccessibilityActive() { return accessibilityActiveProperty == null ? false : accessibilityActiveProperty.get(); } /** * Indicates whether or not accessibility is active. * This property is typically set to true the first time an * assistive technology, such as a screen reader, requests * information about any JavaFX window or its children. * *

This method may be called from any thread.

* * @return the read-only boolean property indicating if accessibility is active * * @since JavaFX 8u40 */ public static ReadOnlyBooleanProperty accessibilityActiveProperty() { if (accessibilityActiveProperty == null) { accessibilityActiveProperty = new ReadOnlyBooleanWrapper(Platform.class, "accessibilityActive"); accessibilityActiveProperty.bind(PlatformImpl.accessibilityActiveProperty()); } return accessibilityActiveProperty.getReadOnlyProperty(); } /** * Gets the preferences of the current platform. *

* The map returned from this method is unmodifiable, which means that keys and values cannot * be added, removed, or updated. Calling any mutator method on the map will always cause * {@code UnsupportedOperationException} to be thrown. However, the mappings will be updated * by JavaFX when the operating system reports that a platform preference has changed. * * @return the {@code Preferences} instance * @see Windows preferences * @see macOS preferences * @see Linux preferences * @since 22 */ public static Preferences getPreferences() { PlatformImpl.checkPreferencesSupport(); return PlatformImpl.getPlatformPreferences(); } /** * Contains UI preferences of the current platform. *

* {@code Preferences} extends {@link ObservableMap} to expose platform preferences as key-value pairs. * The map is unmodifiable, which means that keys and values cannot be added, removed, or updated. * Calling any mutator method on the map will always cause {@code UnsupportedOperationException} to be thrown. * However, the mappings will be updated by JavaFX when the operating system reports that a platform * preference has changed. *

* For convenience, {@link #getInteger}, {@link #getDouble}, {@link #getBoolean}, {@link #getString}, * {@link #getColor}, and {@link #getValue} are provided as typed alternatives to the untyped * {@link #get} method. *

* The preferences that are reported by the platform may be dependent on the operating system version * and its current configuration, so applications should not assume that a particular preference is * always available. *

* The following preferences are potentially available on the specified platforms: *

* * * * * * * * * * * * * * * * * * * * * * * *
Windows
{@code Windows.SPI.HighContrast}{@link Boolean}
{@code Windows.SPI.HighContrastColorScheme}{@link String}
{@code Windows.SysColor.COLOR_3DFACE}{@link Color}
{@code Windows.SysColor.COLOR_BTNTEXT}{@link Color}
{@code Windows.SysColor.COLOR_GRAYTEXT}{@link Color}
{@code Windows.SysColor.COLOR_HIGHLIGHT}{@link Color}
{@code Windows.SysColor.COLOR_HIGHLIGHTTEXT}{@link Color}
{@code Windows.SysColor.COLOR_HOTLIGHT}{@link Color}
{@code Windows.SysColor.COLOR_WINDOW}{@link Color}
{@code Windows.SysColor.COLOR_WINDOWTEXT}{@link Color}
{@code Windows.UIColor.Background}{@link Color}
{@code Windows.UIColor.Foreground}{@link Color}
{@code Windows.UIColor.AccentDark3}{@link Color}
{@code Windows.UIColor.AccentDark2}{@link Color}
{@code Windows.UIColor.AccentDark1}{@link Color}
{@code Windows.UIColor.Accent}{@link Color}
{@code Windows.UIColor.AccentLight1}{@link Color}
{@code Windows.UIColor.AccentLight2}{@link Color}
{@code Windows.UIColor.AccentLight3}{@link Color}
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
macOS
{@code macOS.NSColor.labelColor}{@link Color}
{@code macOS.NSColor.secondaryLabelColor}{@link Color}
{@code macOS.NSColor.tertiaryLabelColor}{@link Color}
{@code macOS.NSColor.quaternaryLabelColor}{@link Color}
{@code macOS.NSColor.textColor}{@link Color}
{@code macOS.NSColor.placeholderTextColor}{@link Color}
{@code macOS.NSColor.selectedTextColor}{@link Color}
{@code macOS.NSColor.textBackgroundColor}{@link Color}
{@code macOS.NSColor.selectedTextBackgroundColor}{@link Color}
{@code macOS.NSColor.keyboardFocusIndicatorColor}{@link Color}
{@code macOS.NSColor.unemphasizedSelectedTextColor}{@link Color}
{@code macOS.NSColor.unemphasizedSelectedTextBackgroundColor}{@link Color}
{@code macOS.NSColor.linkColor}{@link Color}
{@code macOS.NSColor.separatorColor}{@link Color}
{@code macOS.NSColor.selectedContentBackgroundColor}{@link Color}
{@code macOS.NSColor.unemphasizedSelectedContentBackgroundColor}{@link Color}
{@code macOS.NSColor.selectedMenuItemTextColor}{@link Color}
{@code macOS.NSColor.gridColor}{@link Color}
{@code macOS.NSColor.headerTextColor}{@link Color}
{@code macOS.NSColor.alternatingContentBackgroundColors}{@link Color}{@code []}
{@code macOS.NSColor.controlAccentColor}{@link Color}
{@code macOS.NSColor.controlColor}{@link Color}
{@code macOS.NSColor.controlBackgroundColor}{@link Color}
{@code macOS.NSColor.controlTextColor}{@link Color}
{@code macOS.NSColor.disabledControlTextColor}{@link Color}
{@code macOS.NSColor.selectedControlColor}{@link Color}
{@code macOS.NSColor.selectedControlTextColor}{@link Color}
{@code macOS.NSColor.alternateSelectedControlTextColor}{@link Color}
{@code macOS.NSColor.currentControlTint}{@link String}
{@code macOS.NSColor.windowBackgroundColor}{@link Color}
{@code macOS.NSColor.windowFrameTextColor}{@link Color}
{@code macOS.NSColor.underPageBackgroundColor}{@link Color}
{@code macOS.NSColor.findHighlightColor}{@link Color}
{@code macOS.NSColor.highlightColor}{@link Color}
{@code macOS.NSColor.shadowColor}{@link Color}
{@code macOS.NSColor.systemBlueColor}{@link Color}
{@code macOS.NSColor.systemBrownColor}{@link Color}
{@code macOS.NSColor.systemGrayColor}{@link Color}
{@code macOS.NSColor.systemGreenColor}{@link Color}
{@code macOS.NSColor.systemIndigoColor}{@link Color}
{@code macOS.NSColor.systemOrangeColor}{@link Color}
{@code macOS.NSColor.systemPinkColor}{@link Color}
{@code macOS.NSColor.systemPurpleColor}{@link Color}
{@code macOS.NSColor.systemRedColor}{@link Color}
{@code macOS.NSColor.systemTealColor}{@link Color}
{@code macOS.NSColor.systemYellowColor}{@link Color}
* * * * * * * * * * * * * * * * * * * * * * * * *
Linux
{@code GTK.theme_name}{@link String}
{@code GTK.theme_fg_color}{@link Color}
{@code GTK.theme_bg_color}{@link Color}
{@code GTK.theme_base_color}{@link Color}
{@code GTK.theme_selected_bg_color}{@link Color}
{@code GTK.theme_selected_fg_color}{@link Color}
{@code GTK.theme_unfocused_fg_color}{@link Color}
{@code GTK.theme_unfocused_bg_color}{@link Color}
{@code GTK.theme_unfocused_base_color}{@link Color}
{@code GTK.theme_unfocused_selected_bg_color}{@link Color}
{@code GTK.theme_unfocused_selected_fg_color}{@link Color}
{@code GTK.insensitive_bg_color}{@link Color}
{@code GTK.insensitive_fg_color}{@link Color}
{@code GTK.insensitive_base_color}{@link Color}
{@code GTK.borders}{@link Color}
{@code GTK.unfocused_borders}{@link Color}
{@code GTK.warning_color}{@link Color}
{@code GTK.error_color}{@link Color}
{@code GTK.success_color}{@link Color}
* * @since 22 */ public interface Preferences extends ObservableMap { /** * The platform color scheme, which specifies whether applications should prefer light text on * dark backgrounds, or dark text on light backgrounds. *

* If the platform does not report color preferences, this property defaults to {@code LIGHT}. * * @return the {@code colorScheme} property * @defaultValue {@link ColorScheme#LIGHT} */ ReadOnlyObjectProperty colorSchemeProperty(); ColorScheme getColorScheme(); /** * The color used for background regions. *

* If the platform does not report a background color, this property defaults to {@code WHITE}. * * @return the {@code backgroundColor} property * @defaultValue {@link Color#WHITE} */ ReadOnlyObjectProperty backgroundColorProperty(); Color getBackgroundColor(); /** * The color used for foreground elements like text. *

* If the platform does not report a foreground color, this property defaults to {@code BLACK}. * * @return the {@code foregroundColor} property * @defaultValue {@link Color#BLACK} */ ReadOnlyObjectProperty foregroundColorProperty(); Color getForegroundColor(); /** * The accent color, which can be used to highlight the active or important part of a * control and make it stand out from the rest of the user interface. It is usually a * vivid color that contrasts with the foreground and background colors. *

* If the platform does not report an accent color, this property defaults to vivid blue * (corresponding to the hex color value {@code #157EFB}). * * @return the {@code accentColor} property * @defaultValue {@code #157EFB} */ ReadOnlyObjectProperty accentColorProperty(); Color getAccentColor(); /** * Returns an optional {@code Integer} to which the specified key is mapped. * * @param key the key * @throws NullPointerException if {@code key} is null * @throws IllegalArgumentException if the key is not mappable to an {@code Integer} * @return the optional {@code Integer} to which the key is mapped */ Optional getInteger(String key); /** * Returns an optional {@code Double} to which the specified key is mapped. * * @param key the key * @throws NullPointerException if {@code key} is null * @throws IllegalArgumentException if the key is not mappable to a {@code Double} * @return the optional {@code Double} to which the key is mapped */ Optional getDouble(String key); /** * Returns an optional {@code Boolean} to which the specified key is mapped. * * @param key the key * @throws NullPointerException if {@code key} is null * @throws IllegalArgumentException if the key is not mappable to a {@code Boolean} * @return the optional {@code Boolean} to which the key is mapped */ Optional getBoolean(String key); /** * Returns an optional {@code String} to which the specified key is mapped. * * @param key the key * @throws NullPointerException if {@code key} is null * @throws IllegalArgumentException if the key is not mappable to a {@code String} * @return the optional {@code String} to which the key is mapped */ Optional getString(String key); /** * Returns an optional {@code Color} to which the specified key is mapped. * * @param key the key * @throws NullPointerException if {@code key} is null * @throws IllegalArgumentException if the key is not mappable to a {@code Color} * @return the optional {@code Color} instance to which the key is mapped */ Optional getColor(String key); /** * Returns an optional value to which the specified key is mapped. * * @param the type of the value * @param key the key * @param type the type of the value * @throws NullPointerException if {@code key} or {@code type} is null * @throws IllegalArgumentException if the key is not mappable to a value of type {@code T} * @return the optional value to which the key is mapped */ Optional getValue(String key, Class type); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy