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

com.sun.javafx.scene.control.behavior.TextBinding Maven / Gradle / Ivy

There is a newer version: 24-ea+15
Show newest version
/*
 * Copyright (c) 2010, 2021, 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 com.sun.javafx.scene.control.behavior;

import javafx.scene.input.KeyEvent;
import javafx.scene.input.KeyCombination;
import java.util.Objects;

/**
 * 

* The syntax of the string content is as follows: *

* *
    *
  • *

    * Simple mnemonic: the first character preceded by the first {@code _} * character will be treated as the mnemonic. For example, "E_xit" will cause * the text to become "Exit" and the mnemonic will be "x". This is the most * common designation of a mnemonic, and the skin for a control will present the * mnemonic in the string with an underline. To prevent {@code _} from being * treated as the mnemonic prefix character, repeat it twice in a row. A * mnemonic is not required. *

    *
  • *
  • *

    * Extended mnemonic: an optional representation of a mnemonic is * {@code _(c)}, where {@code c} is the mnemonic character. For example, * "Exit_(q)" will cause the text to become "Exit" and the * {@link #extendedMnemonicText} to become "(q)". This is typically provided in * translated strings to support mnemonics where the main text does not have any * characters that map to keyboard keys. In these cases, the skin for the * control will typically present the mnemonic surrounded by parentheses. The * skin will also honor the hiding and presentation of the extended mnemonic * string on platforms where the mnemonic is only displayed when the mnemonic * modifier key is pressed. *

*/ public class TextBinding { /** * the marker symbol used when parsing for mnemonics */ private final char MNEMONIC_SYMBOL = '_'; /** * The source text before mnemonic markup has been removed. */ private String sourceText = null; /** * The text with any markup for the mnemonic and accelerator removed. */ private String text = null; /** * Returns the text with any markup for the mnemonic and accelerator removed * * @return the text with any markup for the mnemonic and accelerator removed */ public String getText() { return text; } /** * The mnemonic or {@code null} if there is no mnemonic. */ private String mnemonic = null; private KeyCombination mnemonicKeyCombination = null; /** * Returns the mnemonic or {@code null} if there is no * mnemonic. * * @return the mnemonic or {@code null} if there is no * mnemonic */ public String getMnemonic() { return mnemonic; } /** * Returns the mnemonic KeyCombination or {@code null} if there is no * mnemonic. * * @return the mnemonic KeyCombination or {@code null} if there is no * mnemonic */ public KeyCombination getMnemonicKeyCombination() { if (mnemonic != null && mnemonicKeyCombination == null) { mnemonicKeyCombination = new MnemonicKeyCombination(mnemonic); } return mnemonicKeyCombination; } /** * The index of the mnemonic character in the text property or -1 if there * is no mnemonic character in the text. This is only non-negative if the * simple {@code _c} syntax was used to specify the mnemonic. */ private int mnemonicIndex = -1; /** * Returns the index of the mnemonic character in the text property or -1 if * there is no mnemonic character in the text. This is only non-negative if * the simple {@code _c} syntax was used to specify the mnemonic. * * @return the index of the mnemonic character in the text property or -1 if * there is no mnemonic character in the text */ public int getMnemonicIndex() { return mnemonicIndex; } /** * The extended mnemonic text (if it exists). This is only non-null if the * extended mnemonic syntax was used to specify the mnemonic. */ private String extendedMnemonicText = null; /** * Returns the extended mnemonic text (if it exists). This is only non-null * if the extended mnemonic syntax was used to specify the mnemonic. * * @return the extended mnemonic text (if it exists) or null */ public String getExtendedMnemonicText() { return extendedMnemonicText; } /** * Creates a new TextBinding instance from the given string. * * @param s the action text string */ public TextBinding(String s) { update(s); } public void update(String s) { if (!Objects.equals(sourceText, s)) { sourceText = s; mnemonic = null; mnemonicKeyCombination = null; mnemonicIndex = -1; extendedMnemonicText = null; parseAndSplit(s); } } /** * Parse and split the given string into the appropriate segments. */ private void parseAndSplit(String s) { if (s == null || s.length() == 0) { text = s; return; } StringBuilder builder = new StringBuilder(s.length()); int i = 0; // Parse the input string and stop after the first mnemonic. for (int length = s.length(); i < length; ++i) { if (isEscapedMnemonicSymbol(s, i)) { builder.append(s.charAt(i++)); } else if (isExtendedMnemonic(s, i)) { mnemonic = String.valueOf(s.charAt(i + 2)); mnemonicIndex = i; extendedMnemonicText = s.substring(i + 1, i + 4); i += 4; break; } else if (isSimpleMnemonic(s, i)) { char c = s.charAt(i + 1); mnemonic = String.valueOf(c); mnemonicIndex = i; i += 1; break; } else { builder.append(s.charAt(i)); } } if (s.length() > i) { builder.append(s.substring(i)); } text = builder.toString(); } /** * Determines whether the string contains an escaped mnemonic symbol at the specified position. */ private boolean isEscapedMnemonicSymbol(String s, int position) { return s.length() > position + 1 && s.charAt(position) == MNEMONIC_SYMBOL && s.charAt(position + 1) == MNEMONIC_SYMBOL; } /** * Determines whether the string contains a simple mnemonic at the specified position. * A simple mnemonic is any two-character string similar to "_x", where x is not an * underscore or a whitespace character. */ private boolean isSimpleMnemonic(String s, int position) { return s.length() > position + 1 && s.charAt(position) == MNEMONIC_SYMBOL && s.charAt(position + 1) != MNEMONIC_SYMBOL && !Character.isWhitespace(s.charAt(position + 1)); } /** * Determines whether the string contains an extended mnemonic at the specified position. * An extended mnemonic is any four-character string similar to "_(x)", where x is any * character except whitespace. */ private boolean isExtendedMnemonic(String s, int position) { return s.length() > position + 3 && s.charAt(position) == MNEMONIC_SYMBOL && s.charAt(position + 1) == '(' && !Character.isWhitespace(s.charAt(position + 2)) && s.charAt(position + 3) == ')'; } /** * A modified version of KeyCharacterCombination, which matches * on the text property of a KeyEvent instead of on the KeyCode. */ public static class MnemonicKeyCombination extends KeyCombination { private String character = ""; /** * Constructs a {@code MnemonicKeyCombination} for the specified main key * character. * * @param character the main key character */ public MnemonicKeyCombination(String character) { super(com.sun.javafx.PlatformUtil.isMac() ? KeyCombination.META_DOWN : KeyCombination.ALT_DOWN); this.character = character; } /** * Gets the key character associated with this key combination. * @return The key character associated with this key combination */ public final String getCharacter() { return character; } /** * Tests whether this key combination matches the key combination in the * given {@code KeyEvent}. * * @param event the key event * @return {@code true} if the key combinations match, {@code false} * otherwise */ @Override public boolean match(final KeyEvent event) { String text = event.getText(); return (text != null && !text.isEmpty() && text.equalsIgnoreCase(getCharacter()) && super.match(event)); } /** * Returns a string representation of this {@code MnemonicKeyCombination}. *

* The string representation consists of sections separated by plus * characters. Each section specifies either a modifier key or the main key. *

* A modifier key section contains the {@code KeyCode} name of a modifier * key. It can be prefixed with the {@code Ignored} keyword. A non-prefixed * modifier key implies its {@code PRESSED} value while the prefixed version * implies the {@code IGNORED} value. If some modifier key is not specified * in the string at all, it means it has the default {@code RELEASED} value. *

* The main key section contains the main key character enclosed in single * quotes and is the last section in the returned string. * * @return the string representation of this {@code MnemonicKeyCombination} */ @Override public String getName() { StringBuilder sb = new StringBuilder(); sb.append(super.getName()); if (sb.length() > 0) { sb.append("+"); } return sb.append('\'').append(character.replace("'", "\\'")) .append('\'').toString(); } /** * Tests whether this {@code MnemonicKeyCombination} equals to the * specified object. * * @param obj the object to compare to * @return {@code true} if the objects are equal, {@code false} otherwise */ @Override public boolean equals(final Object obj) { if (this == obj) { return true; } if (!(obj instanceof MnemonicKeyCombination)) { return false; } return (this.character.equals(((MnemonicKeyCombination)obj).getCharacter()) && super.equals(obj)); } /** * Returns a hash code value for this {@code MnemonicKeyCombination}. * * @return the hash code value */ @Override public int hashCode() { return 23 * super.hashCode() + character.hashCode(); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy