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

com.holonplatform.vaadin.internal.components.NumberField Maven / Gradle / Ivy

/*
 * Copyright 2016-2017 Axioma srl.
 * 
 * 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 com.holonplatform.vaadin.internal.components;

import java.text.DecimalFormatSymbols;
import java.text.NumberFormat;
import java.util.Locale;

import com.holonplatform.core.i18n.Localizable;
import com.holonplatform.core.i18n.LocalizationContext;
import com.holonplatform.core.internal.utils.TypeUtils;
import com.holonplatform.vaadin.components.Field;
import com.holonplatform.vaadin.components.Input;
import com.holonplatform.vaadin.components.builders.NumberInputBuilder;
import com.holonplatform.vaadin.internal.components.builders.AbstractFieldBuilder;
import com.holonplatform.vaadin.internal.converters.StringToNumberConverter;
import com.vaadin.data.ValueContext;
import com.vaadin.event.FieldEvents.BlurListener;
import com.vaadin.event.FieldEvents.FocusListener;
import com.vaadin.shared.Registration;
import com.vaadin.shared.ui.ValueChangeMode;
import com.vaadin.ui.Component;
import com.vaadin.ui.TextField;

/**
 * A field for {@link Number} value type input.
 * 
 * 

* A standard {@link TextField} is used as UI widget for user input, so no client-side character validation is * performed. *

* *

* This field provides common {@link TextField} properties getters and setters, for example {@link #setMaxLength(int)} * or {@link #setInputPrompt(String)}. *

* *

* A {@link StringToNumberConverter} is used for String to numeric value conversions and backward. *

* * @param Number type * * @since 5.0.0 */ public class NumberField extends AbstractCustomField { private static final long serialVersionUID = -6791184576381210657L; private StringToNumberConverter converter; /** * Default constructor * @param Number type * @param numberClass Number field type (not null) */ public NumberField(Class numberClass) { super(TypeUtils.box(numberClass)); addStyleName("h-numberfield", false); } /** * Constructor with caption * @param Number type * @param numberClass Number field type (not null) * @param caption The field caption */ public NumberField(Class numberClass, String caption) { this(numberClass); setCaption(caption); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.internal.components.AbstractCustomField#buildInternalField(java.lang.Class) */ @Override protected NumericTextField buildInternalField(Class type) { return new NumericTextField(); } /** * Sets a fixed NumberFormat to use for value conversions from user input field and back. * @param numberFormat the NumberFormat to set */ public void setNumberFormat(NumberFormat numberFormat) { getConverter().setNumberFormat(numberFormat); } /** * Get the string to number converter to use. * @return the converter */ protected StringToNumberConverter getConverter() { if (converter == null) { converter = new StringToNumberConverter<>(getType()); } return converter; } /* * (non-Javadoc) * @see com.holonplatform.vaadin.internal.components.AbstractCustomField#fromInternalValue(java.lang.Object) */ @Override protected T fromInternalValue(String value) { return getConverter().convertToModel(value, new ValueContext(findLocale())) .getOrThrow(msg -> new IllegalArgumentException(msg)); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.internal.components.AbstractCustomField#toInternalValue(java.lang.Object) */ @Override protected String toInternalValue(T value) { return getConverter().convertToPresentation(value, new ValueContext(findLocale())); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.internal.components.AbstractCustomField#initContent() */ @Override protected Component initContent() { setupLocaleSymbolsInputValidation(); return super.initContent(); } /** * Setup user input validation allowed symbols according to current Locale and number type */ protected void setupLocaleSymbolsInputValidation() { Locale locale = LocalizationContext.getCurrent().filter(l -> l.isLocalized()).flatMap(l -> l.getLocale()) .orElse(getLocale()); if (locale == null) { // use default locale = Locale.getDefault(); } // check grouping boolean useGrouping = true; NumberFormat nf = getConverter().getNumberFormat(locale); if (nf != null) { useGrouping = nf.isGroupingUsed(); } DecimalFormatSymbols dfs = new DecimalFormatSymbols(locale); char[] symbols = null; if (useGrouping) { if (TypeUtils.isDecimalNumber(getType())) { symbols = new char[] { dfs.getGroupingSeparator(), dfs.getDecimalSeparator() }; } else { symbols = new char[] { dfs.getGroupingSeparator() }; } } else { if (TypeUtils.isDecimalNumber(getType())) { symbols = new char[] { dfs.getDecimalSeparator() }; } } getInternalField().setAllowedSymbols(symbols); } /** * Gets whether to allow negative numbers input * @return true to allow negative numbers input */ public boolean isAllowNegative() { return getInternalField().isAllowNegative(); } /** * Sets whether to allow negative numbers input * @param allowNegative true to allow negative numbers input */ public void setAllowNegative(boolean allowNegative) { getInternalField().setAllowNegative(allowNegative); } /** * Gets whether to set html5 input type property as "number" * @return true to set html5 input type property as "number" */ public boolean isHtml5NumberInputType() { return getInternalField().isHtml5NumberInputType(); } /** * Sets whether to set html5 input type property as "number" * @param html5NumberInputType true to set html5 input type property as "number" */ public void setHtml5NumberInputType(boolean html5NumberInputType) { getInternalField().setHtml5NumberInputType(html5NumberInputType); } /** * Returns the maximum number of characters in the field. Value -1 is considered unlimited. * @return the maxLength Maximum number of characters in the field */ public int getMaxLength() { return getInternalField().getMaxLength(); } /** * Set the maximum number of characters in the field * @param maxLength Maximum number of characters in the field, -1 is considered unlimited */ public void setMaxLength(int maxLength) { getInternalField().setMaxLength(maxLength); } /** * Gets the current input prompt * @return the current input prompt, or null if not enabled */ public String getInputPrompt() { return getInternalField().getPlaceholder(); } /** * Sets the input prompt - a textual prompt that is displayed when the field would otherwise be empty, to prompt the * user for input. * @param inputPrompt the input prompt to set, null for none */ public void setInputPrompt(String inputPrompt) { getInternalField().setPlaceholder(inputPrompt); } /** * Sets the mode how the TextField triggers value change events. * @param inputEventMode the new mode */ public void setTextChangeEventMode(ValueChangeMode inputEventMode) { getInternalField().setValueChangeMode(inputEventMode); } /** * Gets the current {@link ValueChangeMode} * @return the mode used to trigger value change events. */ public ValueChangeMode getTextChangeEventMode() { return getInternalField().getValueChangeMode(); } /** * The text change timeout modifies how often text change events are communicated to the application. * @param timeout the timeout in milliseconds */ public void setTextChangeTimeout(int timeout) { getInternalField().setValueChangeTimeout(timeout); } /** * Gets the timeout used to fire text change events. * @return the timeout value in milliseconds */ public int getTextChangeTimeout() { return getInternalField().getValueChangeTimeout(); } /** * Sets the cursor position in the field. As a side effect the field will become focused. * @param pos the position for the cursor */ public void setCursorPosition(int pos) { getInternalField().setCursorPosition(pos); } /** * Returns the last known cursor position of the field. * @return the cursor position */ public int getCursorPosition() { return getInternalField().getCursorPosition(); } /** * Selects all text in the field. */ public void selectAll() { getInternalField().selectAll(); } /** * Adds a listener for focus gained event * @param listener Listener to add * @return the listener {@link Registration} */ public Registration addFocusListener(FocusListener listener) { return getInternalField().addFocusListener(listener); } /** * Adds a listener for focus lost event * @param listener Listener to add * @return the listener {@link Registration} */ public Registration addBlurListener(BlurListener listener) { return getInternalField().addBlurListener(listener); } // Builder public static class Builder extends AbstractFieldBuilder, NumberField, NumberInputBuilder> implements NumberInputBuilder { protected Localizable inputPrompt; public Builder(Class numberClass) { super(new NumberField<>(numberClass)); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.components.builders.TextInputFieldBuilder#maxLength(int) */ @Override public NumberInputBuilder maxLength(int maxLength) { getInstance().setMaxLength(maxLength); return builder(); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.components.builders.TextInputFieldBuilder#inputPrompt(java.lang.String) */ @Override public NumberInputBuilder inputPrompt(String inputPrompt) { getInstance().setInputPrompt(inputPrompt); return builder(); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.components.builders.TextInputFieldBuilder#inputPrompt(java.lang.String, * java.lang.String, java.lang.Object[]) */ @Override public NumberInputBuilder inputPrompt(String defaultInputPrompt, String messageCode, Object... arguments) { this.inputPrompt = Localizable.builder().message(defaultInputPrompt).messageCode(messageCode) .messageArguments(arguments).build(); return builder(); } /* * (non-Javadoc) * @see * com.holonplatform.vaadin.components.builders.TextInputFieldBuilder#inputPrompt(com.holonplatform.core.i18n. * Localizable) */ @Override public NumberInputBuilder inputPrompt(Localizable inputPrompt) { this.inputPrompt = inputPrompt; return builder(); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.components.builders.TextInputFieldBuilder#textChangeEventMode(com.vaadin.ui. * AbstractTextField.TextChangeEventMode) */ @Override public NumberInputBuilder textChangeEventMode(ValueChangeMode inputEventMode) { getInstance().setTextChangeEventMode(inputEventMode); return builder(); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.components.builders.TextInputFieldBuilder#textChangeTimeout(int) */ @Override public NumberInputBuilder textChangeTimeout(int timeout) { getInstance().setTextChangeTimeout(timeout); return builder(); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.components.builders.NumberFieldBuilder#numberFormat(java.text.NumberFormat) */ @Override public NumberInputBuilder numberFormat(NumberFormat numberFormat) { getInstance().setNumberFormat(numberFormat); return builder(); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.components.builders.NumberFieldBuilder#allowNegative(boolean) */ @Override public NumberInputBuilder allowNegative(boolean allowNegative) { getInstance().setAllowNegative(allowNegative); return builder(); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.components.builders.NumberFieldBuilder#html5NumberInputType(boolean) */ @Override public NumberInputBuilder html5NumberInputType(boolean html5NumberInputType) { getInstance().setHtml5NumberInputType(html5NumberInputType); return builder(); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.components.builders.TextInputFieldBuilder#withFocusListener(com.vaadin.event. * FieldEvents.FocusListener) */ @Override public NumberInputBuilder withFocusListener(FocusListener listener) { getInstance().addFocusListener(listener); return builder(); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.components.builders.TextInputFieldBuilder#withBlurListener(com.vaadin.event. * FieldEvents.BlurListener) */ @Override public NumberInputBuilder withBlurListener(BlurListener listener) { getInstance().addBlurListener(listener); return builder(); } /* * (non-Javadoc) * @see com.holonplatform.vaadin.internal.components.builders.AbstractComponentBuilder#builder() */ @Override protected NumberInputBuilder builder() { return this; } /* * (non-Javadoc) * @see com.holonplatform.vaadin.internal.components.builders.AbstractFieldBuilder#localize(com.vaadin.ui. * AbstractField) */ @Override protected void localize(NumberField instance) { super.localize(instance); if (inputPrompt != null) { instance.setInputPrompt(LocalizationContext.translate(inputPrompt, true)); } } /* * (non-Javadoc) * @see * com.holonplatform.vaadin.internal.components.builders.AbstractFieldBuilder#build(com.vaadin.ui.AbstractField) */ @Override protected Input build(NumberField instance) { return instance; } /* * (non-Javadoc) * @see com.holonplatform.vaadin.internal.components.builders.AbstractFieldBuilder#buildAsField(com.vaadin.ui. * AbstractField) */ @Override protected Field buildAsField(NumberField instance) { return instance; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy