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

com.jfoenix.skins.JFXTextFieldSkin Maven / Gradle / Ivy

There is a newer version: 9.0.10
Show newest version
/*
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you 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.jfoenix.skins;

import com.jfoenix.concurrency.JFXUtilities;
import com.jfoenix.controls.IFXTextInputControl;
import com.jfoenix.transitions.CachedTransition;
import com.jfoenix.validation.base.ValidatorBase;
import com.sun.javafx.scene.control.skin.TextFieldSkin;
import javafx.animation.Animation.Status;
import javafx.animation.*;
import javafx.application.Platform;
import javafx.beans.binding.Bindings;
import javafx.beans.binding.BooleanBinding;
import javafx.beans.property.ObjectProperty;
import javafx.beans.property.SimpleObjectProperty;
import javafx.beans.value.ObservableValue;
import javafx.geometry.Insets;
import javafx.geometry.Pos;
import javafx.scene.Node;
import javafx.scene.control.Label;
import javafx.scene.control.TextField;
import javafx.scene.layout.*;
import javafx.scene.paint.Color;
import javafx.scene.paint.Paint;
import javafx.scene.shape.Rectangle;
import javafx.scene.text.Text;
import javafx.scene.transform.Scale;
import javafx.util.Duration;

import java.lang.reflect.Field;

/**
 * 

Material Design Text input control Skin, used for both JFXTextField/JFXPasswordField

* * @author Shadi Shaheen * @version 2.0 * @since 2017-01-25 */ public class JFXTextFieldSkin extends TextFieldSkin { private boolean invalid = true; private StackPane line = new StackPane(); private StackPane focusedLine = new StackPane(); private Label errorLabel = new Label(); private StackPane errorIcon = new StackPane(); private HBox errorContainer = new HBox(); private StackPane promptContainer = new StackPane(); private Text promptText; private Pane textPane; private ParallelTransition transition; private CachedTransition promptTextUpTransition; private CachedTransition promptTextDownTransition; private CachedTransition promptTextColorTransition; private double initScale = 0.05; private final Scale promptTextScale = new Scale(1, 1, 0, 0); private final Scale scale = new Scale(initScale, 1); private Timeline linesAnimation = new Timeline( new KeyFrame(Duration.ZERO, new KeyValue(scale.xProperty(), initScale, Interpolator.EASE_BOTH), new KeyValue(focusedLine.opacityProperty(), 0, Interpolator.EASE_BOTH)), new KeyFrame(Duration.millis(1), new KeyValue(focusedLine.opacityProperty(), 1, Interpolator.EASE_BOTH)), new KeyFrame(Duration.millis(160), new KeyValue(scale.xProperty(), 1, Interpolator.EASE_BOTH)) ); private Paint oldPromptTextFill; private BooleanBinding usePromptText = Bindings.createBooleanBinding(this::usePromptText, getSkinnable().textProperty(), getSkinnable().promptTextProperty()); private final Rectangle errorContainerClip = new Rectangle(); private final Scale errorClipScale = new Scale(1, 0, 0, 0); private Timeline errorHideTransition = new Timeline(new KeyFrame(Duration.millis(80), new KeyValue(errorContainer.opacityProperty(), 0, Interpolator.LINEAR))); private Timeline errorShowTransition = new Timeline(new KeyFrame(Duration.millis(80), new KeyValue(errorContainer.opacityProperty(), 1, Interpolator.EASE_OUT))); private Timeline scale1 = new Timeline(); private Timeline scaleLess1 = new Timeline(); private final ObjectProperty animatedPromptTextFill = new SimpleObjectProperty<>(super.promptTextFill.get()); public JFXTextFieldSkin(T field) { super(field); // initial styles field.setBackground(new Background(new BackgroundFill(Color.TRANSPARENT, CornerRadii.EMPTY, Insets.EMPTY))); field.setPadding(new Insets(4, 0, 4, 0)); // add style classes errorLabel.getStyleClass().add("error-label"); line.getStyleClass().add("input-line"); focusedLine.getStyleClass().add("input-focused-line"); // draw lines line.setPrefHeight(1); line.setTranslateY(1); // translate = prefHeight + init_translation line.setBackground(new Background(new BackgroundFill(((IFXTextInputControl) getSkinnable()).getUnFocusColor(), CornerRadii.EMPTY, Insets.EMPTY))); if (getSkinnable().isDisabled()) { line.setBorder(new Border(new BorderStroke(((IFXTextInputControl) getSkinnable()).getUnFocusColor(), BorderStrokeStyle.DASHED, CornerRadii.EMPTY, new BorderWidths(1)))); line.setBackground(new Background(new BackgroundFill(Color.TRANSPARENT, CornerRadii.EMPTY, Insets.EMPTY))); } // focused line focusedLine.setPrefHeight(2); focusedLine.setTranslateY(0); // translate = prefHeight + init_translation(-1) focusedLine.setBackground(new Background(new BackgroundFill(((IFXTextInputControl) getSkinnable()).getFocusColor(), CornerRadii.EMPTY, Insets.EMPTY))); focusedLine.setOpacity(0); focusedLine.getTransforms().add(scale); // error container errorContainer.getChildren().setAll(new StackPane(errorLabel), errorIcon); errorContainer.setAlignment(Pos.CENTER_LEFT); errorContainer.setSpacing(8); errorContainer.setPadding(new Insets(4, 0, 0, 0)); errorContainer.setVisible(false); errorContainer.setOpacity(0); errorContainer.setManaged(false); StackPane.setAlignment(errorLabel, Pos.TOP_LEFT); HBox.setHgrow(errorLabel.getParent(), Priority.ALWAYS); errorContainerClip.getTransforms().add(errorClipScale); errorContainer.setClip(field.isDisableAnimation() ? null : errorContainerClip); getChildren().addAll(line, focusedLine, promptContainer, errorContainer); field.activeValidatorProperty().addListener((ObservableValue o, ValidatorBase oldVal, ValidatorBase newVal) -> { if (textPane != null) { if (!((IFXTextInputControl) getSkinnable()).isDisableAnimation()) { if (newVal != null) { errorHideTransition.setOnFinished(finish -> { showError(newVal); final double w = getSkinnable().getWidth(); double errorContainerHeight = computeErrorHeight(computeErrorWidth(w)); if(errorLabel.isWrapText()){ // animate opacity + scale if (errorContainerHeight < errorContainer.getHeight()) { // update animation frames scaleLess1.getKeyFrames().setAll(createSmallerScaleFrame(errorContainerHeight)); scaleLess1.setOnFinished(event -> { updateErrorContainerSize(w, errorContainerHeight); errorClipScale.setY(1); }); SequentialTransition transition = new SequentialTransition(scaleLess1, errorShowTransition); transition.play(); } else { errorClipScale.setY(oldVal == null ? 0 : errorContainer.getHeight() / errorContainerHeight); updateErrorContainerSize(w, errorContainerHeight); // update animation frames scale1.getKeyFrames().setAll(createScaleToOneFrames()); // play animation ParallelTransition parallelTransition = new ParallelTransition(); parallelTransition.getChildren().addAll(scale1, errorShowTransition); parallelTransition.play(); } }else{ // animate opacity only errorClipScale.setY(1); updateErrorContainerSize(w, errorContainerHeight); ParallelTransition parallelTransition = new ParallelTransition(errorShowTransition); parallelTransition.play(); } }); errorHideTransition.play(); } else { errorHideTransition.setOnFinished(null); if(errorLabel.isWrapText()){ // animate scale only scaleLess1.getKeyFrames().setAll(new KeyFrame(Duration.millis(100), new KeyValue(errorClipScale.yProperty(), 0, Interpolator.EASE_BOTH))); scaleLess1.setOnFinished(event -> { hideError(); errorClipScale.setY(0); }); SequentialTransition transition = new SequentialTransition(scaleLess1); transition.play(); }else{ errorClipScale.setY(0); } // animate opacity only errorHideTransition.play(); } } else { if (newVal != null) { JFXUtilities.runInFXAndWait(() -> showError(newVal)); } else { JFXUtilities.runInFXAndWait(this::hideError); } } } }); field.focusColorProperty().addListener((o, oldVal, newVal) -> { if (newVal != null) { focusedLine.setBackground(new Background(new BackgroundFill(newVal, CornerRadii.EMPTY, Insets.EMPTY))); if (((IFXTextInputControl) getSkinnable()).isLabelFloat()) { promptTextColorTransition = createColorTransition(newVal); // reset transition resetFocusTransition(); } } }); field.unFocusColorProperty().addListener((o, oldVal, newVal) -> { if (newVal != null) { line.setBackground(new Background(new BackgroundFill(newVal, CornerRadii.EMPTY, Insets.EMPTY))); } }); // handle animation on focus gained/lost event field.focusedProperty().addListener((o, oldVal, newVal) -> { if (newVal) { focus(); } else { unFocus(); } }); // handle text changing at runtime field.textProperty().addListener((o, oldVal, newVal) -> { if (!getSkinnable().isFocused() && ((IFXTextInputControl) getSkinnable()).isLabelFloat()) { if (newVal == null || newVal.isEmpty()) { animateFloatingLabel(false); } else { animateFloatingLabel(true); } } }); field.disabledProperty().addListener((o, oldVal, newVal) -> { line.setBorder(newVal ? new Border(new BorderStroke(((IFXTextInputControl) getSkinnable()).getUnFocusColor(), BorderStrokeStyle.DASHED, CornerRadii.EMPTY, new BorderWidths(line.getHeight()))) : Border.EMPTY); line.setBackground(new Background(new BackgroundFill(newVal ? Color.TRANSPARENT : ((IFXTextInputControl) getSkinnable()) .getUnFocusColor(), CornerRadii.EMPTY, Insets.EMPTY))); }); promptTextFill.addListener((o, oldVal, newVal) -> { oldPromptTextFill = newVal; animatedPromptTextFill.set(newVal); }); registerChangeListener(field.disableAnimationProperty(), "DISABLE_ANIMATION"); registerChangeListener(field.labelFloatProperty(), "LABEL_FLOAT"); } @Override protected void handleControlPropertyChanged(String propertyReference) { if("LABEL_FLOAT".equals(propertyReference)){ boolean isLabelFloat = ((IFXTextInputControl) getSkinnable()).isLabelFloat(); if (isLabelFloat) { JFXUtilities.runInFX(this::createFloatingLabel); } else { if(promptText!=null){ // remove parent focus listener to prevent editor class cast exception promptText.visibleProperty().bind(usePromptText); } } createFocusTransition(); // update prompt text position if(isLabelFloat) animateFloatingLabel(!getSkinnable().getText().isEmpty()); }else if ("DISABLE_ANIMATION".equals(propertyReference)) { // remove error clip if animation is disabled errorContainer.setClip(((IFXTextInputControl) getSkinnable()).isDisableAnimation() ? null : errorContainerClip); } else { super.handleControlPropertyChanged(propertyReference); } } @Override protected void layoutChildren(final double x, final double y, final double w, final double h) { super.layoutChildren(x, y, w, h); // change control properties if and only if animations are stopped if (transition == null || transition.getStatus() == Status.STOPPED) { if (getSkinnable().isFocused() && ((IFXTextInputControl) getSkinnable()).isLabelFloat()) { animatedPromptTextFill.set(((IFXTextInputControl) getSkinnable()).getFocusColor()); } } if (invalid) { invalid = false; animatedPromptTextFill.set(promptTextFill.get()); textPane = (Pane) this.getChildren().get(0); // create floating label createFloatingLabel(); // update validation container final ValidatorBase activeValidator = ((IFXTextInputControl) getSkinnable()).getActiveValidator(); if (activeValidator != null) { showError(activeValidator); final double errorContainerWidth = w - errorIcon.prefWidth(-1); errorContainer.setOpacity(1); errorContainer.resize(w, computeErrorHeight(errorContainerWidth)); errorContainerClip.setWidth(w); errorContainerClip.setHeight(errorContainer.getHeight()); errorClipScale.setY(1); } // to position the prompt node properly super.layoutChildren(x, y, w, h); // focus createFocusTransition(); if (getSkinnable().isFocused()) { focus(); } } final double height = getSkinnable().getHeight(); final double focusedLineHeight = focusedLine.prefHeight(-1); focusedLine.resizeRelocate(x, height, w, focusedLineHeight); line.resizeRelocate(x, height, w, line.prefHeight(-1)); errorContainer.relocate(x, height + focusedLineHeight); // resize error container if animation is disabled if (((IFXTextInputControl) getSkinnable()).isDisableAnimation()) { errorContainer.resize(w, computeErrorHeight(computeErrorWidth(w))); } scale.setPivotX(w / 2); } private double computeErrorWidth(double w) { return w - errorIcon.prefWidth(-1); } private double computeErrorHeight(double errorContainerWidth) { return errorLabel.prefHeight(errorContainerWidth) + errorContainer.snappedBottomInset() + errorContainer.snappedTopInset(); } /** * update the size of error container and its clip * @param w * @param errorContainerHeight */ private void updateErrorContainerSize(double w, double errorContainerHeight) { errorContainerClip.setWidth(w); errorContainerClip.setHeight(errorContainerHeight); errorContainer.resize(w, errorContainerHeight); } /** * creates error animation frames when moving from large -> small error container * @param errorContainerHeight * @return */ private KeyFrame createSmallerScaleFrame(double errorContainerHeight) { return new KeyFrame(Duration.millis(100), new KeyValue(errorClipScale.yProperty(), errorContainerHeight / errorContainer.getHeight(), Interpolator.EASE_BOTH)); } /** * creates error animation frames when moving from small -> large error container * @return */ private KeyFrame createScaleToOneFrames() { return new KeyFrame(Duration.millis(100), new KeyValue(errorClipScale.yProperty(), 1, Interpolator.EASE_BOTH)); } private void createFloatingLabel() { if (((IFXTextInputControl) getSkinnable()).isLabelFloat()) { if (promptText == null) { // get the prompt text node or create it boolean triggerFloatLabel = false; if (textPane.getChildren().get(0) instanceof Text) { promptText = (Text) textPane.getChildren().get(0); } else { Field field; try { field = TextFieldSkin.class.getDeclaredField("promptNode"); field.setAccessible(true); createPromptNode(); field.set(this, promptText); // replace parent promptNode with promptText field triggerFloatLabel = true; } catch (NoSuchFieldException | SecurityException | IllegalAccessException | IllegalArgumentException ex) { ex.printStackTrace(); } } promptText.fillProperty().bind(animatedPromptTextFill); promptText.getTransforms().add(promptTextScale); promptContainer.getChildren().add(promptText); if (triggerFloatLabel) { promptText.setTranslateY(-textPane.getHeight()); promptTextScale.setX(0.85); promptTextScale.setY(0.85); } } promptTextUpTransition = new CachedTransition(textPane, new Timeline( new KeyFrame(Duration.millis(1300), new KeyValue(promptText.translateYProperty(), -textPane.getHeight(), Interpolator.EASE_BOTH), new KeyValue(promptTextScale.xProperty(), 0.85, Interpolator.EASE_BOTH), new KeyValue(promptTextScale.yProperty(), 0.85, Interpolator.EASE_BOTH)))) { private double finalHeight = textPane.getHeight(); { setDelay(Duration.millis(0)); setCycleDuration(Duration.millis(240)); } @Override protected void stopping() { super.stopping(); promptText.setTranslateY(-finalHeight); promptTextScale.setX(0.85); promptTextScale.setY(0.85); } }; promptTextColorTransition = createColorTransition(((IFXTextInputControl) getSkinnable()).getFocusColor()); promptTextDownTransition = new CachedTransition(textPane, new Timeline( new KeyFrame(Duration.millis(1300), new KeyValue(promptText.translateYProperty(), 0, Interpolator.EASE_BOTH), new KeyValue(promptTextScale.xProperty(), 1, Interpolator.EASE_BOTH), new KeyValue(promptTextScale.yProperty(), 1, Interpolator.EASE_BOTH)))) { { setDelay(Duration.millis(0)); setCycleDuration(Duration.millis(240)); } @Override protected void stopping() { super.stopping(); promptText.setTranslateY(0); promptTextScale.setX(1); promptTextScale.setY(1); } }; promptText.visibleProperty().unbind(); promptText.visibleProperty().set(true); } } private void createPromptNode() { promptText = new Text(); promptText.setManaged(false); promptText.getStyleClass().add("text"); promptText.visibleProperty().bind(usePromptText); promptText.fontProperty().bind(getSkinnable().fontProperty()); promptText.textProperty().bind(getSkinnable().promptTextProperty()); promptText.fillProperty().bind(animatedPromptTextFill); promptText.setLayoutX(1); } private void focus() { // in case the method request layout is not called before focused // this bug is reported while editing TreeTableView cells if (textPane == null) { Platform.runLater(this::focus); } else { // create the focus animations if (transition == null) { createFocusTransition(); } transition.play(); } } private void unFocus() { if (transition != null) { transition.stop(); } scale.setX(initScale); focusedLine.setOpacity(0); if (((IFXTextInputControl) getSkinnable()).isLabelFloat() && oldPromptTextFill != null) { animatedPromptTextFill.set(oldPromptTextFill); if (usePromptText()) { promptTextDownTransition.play(); } } } private CachedTransition createColorTransition(Paint newVal) { return new CachedTransition(textPane, new Timeline( new KeyFrame(Duration.millis(1300), new KeyValue(animatedPromptTextFill, newVal, Interpolator.EASE_BOTH)))) { { setDelay(Duration.millis(0)); setCycleDuration(Duration.millis(160)); } protected void starting() { super.starting(); oldPromptTextFill = animatedPromptTextFill.get(); } }; } private void createFocusTransition() { transition = new ParallelTransition(); if (((IFXTextInputControl) getSkinnable()).isLabelFloat()) { transition.getChildren().add(promptTextUpTransition); transition.getChildren().add(promptTextColorTransition); } transition.getChildren().add(linesAnimation); } /** * this method is called when the text property is changed when the * field is not focused (changed in code) * * @param up */ private void animateFloatingLabel(boolean up) { if (promptText == null) { Platform.runLater(() -> animateFloatingLabel(up)); } else { resetFocusTransition(); if (up && promptText.getTranslateY() == 0) { promptTextDownTransition.stop(); promptTextUpTransition.play(); } else if (!up) { promptTextUpTransition.stop(); promptTextDownTransition.play(); } } } private void resetFocusTransition() { if (transition != null) { transition.stop(); transition.getChildren().remove(promptTextUpTransition); transition = null; } } private boolean usePromptText() { String txt = getSkinnable().getText(); String promptTxt = getSkinnable().getPromptText(); boolean isLabelFloat = ((IFXTextInputControl) getSkinnable()).isLabelFloat(); return (txt == null || txt.isEmpty()) && promptTxt != null && !promptTxt.isEmpty() && (!promptTextFill.get().equals(Color.TRANSPARENT) || isLabelFloat); } private void showError(ValidatorBase validator) { // set text in error label errorLabel.setText(validator.getMessage()); // show error icon Node icon = validator.getIcon(); errorIcon.getChildren().clear(); if (icon != null) { errorIcon.getChildren().setAll(icon); StackPane.setAlignment(icon, Pos.CENTER_RIGHT); } errorContainer.setVisible(true); } private void hideError() { // clear error label text errorLabel.setText(null); // clear error icon errorIcon.getChildren().clear(); // reset the height of the text field // hide error container errorContainer.setVisible(false); } }




© 2015 - 2024 Weber Informatics LLC | Privacy Policy