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

gwt.material.design.client.base.BaseCheckBox Maven / Gradle / Ivy

There is a newer version: 2.8.3
Show newest version
/*
 * #%L
 * GwtMaterial
 * %%
 * Copyright (C) 2015 - 2017 GwtMaterialDesign
 * %%
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 * 
 *      http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 * #L%
 */
/*
 * Copyright 2009 Google Inc.
 * 
 * Licensed under the Apache License, Version 2.0 (the "License"); you may not
 * use this file except in compliance with the License. You may obtain a copy of
 * the License at
 * 
 * http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
 * License for the specific language governing permissions and limitations under
 * the License.
 */
package gwt.material.design.client.base;

import com.google.gwt.dom.client.Document;
import com.google.gwt.dom.client.Element;
import com.google.gwt.dom.client.InputElement;
import com.google.gwt.dom.client.LabelElement;
import com.google.gwt.dom.client.Style.WhiteSpace;
import com.google.gwt.editor.client.IsEditor;
import com.google.gwt.editor.client.LeafValueEditor;
import com.google.gwt.editor.client.adapters.TakesValueEditor;
import com.google.gwt.event.logical.shared.ValueChangeEvent;
import com.google.gwt.event.logical.shared.ValueChangeHandler;
import com.google.gwt.event.shared.HandlerRegistration;
import com.google.gwt.i18n.client.HasDirection.Direction;
import com.google.gwt.i18n.shared.DirectionEstimator;
import com.google.gwt.i18n.shared.HasDirectionEstimator;
import com.google.gwt.safehtml.shared.SafeHtml;
import com.google.gwt.user.client.DOM;
import com.google.gwt.user.client.Event;
import com.google.gwt.user.client.ui.*;
import gwt.material.design.client.constants.CssName;

/**
 * A standard check box widget.
 * 

* This class also serves as a base class for * {@link RadioButton}. *

*

* *

*

*

*

Built-in Bidi Text Support

* This widget is capable of automatically adjusting its direction according to * its content. This feature is controlled by {@link #setDirectionEstimator} or * passing a DirectionEstimator parameter to the constructor, and is off by * default. *

*

*

CSS Style Rules

*
*
.gwt-CheckBox
*
the outer element
*
.gwt-CheckBox-disabled
*
applied when Checkbox is disabled
*
*

*

*

Example

* { @example com.google.gwt.examples.CheckBoxExample} *

*/ public class BaseCheckBox extends ButtonBase implements HasName, HasValue, HasWordWrap, HasDirectionalSafeHtml, HasDirectionEstimator, IsEditor> { public static final DirectionEstimator DEFAULT_DIRECTION_ESTIMATOR = DirectionalTextHelper.DEFAULT_DIRECTION_ESTIMATOR; final DirectionalTextHelper directionalTextHelper; InputElement inputElem; LabelElement labelElem; private LeafValueEditor editor; private boolean valueChangeHandlerInitialized; /** * Creates a check box with no label. */ public BaseCheckBox() { this(DOM.createSpan()); setStyleName(CssName.GWT_CHECKBOX); } /** * Creates a check box with the specified text label. * * @param label the check box's label */ public BaseCheckBox(SafeHtml label) { this(label.asString(), true); } /** * Creates a check box with the specified text label. * * @param label the check box's label * @param dir the text's direction. Note that {@code DEFAULT} means direction * should be inherited from the widget's parent element. */ public BaseCheckBox(SafeHtml label, Direction dir) { this(); setHTML(label, dir); } /** * Creates a check box with the specified text label. * * @param label the check box's label * @param directionEstimator A DirectionEstimator object used for automatic * direction adjustment. For convenience, * {@link #DEFAULT_DIRECTION_ESTIMATOR} can be used. */ public BaseCheckBox(SafeHtml label, DirectionEstimator directionEstimator) { this(); setDirectionEstimator(directionEstimator); setHTML(label.asString()); } /** * Creates a check box with the specified text label. * * @param label the check box's label */ public BaseCheckBox(String label) { this(); setText(label); } /** * Creates a check box with the specified text label. * * @param label the check box's label * @param dir the text's direction. Note that {@code DEFAULT} means direction * should be inherited from the widget's parent element. */ public BaseCheckBox(String label, Direction dir) { this(); setText(label, dir); } /** * Creates a label with the specified text and a default direction estimator. * * @param label the check box's label * @param directionEstimator A DirectionEstimator object used for automatic * direction adjustment. For convenience, * {@link #DEFAULT_DIRECTION_ESTIMATOR} can be used. */ public BaseCheckBox(String label, DirectionEstimator directionEstimator) { this(); setDirectionEstimator(directionEstimator); setText(label); } /** * Creates a check box with the specified text label. * * @param label the check box's label * @param asHTML true to treat the specified label as html */ public BaseCheckBox(String label, boolean asHTML) { this(); if (asHTML) { setHTML(label); } else { setText(label); } } protected BaseCheckBox(Element elem) { super(elem); inputElem = InputElement.as(DOM.createInputCheck()); labelElem = Document.get().createLabelElement(); getElement().appendChild(inputElem); getElement().appendChild(labelElem); String uid = DOM.createUniqueId(); inputElem.setPropertyString("id", uid); labelElem.setHtmlFor(uid); directionalTextHelper = new DirectionalTextHelper(labelElem, true); // Accessibility: setting tab index to be 0 by default, ensuring element // appears in tab sequence. FocusWidget's setElement method already // calls setTabIndex, which is overridden below. However, at the time // that this call is made, inputElem has not been created. So, we have // to call setTabIndex again, once inputElem has been created. setTabIndex(0); } @Override public HandlerRegistration addValueChangeHandler( final ValueChangeHandler handler) { // Is this the first value change handler? If so, time to add handlers if (!valueChangeHandlerInitialized) { ensureDomEventHandlers(); valueChangeHandlerInitialized = true; } return addHandler(handler, ValueChangeEvent.getType()); } @Override public LeafValueEditor asEditor() { if (editor == null) { editor = TakesValueEditor.of(this); } return editor; } @Override public DirectionEstimator getDirectionEstimator() { return directionalTextHelper.getDirectionEstimator(); } /** * Returns the value property of the input element that backs this widget. * This is the value that will be associated with the CheckBox name and * submitted to the server if a {@link FormPanel} that holds it is submitted * and the box is checked. *

* Don't confuse this with {@link #getValue}, which returns true or false if * the widget is checked. */ public String getFormValue() { return inputElem.getValue(); } @Override public String getHTML() { return directionalTextHelper.getTextOrHtml(true); } @Override public String getName() { return inputElem.getName(); } @Override public int getTabIndex() { return inputElem.getTabIndex(); } @Override public String getText() { return directionalTextHelper.getTextOrHtml(false); } @Override public Direction getTextDirection() { return directionalTextHelper.getTextDirection(); } /** * Determines whether this check box is currently checked. *

* Note that this does not return the value property of the checkbox * input element wrapped by this widget. For access to that property, see * {@link #getFormValue()} * * @return true if the check box is checked, false otherwise. * Will not return null */ @Override public Boolean getValue() { if (isAttached()) { return inputElem.isChecked(); } else { return inputElem.isDefaultChecked(); } } @Override public boolean getWordWrap() { return !WhiteSpace.NOWRAP.getCssName().equals(getElement().getStyle().getWhiteSpace()); } /** * Determines whether this check box is currently checked. * * @return true if the check box is checked * @deprecated Use {@link #getValue} instead */ @Deprecated public boolean isChecked() { // Funny comparison b/c getValue could in theory return null return getValue() == true; } @Override public boolean isEnabled() { return !inputElem.isDisabled(); } @Override public void setAccessKey(char key) { inputElem.setAccessKey("" + key); } /** * {@inheritDoc} *

* See note at {@link #setDirectionEstimator(DirectionEstimator)}. */ @Override public void setDirectionEstimator(boolean enabled) { directionalTextHelper.setDirectionEstimator(enabled); } /** * {@inheritDoc} *

* Note: DirectionEstimator should be set before the label has any content; * it's highly recommended to set it using a constructor. Reason: if the * label already has non-empty content, this will update its direction * according to the new estimator's result. This may cause flicker, and thus * should be avoided. */ @Override public void setDirectionEstimator(DirectionEstimator directionEstimator) { directionalTextHelper.setDirectionEstimator(directionEstimator); } @Override public void setEnabled(boolean enabled) { inputElem.setDisabled(!enabled); if (enabled) { removeStyleDependentName(CssName.DISABLED); } else { addStyleDependentName(CssName.DISABLED); } } @Override public void setFocus(boolean focused) { if (focused) { inputElem.focus(); } else { inputElem.blur(); } } /** * Set the value property on the input element that backs this widget. This is * the value that will be associated with the CheckBox's name and submitted to * the server if a {@link FormPanel} that holds it is submitted and the box is * checked. *

* Don't confuse this with {@link #setValue}, which actually checks and * unchecks the box. * * @param value */ public void setFormValue(String value) { inputElem.setAttribute("value", value); } @Override public void setHTML(SafeHtml html, Direction dir) { directionalTextHelper.setTextOrHtml(html.asString(), dir, true); } @Override public void setHTML(String html) { directionalTextHelper.setTextOrHtml(html, true); } @Override public void setName(String name) { inputElem.setName(name); } @Override public void setTabIndex(int index) { // Need to guard against call to setTabIndex before inputElem is // initialized. This happens because FocusWidget's (a superclass of // CheckBox) setElement method calls setTabIndex before inputElem is // initialized. See CheckBox's protected constructor for more information. if (inputElem != null) { inputElem.setTabIndex(index); } } @Override public void setText(String text) { directionalTextHelper.setTextOrHtml(text, false); } @Override public void setText(String text, Direction dir) { directionalTextHelper.setTextOrHtml(text, dir, false); } /** * Checks or unchecks the check box. *

* Note that this does not set the value property of the checkbox * input element wrapped by this widget. For access to that property, see * {@link #setFormValue(String)} * * @param value true to check, false to uncheck; null value implies false */ @Override public void setValue(Boolean value) { setValue(value, false); } /** * Checks or unchecks the check box, firing {@link ValueChangeEvent} if * appropriate. *

* Note that this does not set the value property of the checkbox * input element wrapped by this widget. For access to that property, see * {@link #setFormValue(String)} * * @param value true to check, false to uncheck; null value implies false * @param fireEvents If true, and value has changed, fire a * {@link ValueChangeEvent} */ @Override public void setValue(Boolean value, boolean fireEvents) { if (value == null) { value = Boolean.FALSE; } Boolean oldValue = getValue(); inputElem.setChecked(value); inputElem.setDefaultChecked(value); if (value.equals(oldValue)) { return; } if (fireEvents) { ValueChangeEvent.fire(this, value); } } @Override public void setWordWrap(boolean wrap) { getElement().getStyle().setWhiteSpace(wrap ? WhiteSpace.NORMAL : WhiteSpace.NOWRAP); } // Unlike other widgets the CheckBox sinks on its inputElement, not // its wrapper @Override public void sinkEvents(int eventBitsToAdd) { if (isOrWasAttached()) { Event.sinkEvents(inputElem, eventBitsToAdd | Event.getEventsSunk(inputElem)); } else { super.sinkEvents(eventBitsToAdd); } } protected void ensureDomEventHandlers() { addClickHandler(event -> { // Checkboxes always toggle their value, no need to compare // with old value. Radio buttons are not so lucky, see // overrides in RadioButton ValueChangeEvent.fire(BaseCheckBox.this, getValue()); }); } /** * Affected Elements: *

    *
  • -label = label next to checkbox.
  • *
* * @see UIObject#onEnsureDebugId(String) */ @Override protected void onEnsureDebugId(String baseID) { super.onEnsureDebugId(baseID); ensureDebugId(labelElem, baseID, "label"); ensureDebugId(inputElem, baseID, "input"); labelElem.setHtmlFor(inputElem.getId()); } /** * This method is called when a widget is attached to the browser's document. * onAttach needs special handling for the CheckBox case. Must still call * {@link Widget#onAttach()} to preserve the onAttach contract. */ @Override protected void onLoad() { DOM.setEventListener(inputElem, this); } /** * This method is called when a widget is detached from the browser's * document. Overridden because of IE bug that throws away checked state and * in order to clear the event listener off of the inputElem. */ @Override protected void onUnload() { // Clear out the inputElem's event listener (breaking the circular // reference between it and the widget). DOM.setEventListener(inputElem, null); setValue(getValue()); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy