/*
 * Copyright (c) 2005, 2014, 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 javax.swing.plaf.nimbus;

import java.util.HashMap;
import java.util.Map;
import javax.swing.JComponent;
import javax.swing.plaf.synth.SynthConstants;

Represents a built in, or custom, state in Nimbus.

Synth provides several built in states, which are:

  • Enabled
  • Mouse Over
  • Pressed
  • Disabled
  • Focused
  • Selected
  • Default

However, there are many more states that could be described in a LookAndFeel, and it would be nice to style components differently based on these different states. For example, a progress bar could be "indeterminate". It would be very convenient to allow this to be defined as a "state".

This class, State, is intended to be used for such situations. Simply implement the abstract #isInState method. It returns true if the given JComponent is "in this state", false otherwise. This method will be called many times in performance sensitive loops. It must execute very quickly.

For example, the following might be an implementation of a custom "Indeterminate" state for JProgressBars:


    public final class IndeterminateState extends State<JProgressBar> {
        public IndeterminateState() {
            super("Indeterminate");
        }
        @Override
        protected boolean isInState(JProgressBar c) {
            return c.isIndeterminate();
        }
    }
/** * <p>Represents a built in, or custom, state in Nimbus.</p> * * <p>Synth provides several built in states, which are: * <ul> * <li>Enabled</li> * <li>Mouse Over</li> * <li>Pressed</li> * <li>Disabled</li> * <li>Focused</li> * <li>Selected</li> * <li>Default</li> * </ul> * * <p>However, there are many more states that could be described in a LookAndFeel, and it * would be nice to style components differently based on these different states. * For example, a progress bar could be "indeterminate". It would be very convenient * to allow this to be defined as a "state".</p> * * <p>This class, State, is intended to be used for such situations. * Simply implement the abstract #isInState method. It returns true if the given * JComponent is "in this state", false otherwise. This method will be called * <em>many</em> times in <em>performance sensitive loops</em>. It must execute * very quickly.</p> * * <p>For example, the following might be an implementation of a custom * "Indeterminate" state for JProgressBars:</p> * * <pre><code> * public final class IndeterminateState extends State&lt;JProgressBar&gt; { * public IndeterminateState() { * super("Indeterminate"); * } * * &#64;Override * protected boolean isInState(JProgressBar c) { * return c.isIndeterminate(); * } * } * </code></pre> */
public abstract class State<T extends JComponent>{ static final Map<String, StandardState> standardStates = new HashMap<String, StandardState>(7); static final State<JComponent> Enabled = new StandardState(SynthConstants.ENABLED); static final State<JComponent> MouseOver = new StandardState(SynthConstants.MOUSE_OVER); static final State<JComponent> Pressed = new StandardState(SynthConstants.PRESSED); static final State<JComponent> Disabled = new StandardState(SynthConstants.DISABLED); static final State<JComponent> Focused = new StandardState(SynthConstants.FOCUSED); static final State<JComponent> Selected = new StandardState(SynthConstants.SELECTED); static final State<JComponent> Default = new StandardState(SynthConstants.DEFAULT); private String name;

Create a new custom State. Specify the name for the state. The name should be unique within the states set for any one particular component. The name of the state should coincide with the name used in UIDefaults.

For example, the following would be correct:


    defaults.put("Button.States", "Enabled, Foo, Disabled");
    defaults.put("Button.Foo", new FooState("Foo"));
Params:
  • name – a simple user friendly name for the state, such as "Indeterminate" or "EmbeddedPanel" or "Blurred". It is customary to use camel case, with the first letter capitalized.
/** * <p>Create a new custom State. Specify the name for the state. The name should * be unique within the states set for any one particular component. * The name of the state should coincide with the name used in UIDefaults.</p> * * <p>For example, the following would be correct:</p> * <pre><code> * defaults.put("Button.States", "Enabled, Foo, Disabled"); * defaults.put("Button.Foo", new FooState("Foo")); * </code></pre> * * @param name a simple user friendly name for the state, such as "Indeterminate" * or "EmbeddedPanel" or "Blurred". It is customary to use camel case, * with the first letter capitalized. */
protected State(String name) { this.name = name; } @Override public String toString() { return name; }

This is the main entry point, called by NimbusStyle.

There are both custom states and standard states. Standard states correlate to the states defined in SynthConstants. When a UI delegate constructs a SynthContext, it specifies the state that the component is in according to the states defined in SynthConstants. Our NimbusStyle will then take this state, and query each State instance in the style asking whether isInState(c, s).

Now, only the standard states care about the "s" param. So we have this odd arrangement:

  • NimbusStyle calls State.isInState(c, s)
  • State.isInState(c, s) simply delegates to State.isInState(c)
  • EXCEPT, StandardState overrides State.isInState(c, s) and returns directly from that method after checking its state, and does not call isInState(c) (since it is not needed for standard states).
/** * <p>This is the main entry point, called by NimbusStyle.</p> * * <p>There are both custom states and standard states. Standard states * correlate to the states defined in SynthConstants. When a UI delegate * constructs a SynthContext, it specifies the state that the component is * in according to the states defined in SynthConstants. Our NimbusStyle * will then take this state, and query each State instance in the style * asking whether isInState(c, s).</p> * * <p>Now, only the standard states care about the "s" param. So we have * this odd arrangement:</p> * <ul> * <li>NimbusStyle calls State.isInState(c, s)</li> * <li>State.isInState(c, s) simply delegates to State.isInState(c)</li> * <li><em>EXCEPT</em>, StandardState overrides State.isInState(c, s) and * returns directly from that method after checking its state, and * does not call isInState(c) (since it is not needed for standard states).</li> * </ul> */
boolean isInState(T c, int s) { return isInState(c); }

Gets whether the specified JComponent is in the custom state represented by this class. This is an extremely performance sensitive loop. Please take proper precautions to ensure that it executes quickly.

Nimbus uses this method to help determine what state a JComponent is in. For example, a custom State could exist for JProgressBar such that it would return true when the progress bar is indeterminate. Such an implementation of this method would simply be:

 return c.isIndeterminate();
Params:
  • c – the JComponent to test. This will never be null.
Returns:true if c is in the custom state represented by this State instance
/** * <p>Gets whether the specified JComponent is in the custom state represented * by this class. <em>This is an extremely performance sensitive loop.</em> * Please take proper precautions to ensure that it executes quickly.</p> * * <p>Nimbus uses this method to help determine what state a JComponent is * in. For example, a custom State could exist for JProgressBar such that * it would return <code>true</code> when the progress bar is indeterminate. * Such an implementation of this method would simply be:</p> * * <pre><code> return c.isIndeterminate();</code></pre> * * @param c the JComponent to test. This will never be null. * @return true if <code>c</code> is in the custom state represented by * this <code>State</code> instance */
protected abstract boolean isInState(T c); String getName() { return name; } static boolean isStandardStateName(String name) { return standardStates.containsKey(name); } static StandardState getStandardState(String name) { return standardStates.get(name); } static final class StandardState extends State<JComponent> { private int state; private StandardState(int state) { super(toString(state)); this.state = state; standardStates.put(getName(), this); } public int getState() { return state; } @Override boolean isInState(JComponent c, int s) { return (s & state) == state; } @Override protected boolean isInState(JComponent c) { throw new AssertionError("This method should never be called"); } private static String toString(int state) { StringBuilder sb = new StringBuilder(); if ((state & SynthConstants.DEFAULT) == SynthConstants.DEFAULT) { sb.append("Default"); } if ((state & SynthConstants.DISABLED) == SynthConstants.DISABLED) { if (sb.length() > 0) sb.append("+"); sb.append("Disabled"); } if ((state & SynthConstants.ENABLED) == SynthConstants.ENABLED) { if (sb.length() > 0) sb.append("+"); sb.append("Enabled"); } if ((state & SynthConstants.FOCUSED) == SynthConstants.FOCUSED) { if (sb.length() > 0) sb.append("+"); sb.append("Focused"); } if ((state & SynthConstants.MOUSE_OVER) == SynthConstants.MOUSE_OVER) { if (sb.length() > 0) sb.append("+"); sb.append("MouseOver"); } if ((state & SynthConstants.PRESSED) == SynthConstants.PRESSED) { if (sb.length() > 0) sb.append("+"); sb.append("Pressed"); } if ((state & SynthConstants.SELECTED) == SynthConstants.SELECTED) { if (sb.length() > 0) sb.append("+"); sb.append("Selected"); } return sb.toString(); } } }