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

com.jfoenix.skins.JFXComboBoxListViewSkin 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.controls.JFXComboBox;
import com.jfoenix.transitions.JFXAnimationTimer;
import com.jfoenix.transitions.JFXKeyFrame;
import com.jfoenix.transitions.JFXKeyValue;
import com.sun.javafx.css.converters.PaintConverter;
import com.sun.javafx.scene.control.skin.ComboBoxListViewSkin;
import javafx.animation.Interpolator;
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.css.CssMetaData;
import javafx.css.Styleable;
import javafx.css.StyleableObjectProperty;
import javafx.css.StyleableProperty;
import javafx.geometry.Insets;
import javafx.geometry.Pos;
import javafx.scene.control.ListCell;
import javafx.scene.layout.*;
import javafx.scene.paint.Color;
import javafx.scene.paint.Paint;
import javafx.scene.text.Text;
import javafx.scene.transform.Scale;
import javafx.util.Duration;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;

/**
 * 

Material Design ComboBox Skin

* * @author Shadi Shaheen * @version 2.0 * @since 2017-01-25 */ public class JFXComboBoxListViewSkin extends ComboBoxListViewSkin { /*************************************************************************** * * * Private fields * * * **************************************************************************/ private boolean invalid = true; private StackPane customPane; private StackPane line = new StackPane(); private StackPane focusedLine = new StackPane(); private Text promptText = new Text(); private double initScale = 0.05; private Scale scale = new Scale(initScale, 1); private Scale promptTextScale = new Scale(1, 1, 0, 0); protected final ObjectProperty promptTextFill = new StyleableObjectProperty(Color.GRAY) { @Override public Object getBean() { return JFXComboBoxListViewSkin.this; } @Override public String getName() { return "promptTextFill"; } @Override public CssMetaData getCssMetaData() { return StyleableProperties.PROMPT_TEXT_FILL; } }; private Paint oldPromptTextFill = promptTextFill.get(); protected final ObjectProperty animatedPromptTextFill = new SimpleObjectProperty<>(promptTextFill.get()); JFXAnimationTimer focusTimer = new JFXAnimationTimer( new JFXKeyFrame(Duration.millis(1), JFXKeyValue.builder() .setTarget(focusedLine.opacityProperty()) .setEndValue(1) .setInterpolator(Interpolator.EASE_BOTH) .setAnimateCondition(()->getSkinnable().isFocused()).build()), new JFXKeyFrame(Duration.millis(160), JFXKeyValue.builder() .setTarget(scale.xProperty()) .setEndValue(1) .setInterpolator(Interpolator.EASE_BOTH).build(), JFXKeyValue.builder() .setTargetSupplier(()-> !((JFXComboBox) getSkinnable()).isLabelFloat() ? null : animatedPromptTextFill) .setEndValueSupplier(()->((JFXComboBox) getSkinnable()).getFocusColor()) .setInterpolator(Interpolator.EASE_BOTH) .setAnimateCondition(()->getSkinnable().isFocused()).build(), JFXKeyValue.builder() .setTargetSupplier(()-> !((JFXComboBox) getSkinnable()).isLabelFloat() ? null : promptText.translateYProperty()) .setEndValueSupplier(() -> -customPane.getHeight() + 6.05) .setInterpolator(Interpolator.EASE_BOTH).build(), JFXKeyValue.builder() .setTargetSupplier(()-> !((JFXComboBox) getSkinnable()).isLabelFloat() ? null : promptTextScale.xProperty()) .setEndValue(0.85) .setInterpolator(Interpolator.EASE_BOTH).build(), JFXKeyValue.builder() .setTargetSupplier(()-> !((JFXComboBox) getSkinnable()).isLabelFloat() ? null : promptTextScale.yProperty()) .setEndValue(0.85) .setInterpolator(Interpolator.EASE_BOTH).build()) ); JFXAnimationTimer unfocusTimer = new JFXAnimationTimer( new JFXKeyFrame(Duration.millis(160), JFXKeyValue.builder() .setTargetSupplier(()-> !((JFXComboBox) getSkinnable()).isLabelFloat() ? null : promptText.translateYProperty()) .setEndValue(0) .setInterpolator(Interpolator.EASE_BOTH).build(), JFXKeyValue.builder() .setTargetSupplier(()-> !((JFXComboBox) getSkinnable()).isLabelFloat() ? null : promptTextScale.xProperty()) .setEndValue(1) .setInterpolator(Interpolator.EASE_BOTH).build(), JFXKeyValue.builder() .setTargetSupplier(()-> !((JFXComboBox) getSkinnable()).isLabelFloat() ? null : promptTextScale.yProperty()) .setEndValue(1) .setInterpolator(Interpolator.EASE_BOTH).build()) ); private BooleanBinding usePromptText = Bindings.createBooleanBinding(() -> usePromptText(), ((JFXComboBox) getSkinnable()).valueProperty(), getSkinnable().promptTextProperty()); /*************************************************************************** * * * Constructors * * * **************************************************************************/ public JFXComboBoxListViewSkin(final JFXComboBox comboBox) { super(comboBox); // create my custom pane for the prompt node promptText.textProperty().bind(comboBox.promptTextProperty()); promptText.fillProperty().bind(animatedPromptTextFill); promptText.getStyleClass().addAll("text", "prompt-text"); promptText.getTransforms().add(promptTextScale); if (!comboBox.isLabelFloat()) { promptText.visibleProperty().bind(usePromptText); } customPane = new StackPane(); customPane.setMouseTransparent(true); customPane.getStyleClass().add("combo-box-button-container"); customPane.getChildren().add(promptText); getChildren().add(0, customPane); StackPane.setAlignment(promptText, Pos.CENTER_LEFT); // add lines line.getStyleClass().add("input-line"); focusedLine.getStyleClass().add("input-focused-line"); getChildren().add(line); getChildren().add(focusedLine); line.setPrefHeight(1); line.setTranslateY(1); // translate = prefHeight + init_translation line.setManaged(false); line.setBackground(new Background(new BackgroundFill(((JFXComboBox) getSkinnable()).getUnFocusColor(), CornerRadii.EMPTY, Insets.EMPTY))); if (getSkinnable().isDisabled()) { line.setBorder(new Border(new BorderStroke(((JFXComboBox) 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(((JFXComboBox) getSkinnable()).getFocusColor(), CornerRadii.EMPTY, Insets.EMPTY))); focusedLine.setOpacity(0); focusedLine.getTransforms().add(scale); focusedLine.setManaged(false); if (comboBox.isEditable()) { comboBox.getEditor().setStyle("-fx-background-color:TRANSPARENT;-fx-padding: 4 0 4 0"); comboBox.getEditor().promptTextProperty().unbind(); comboBox.getEditor().setPromptText(null); comboBox.getEditor().textProperty().addListener((o, oldVal, newVal) -> { usePromptText.invalidate(); comboBox.setValue(getConverter().fromString(newVal)); }); } comboBox.labelFloatProperty().addListener((o, oldVal, newVal) -> { if (newVal) { promptText.visibleProperty().unbind(); } else { promptText.visibleProperty().bind(usePromptText); } }); comboBox.focusColorProperty().addListener(observable -> { if (comboBox.getFocusColor() != null) { focusedLine.setBackground(new Background(new BackgroundFill(comboBox.getFocusColor(), CornerRadii.EMPTY, Insets.EMPTY))); } }); comboBox.unFocusColorProperty().addListener(observable -> { if (comboBox.getUnFocusColor() != null) { line.setBackground(new Background(new BackgroundFill(comboBox.getUnFocusColor(), CornerRadii.EMPTY, Insets.EMPTY))); } }); comboBox.disabledProperty().addListener(observable -> { line.setBorder(comboBox.isDisabled() ? new Border(new BorderStroke(((JFXComboBox) getSkinnable()).getUnFocusColor(), BorderStrokeStyle.DASHED, CornerRadii.EMPTY, new BorderWidths(line.getHeight()))) : Border.EMPTY); line.setBackground(new Background(new BackgroundFill(comboBox.isDisabled() ? Color.TRANSPARENT : ((JFXComboBox) getSkinnable()) .getUnFocusColor(), CornerRadii.EMPTY, Insets.EMPTY))); }); // handle animation on focus gained/lost event comboBox.focusedProperty().addListener((o, oldVal, newVal) -> { if (newVal) { focus(); } else { unFocus(); } }); promptTextFill.addListener(observable -> { oldPromptTextFill = promptTextFill.get(); animatedPromptTextFill.set(promptTextFill.get()); }); // handle animation on value changed comboBox.valueProperty().addListener(observable -> { if (!getSkinnable().isFocused() && ((JFXComboBox) getSkinnable()).isLabelFloat()) { T value = comboBox.getValue(); if (value == null || value.toString().isEmpty()) { animateFloatingLabel(false); } else { animateFloatingLabel(true); } } }); } /*************************************************************************** * * * Public API * * * **************************************************************************/ @Override protected void layoutChildren(final double x, final double y, final double w, final double h) { super.layoutChildren(x, y, w, h); customPane.resizeRelocate(x, y, w, h); if (invalid) { invalid = false; // create floating label // set initial prompt text fill using javafx prompt node fill animatedPromptTextFill.set(promptTextFill.get()); // focusTimer.setCacheNodes(promptText); // unfocusTimer.setCacheNodes(promptText); if(!getSkinnable().isEditable()){ final Text javaPromptText = (Text) ((ListCell) super.getDisplayNode()).lookup(".text"); if(javaPromptText!=null) animatedPromptTextFill.set(javaPromptText.getFill()); } if(getSkinnable().getValue()!=null) animateFloatingLabel(true); } focusedLine.resizeRelocate(x, getSkinnable().getHeight(), w, focusedLine.prefHeight(-1)); line.resizeRelocate(x, getSkinnable().getHeight(), w, line.prefHeight(-1)); scale.setPivotX(w / 2); } private void focus() { // create the focus animations unfocusTimer.stop(); focusTimer.start(); } /** * 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 { if (up) { unfocusTimer.stop(); focusTimer.start(); } else if (!up) { focusTimer.stop(); unfocusTimer.start(); } } } private void unFocus() { focusTimer.stop(); scale.setX(initScale); focusedLine.setOpacity(0); if (((JFXComboBox) getSkinnable()).isLabelFloat() && oldPromptTextFill != null) { animatedPromptTextFill.set(oldPromptTextFill); if (usePromptText()) { unfocusTimer.start(); } } } private boolean usePromptText() { Object txt = ((JFXComboBox) getSkinnable()).getValue(); String promptTxt = getSkinnable().getPromptText(); boolean isLabelFloat = ((JFXComboBox) getSkinnable()).isLabelFloat(); return (txt == null || txt.toString().isEmpty()) && promptTxt != null && !promptTxt.isEmpty() && (!promptTextFill.get().equals(Color.TRANSPARENT) || isLabelFloat); } private static class StyleableProperties { private static final CssMetaData PROMPT_TEXT_FILL = new CssMetaData("-fx-prompt-text-fill", PaintConverter.getInstance(), Color.GRAY) { @Override public boolean isSettable(JFXComboBox n) { final JFXComboBoxListViewSkin skin = (JFXComboBoxListViewSkin) n.getSkin(); return skin.promptTextFill == null || !skin.promptTextFill.isBound(); } @Override @SuppressWarnings("unchecked") public StyleableProperty getStyleableProperty(JFXComboBox n) { final JFXComboBoxListViewSkin skin = (JFXComboBoxListViewSkin) n.getSkin(); return (StyleableProperty)skin.promptTextFill; } }; private static final List> STYLEABLES; static { List> styleables = new ArrayList>(ComboBoxListViewSkin.getClassCssMetaData()); styleables.add(PROMPT_TEXT_FILL); STYLEABLES = Collections.unmodifiableList(styleables); } } /** * @return The CssMetaData associated with this class, which may include the * CssMetaData of its super classes. */ public static List> getClassCssMetaData() { return StyleableProperties.STYLEABLES; } /** * {@inheritDoc} */ @Override public List> getCssMetaData() { return getClassCssMetaData(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy