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

com.jfoenix.controls.JFXSnackbar Maven / Gradle / Ivy

There is a newer version: 17-0.1.1
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.controls;

import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.atomic.AtomicBoolean;

import javafx.animation.Interpolator;
import javafx.animation.KeyFrame;
import javafx.animation.KeyValue;
import javafx.animation.PauseTransition;
import javafx.animation.Timeline;
import javafx.application.Platform;
import javafx.beans.value.ChangeListener;
import javafx.beans.value.WeakChangeListener;
import javafx.css.PseudoClass;
import javafx.event.Event;
import javafx.event.EventType;
import javafx.geometry.Bounds;
import javafx.scene.Group;
import javafx.scene.Node;
import javafx.scene.layout.Pane;
import javafx.scene.layout.StackPane;
import javafx.util.Duration;

/**
 * "Snackbars provide brief messages about app processes at the bottom of the screen"
 * (Material Design Guidelines).
 * 

* To show a snackbar you need to *

    *
  1. Have a {@link Pane} (snackbarContainer) to show the snackbar on top of. Register it in * {@link #JFXSnackbar(Pane) the JFXSnackbar constructor} or using the * {@link #registerSnackbarContainer(Pane)} method.
  2. *
  3. Have or create a {@link JFXSnackbar}. *
      *
    • Having one snackbar where you pass all your {@link JFXSnackbar.SnackbarEvent SnackbarEvents} * will ensure that the {@link JFXSnackbar#enqueue(SnackbarEvent) enqueue method} works as * intended.
    • *
    *
  4. *
  5. Have something to show in the snackbar. A {@link JFXSnackbarLayout} is nice and pretty, but * any arbitrary {@link Node} will do.
  6. *
  7. Create a {@link JFXSnackbar.SnackbarEvent SnackbarEvent} specifying the contents and the * duration.
  8. *
*

* Finally, with all those things prepared, show your snackbar using * {@link JFXSnackbar#enqueue(SnackbarEvent) snackbar.enqueue(snackbarEvent);}. *

* It's most convenient to create functions to do most of this (creating the layout and event) with * the default settings; that way all you need to do to show a snackbar is specify the message or * just the message and the duration. * * @see The Material Design * Snackbar */ public class JFXSnackbar extends Group { private static final String DEFAULT_STYLE_CLASS = "jfx-snackbar"; private Pane snackbarContainer; private ChangeListener sizeListener = (o, oldVal, newVal) -> refreshPopup(); private WeakChangeListener weakSizeListener = new WeakChangeListener<>(sizeListener); private AtomicBoolean processingQueue = new AtomicBoolean(false); private ConcurrentLinkedQueue eventQueue = new ConcurrentLinkedQueue<>(); private ConcurrentHashMap.KeySetView eventsSet = ConcurrentHashMap.newKeySet(); private Interpolator easeInterpolator = Interpolator.SPLINE(0.250, 0.100, 0.250, 1.000); private Pane content; private PseudoClass activePseudoClass = null; private PauseTransition pauseTransition; /** * This constructor assumes that you will eventually call the * {@link #registerSnackbarContainer(Pane)} method before calling the * {@link #enqueue(SnackbarEvent)} method. Otherwise, how will the snackbar know where to show * itself? *

*

* "Snackbars provide brief messages about app processes at the bottom of the screen" * (Material Design Guidelines). *

* To show a snackbar you need to *

    *
  1. Have a {@link Pane} (snackbarContainer) to show the snackbar on top of. Register it in * {@link #JFXSnackbar(Pane) the JFXSnackbar constructor} or using the * {@link #registerSnackbarContainer(Pane)} method.
  2. *
  3. Have or create a {@link JFXSnackbar}. *
      *
    • Having one snackbar where you pass all your {@link JFXSnackbar.SnackbarEvent SnackbarEvents} * will ensure that the {@link JFXSnackbar#enqueue(SnackbarEvent) enqueue method} works as * intended.
    • *
    *
  4. *
  5. Have something to show in the snackbar. A {@link JFXSnackbarLayout} is nice and pretty, but * any arbitrary {@link Node} will do.
  6. *
  7. Create a {@link JFXSnackbar.SnackbarEvent SnackbarEvent} specifying the contents and the * duration.
  8. *
*

* Finally, with all those things prepared, show your snackbar using * {@link JFXSnackbar#enqueue(SnackbarEvent) snackbar.enqueue(snackbarEvent);}. *

*/ public JFXSnackbar() { this(null); } /** * "Snackbars provide brief messages about app processes at the bottom of the screen" * (Material Design Guidelines). *

* To show a snackbar you need to *

    *
  1. Have a {@link Pane} (snackbarContainer) to show the snackbar on top of. Register it in * {@link #JFXSnackbar(Pane) the JFXSnackbar constructor} or using the * {@link #registerSnackbarContainer(Pane)} method.
  2. *
  3. Have or create a {@link JFXSnackbar}. *
      *
    • Having one snackbar where you pass all your {@link JFXSnackbar.SnackbarEvent SnackbarEvents} * will ensure that the {@link JFXSnackbar#enqueue(SnackbarEvent) enqueue method} works as * intended.
    • *
    *
  4. *
  5. Have something to show in the snackbar. A {@link JFXSnackbarLayout} is nice and pretty, but * any arbitrary {@link Node} will do.
  6. *
  7. Create a {@link JFXSnackbar.SnackbarEvent SnackbarEvent} specifying the contents and the * duration.
  8. *
*

* Finally, with all those things prepared, show your snackbar using * {@link JFXSnackbar#enqueue(SnackbarEvent) snackbar.enqueue(snackbarEvent);}. *

* * @param snackbarContainer where the snackbar will appear. Using a single snackbar instead of many, * will ensure that the {@link #enqueue(SnackbarEvent)} method works * correctly. */ public JFXSnackbar(Pane snackbarContainer) { initialize(); content = new StackPane(); content.getStyleClass().add("jfx-snackbar-content"); //wrap the content in a group so that the content is managed inside its own container //but the group is not managed in the snackbarContainer so it does not affect any layout calculations getChildren().add(content); setManaged(false); setVisible(false); // register the container before resizing it registerSnackbarContainer(snackbarContainer); // resize the popup if its layout has been changed layoutBoundsProperty().addListener((o, oldVal, newVal) -> refreshPopup()); addEventHandler(SnackbarEvent.SNACKBAR, e -> enqueue(e)); } private void initialize() { this.getStyleClass().add(DEFAULT_STYLE_CLASS); } /////////////////////////////////////////////////////////////////////////// // Setters / Getters /////////////////////////////////////////////////////////////////////////// public Pane getPopupContainer() { return snackbarContainer; } public void setPrefWidth(double width) { content.setPrefWidth(width); } public double getPrefWidth() { return content.getPrefWidth(); } /////////////////////////////////////////////////////////////////////////// // Public API /////////////////////////////////////////////////////////////////////////// public void registerSnackbarContainer(Pane snackbarContainer) { if (snackbarContainer != null) { if (this.snackbarContainer != null) { //since listeners are added the container should be properly registered/unregistered throw new IllegalArgumentException("Snackbar Container already set"); } this.snackbarContainer = snackbarContainer; this.snackbarContainer.getChildren().add(this); this.snackbarContainer.heightProperty().addListener(weakSizeListener); this.snackbarContainer.widthProperty().addListener(weakSizeListener); } } public void unregisterSnackbarContainer(Pane snackbarContainer) { if (snackbarContainer != null) { if (this.snackbarContainer == null) { throw new IllegalArgumentException("Snackbar Container not set"); } this.snackbarContainer.getChildren().remove(this); this.snackbarContainer.heightProperty().removeListener(weakSizeListener); this.snackbarContainer.widthProperty().removeListener(weakSizeListener); this.snackbarContainer = null; } } private void show(SnackbarEvent event) { content.getChildren().setAll(event.getContent()); openAnimation = getTimeline(event.getTimeout()); if (event.getPseudoClass() != null) { activePseudoClass = event.getPseudoClass(); content.pseudoClassStateChanged(activePseudoClass, true); } openAnimation.play(); } private Timeline openAnimation = null; private Timeline getTimeline(Duration timeout) { Timeline animation; animation = new Timeline(new KeyFrame(Duration.ZERO, e -> this.toBack(), new KeyValue(this.visibleProperty(), false, Interpolator.EASE_BOTH), new KeyValue(this.translateYProperty(), this.getLayoutBounds().getHeight(), easeInterpolator), new KeyValue(this.opacityProperty(), 0, easeInterpolator)), new KeyFrame(Duration.millis(10), e -> this.toFront(), new KeyValue(this.visibleProperty(), true, Interpolator.EASE_BOTH)), new KeyFrame(Duration.millis(300), new KeyValue(this.opacityProperty(), 1, easeInterpolator), new KeyValue(this.translateYProperty(), 0, easeInterpolator))); animation.setCycleCount(1); pauseTransition = Duration.INDEFINITE.equals(timeout) ? null : new PauseTransition(timeout); if (pauseTransition != null) { animation.setOnFinished(finish -> { pauseTransition.setOnFinished(done -> { pauseTransition = null; eventsSet.remove(currentEvent); currentEvent = eventQueue.peek(); close(); }); pauseTransition.play(); }); } return animation; } public void close() { if (openAnimation != null) { openAnimation.stop(); } if (this.isVisible()) { Timeline closeAnimation = new Timeline(new KeyFrame(Duration.ZERO, e -> this.toFront(), new KeyValue(this.opacityProperty(), 1, easeInterpolator), new KeyValue(this.translateYProperty(), 0, easeInterpolator)), new KeyFrame(Duration.millis(290), new KeyValue(this.visibleProperty(), true, Interpolator.EASE_BOTH)), new KeyFrame(Duration.millis(300), e -> this.toBack(), new KeyValue(this.visibleProperty(), false, Interpolator.EASE_BOTH), new KeyValue(this.translateYProperty(), this.getLayoutBounds().getHeight(), easeInterpolator), new KeyValue(this.opacityProperty(), 0, easeInterpolator))); closeAnimation.setCycleCount(1); closeAnimation.setOnFinished(e -> { resetPseudoClass(); processSnackbar(); }); closeAnimation.play(); } } private SnackbarEvent currentEvent = null; public SnackbarEvent getCurrentEvent() { return currentEvent; } /** * Shows {@link SnackbarEvent SnackbarEvents} one by one. The next event will be shown after the * current event's duration. * * @param event the {@link SnackbarEvent event} to put in the queue. */ public void enqueue(SnackbarEvent event) { synchronized (this) { if (!eventsSet.contains(event)) { eventsSet.add(event); eventQueue.offer(event); } else if (currentEvent == event && pauseTransition != null) { pauseTransition.playFromStart(); } } if (processingQueue.compareAndSet(false, true)) { Platform.runLater(() -> { currentEvent = eventQueue.poll(); if (currentEvent != null) { show(currentEvent); } }); } } private void resetPseudoClass() { if (activePseudoClass != null) { content.pseudoClassStateChanged(activePseudoClass, false); activePseudoClass = null; } } private void processSnackbar() { currentEvent = eventQueue.poll(); if (currentEvent != null) { eventsSet.remove(currentEvent); show(currentEvent); } else { //The enqueue method and this listener should be executed sequentially on the FX Thread so there //should not be a race condition processingQueue.getAndSet(false); } } private void refreshPopup() { if (snackbarContainer == null) { return; } Bounds contentBound = this.getLayoutBounds(); double offsetX = Math.ceil(snackbarContainer.getWidth() / 2) - Math.ceil(contentBound.getWidth() / 2); double offsetY = snackbarContainer.getHeight() - contentBound.getHeight(); this.setLayoutX(offsetX); this.setLayoutY(offsetY); } /////////////////////////////////////////////////////////////////////////// // Event API /////////////////////////////////////////////////////////////////////////// /** * Specifies what and how long to show a {@link JFXSnackbar}. *

* The what can be any arbitrary {@link Node}; the {@link JFXSnackbarLayout} is a great * choice. *

* The how long is specified in the form of a {@link javafx.util.Duration * javafx.util.Duration}, not to be confused with the {@link java.time.Duration}. */ public static class SnackbarEvent extends Event { public static final EventType SNACKBAR = new EventType<>(Event.ANY, "SNACKBAR"); /** * The amount of time the snackbar will show for, if not otherwise specified. *

* It's 1.5 seconds. */ public static Duration DEFAULT_DURATION = Duration.seconds(1.5); private final Node content; private final PseudoClass pseudoClass; private final Duration timeout; /** * Creates a {@link SnackbarEvent} with the {@link #DEFAULT_DURATION default duration} and no * pseudoClass. * * @param content what you want shown in the snackbar; a {@link JFXSnackbarLayout} is a great * choice. */ public SnackbarEvent(Node content) { this(content, DEFAULT_DURATION, null); } /** * Creates a {@link SnackbarEvent} with the {@link #DEFAULT_DURATION default duration}; you specify * the contents and pseudoClass. * * @param content what you want shown in the snackbar; a {@link JFXSnackbarLayout} is a great * choice. * @param pseudoClass */ public SnackbarEvent(Node content, PseudoClass pseudoClass) { this(content, DEFAULT_DURATION, pseudoClass); } /** * Creates a SnackbarEvent with no pseudoClass; you specify the contents and duration. pseudoClass. * * @param content what you want shown in the snackbar; a {@link JFXSnackbarLayout} is a great * choice. * @param timeout the amount of time you want the snackbar to show for. */ public SnackbarEvent(Node content, Duration timeout) { this(content, timeout, null); } /** * Creates a SnackbarEvent; you specify the contents, duration and pseudoClass. *

* If you don't need so much customization, try one of the other constructors. * * @param content what you want shown in the snackbar; a {@link JFXSnackbarLayout} is a great * choice. * @param timeout the amount of time you want the snackbar to show for. * @param pseudoClass */ public SnackbarEvent(Node content, Duration timeout, PseudoClass pseudoClass) { super(SNACKBAR); this.content = content; this.pseudoClass = pseudoClass; this.timeout = timeout; } public Node getContent() { return content; } public PseudoClass getPseudoClass() { return pseudoClass; } public Duration getTimeout() { return timeout; } @Override public EventType getEventType() { return (EventType) super.getEventType(); } public boolean isPersistent() { return Duration.INDEFINITE.equals(getTimeout()); } } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy