LWCursorManager.java 4.6 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38
/*
 * Copyright (c) 2011, 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 sun.lwawt;

import java.awt.Component;
import java.awt.Container;
import java.awt.Cursor;
import java.awt.Point;

import java.util.concurrent.atomic.AtomicBoolean;

import sun.awt.SunToolkit;

public abstract class LWCursorManager {

39 40 41 42 43
    /**
     * A flag to indicate if the update is scheduled, so we don't process it
     * twice.
     */
    private final AtomicBoolean updatePending = new AtomicBoolean(false);
44 45 46 47

    protected LWCursorManager() {
    }

48
    /**
49 50 51 52 53
     * Sets the cursor to correspond the component currently under mouse.
     *
     * This method should not be executed on the toolkit thread as it
     * calls to user code (e.g. Container.findComponentAt).
     */
54
    public final void updateCursor() {
55 56 57 58
        updatePending.set(false);
        updateCursorImpl();
    }

59
    /**
60 61 62 63 64 65
     * Schedules updating the cursor on the corresponding event dispatch
     * thread for the given window.
     *
     * This method is called on the toolkit thread as a result of a
     * native update cursor request (e.g. WM_SETCURSOR on Windows).
     */
66
    public final void updateCursorLater(final LWWindowPeer window) {
67 68 69 70 71 72 73 74 75 76 77 78
        if (updatePending.compareAndSet(false, true)) {
            Runnable r = new Runnable() {
                @Override
                public void run() {
                    updateCursor();
                }
            };
            SunToolkit.executeOnEventHandlerThread(window.getTarget(), r);
        }
    }

    private void updateCursorImpl() {
79 80 81 82 83 84 85 86 87 88 89
        final Point cursorPos = getCursorPosition();
        final Component c = findComponent(cursorPos);
        final Cursor cursor;
        final Object peer = LWToolkit.targetToPeer(c);
        if (peer instanceof LWComponentPeer) {
            final LWComponentPeer<?, ?> lwpeer = (LWComponentPeer<?, ?>) peer;
            final Point p = lwpeer.getLocationOnScreen();
            cursor = lwpeer.getCursor(new Point(cursorPos.x - p.x,
                                                cursorPos.y - p.y));
        } else {
            cursor = (c != null) ? c.getCursor() : null;
90
        }
91 92 93 94 95
        setCursor(cursor);
    }

    /**
     * Returns the first visible, enabled and showing component under cursor.
96
     * Returns null for modal blocked windows.
97 98
     *
     * @param cursorPos Current cursor position.
99
     * @return Component or null.
100 101 102 103
     */
    private static final Component findComponent(final Point cursorPos) {
        final LWComponentPeer<?, ?> peer = LWWindowPeer.getPeerUnderCursor();
        Component c = null;
104
        if (peer != null && peer.getWindowPeerOrSelf().getBlocker() == null) {
105
            c = peer.getTarget();
106
            if (c instanceof Container) {
107 108 109
                final Point p = peer.getLocationOnScreen();
                c = ((Container) c).findComponentAt(cursorPos.x - p.x,
                                                    cursorPos.y - p.y);
110 111 112 113 114 115 116 117
            }
            while (c != null) {
                if (c.isVisible() && c.isEnabled() && (c.getPeer() != null)) {
                    break;
                }
                c = c.getParent();
            }
        }
118
        return c;
119 120
    }

121
    /**
122 123 124 125 126
     * Returns the current cursor position.
     */
    // TODO: make it public to reuse for MouseInfo
    protected abstract Point getCursorPosition();

127 128 129 130
    /**
     * Sets a cursor. The cursor can be null if the mouse is not over a Java
     * window.
     * @param cursor the new {@code Cursor}.
131
     */
132
    protected abstract void setCursor(Cursor cursor);
133
}