
gwt.material.design.addins.client.pathanimator.MaterialPathAnimator Maven / Gradle / Ivy
/*
* #%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%
*/
package gwt.material.design.addins.client.pathanimator;
import com.google.gwt.dom.client.Element;
import com.google.gwt.dom.client.Style;
import com.google.gwt.user.client.ui.Widget;
import gwt.material.design.addins.client.MaterialAddins;
import gwt.material.design.addins.client.base.constants.AddinsCssName;
import gwt.material.design.addins.client.pathanimator.base.HasPathStyles;
import gwt.material.design.addins.client.pathanimator.base.PathStyleProperty;
import gwt.material.design.addins.client.pathanimator.base.PathStylerMixin;
import gwt.material.design.addins.client.pathanimator.js.JsPathAnimator;
import gwt.material.design.addins.client.pathanimator.js.JsPathAnimatorOptions;
import gwt.material.design.client.MaterialDesignBase;
import gwt.material.design.client.base.HasDurationTransition;
import gwt.material.design.client.base.helper.ScrollHelper;
import gwt.material.design.client.constants.Color;
import gwt.material.design.client.constants.OffsetPosition;
import gwt.material.design.jquery.client.api.Functions;
import static gwt.material.design.jquery.client.api.JQuery.$;
//@formatter:off
/**
* Custom component that provides meaningfull transition between two elements to show visual continuity.
*
*
* {@code
*
* // CAN BE CALLED AS A HELPER STATIC CONTEXT
* MaterialPathAnimator.animate(Element source, Element target, Functions.Func animateCallback);
*
* // INSTANTIATE THE PUSHPIN COMPONENT
* MaterialPathAnimator animator = new MaterialPathAnimator();
* animator.setSourceElement(btnSource1.getElement());
* animator.setTargetElement(panelTarget1.getElement());
* animator.animate();
* // Reverse Animate
* animator.reverseAnimate();
* }
*
*
* @author kevzlou7979
* @see Material Path Animator
* @see CTAJs 0.3.2
*/
//@formatter:on
public class MaterialPathAnimator implements HasDurationTransition, HasPathStyles {
static {
if (MaterialAddins.isDebug()) {
MaterialDesignBase.injectDebugJs(MaterialPathAnimatorDebugClientBundle.INSTANCE.pathanimatorDebugJs());
} else {
MaterialDesignBase.injectJs(MaterialPathAnimatorClientBundle.INSTANCE.pathanimatorJs());
}
}
private ScrollHelper scrollHelper;
private Element sourceElement;
private Element targetElement;
private Functions.Func animateCallback, startAnimateCallback, reverseCallback, completedCallback;
private JsPathAnimatorOptions options = JsPathAnimatorOptions.create();
private PathStylerMixin stylerMixin;
public MaterialPathAnimator() {
this.scrollHelper = new ScrollHelper();
}
public MaterialPathAnimator(Element sourceElement, Element targetElement) {
this();
this.sourceElement = sourceElement;
this.targetElement = targetElement;
}
/**
* Helper method to apply the path animator.
*
* @param source Source element to apply the Path Animator
* @param target Target element to apply the Path Animator
*/
public static void animate(Element source, final Element target) {
animate(source, target, null);
}
/**
* Helper method to apply the path animator.
*
* @param source Source widget to apply the Path Animator
* @param target Target widget to apply the Path Animator
*/
public static void animate(Widget source, final Widget target) {
animate(source.getElement(), target.getElement());
}
/**
* Helper method to apply the path animator with callback.
*
* @param source Source widget to apply the Path Animator
* @param target Target widget to apply the Path Animator
* @param callback The callback method to be called when the path animator is applied
*/
public static void animate(Widget source, Widget target, Functions.Func callback) {
animate(source.getElement(), target.getElement(), callback);
}
/**
* Helper method to apply the path animator with callback.
*
* @param sourceElement Source widget to apply the Path Animator
* @param targetElement Target widget to apply the Path Animator
* @param animateCallback The callback method to be called when the path animator is applied
*/
public static void animate(Element sourceElement, Element targetElement, Functions.Func animateCallback) {
MaterialPathAnimator animator = new MaterialPathAnimator();
animator.setSourceElement(sourceElement);
animator.setTargetElement(targetElement);
animator.setAnimateCallback(animateCallback);
animator.animate();
}
/**
* Animate the path animator
*/
public void animate() {
detectOutOfScopeElement(targetElement, () -> {
$("document").ready(() -> {
onStartAnimateCallback();
JsPathAnimator.cta(sourceElement, targetElement, options, () -> {
if (animateCallback != null) {
animateCallback.call();
} else {
// For default animateCallback when animateCallback is null
targetElement.getStyle().setVisibility(Style.Visibility.VISIBLE);
targetElement.getStyle().setOpacity(1);
}
if (completedCallback != null) {
completedCallback.call();
}
});
});
});
}
/**
* Reverse the Animation
*/
public void reverseAnimate() {
onStartAnimateCallback();
$("document").ready(() -> {
if (reverseCallback != null) {
reverseCallback.call();
} else {
targetElement.getStyle().setVisibility(Style.Visibility.HIDDEN);
targetElement.getStyle().setOpacity(0);
}
detectOutOfScopeElement(sourceElement, () -> JsPathAnimator.cta(targetElement, sourceElement, options, () -> onCompleteCallback()));
});
}
/**
* Will detect if the target / source element is out of scope in the viewport.
* If it is then we will call {@link ScrollHelper#scrollTo(double)} with default offset position
* of {@link OffsetPosition#MIDDLE}.
*/
protected void detectOutOfScopeElement(Element element, Functions.Func callback) {
if (scrollHelper.isInViewPort(element)) {
callback.call();
} else {
scrollHelper.setOffsetPosition(OffsetPosition.MIDDLE);
scrollHelper.setCompleteCallback(() -> callback.call());
scrollHelper.scrollTo(element);
}
}
protected void onStartAnimateCallback() {
if (startAnimateCallback != null) {
startAnimateCallback.call();
}
}
protected void onCompleteCallback() {
if (completedCallback != null) {
completedCallback.call();
}
}
/**
* Helper method to reverse animate the source element to target element.
*
* @param sourceElement Source element to apply the Path Animator
* @param targetElement Target element to apply the Path Animator
*/
public static void reverseAnimate(final Element sourceElement, final Element targetElement) {
reverseAnimate(sourceElement, targetElement, null);
}
/**
* Helper method to reverse animate the source element to target element.
*
* @param source Source widget to apply the Path Animator
* @param target Target widget to apply the Path Animator
*/
public static void reverseAnimate(final Widget source, final Widget target) {
reverseAnimate(source.getElement(), target.getElement());
}
/**
* Helper method to reverse animate the source element to target element with reverse callback.
*
* @param source Source widget to apply the Path Animator
* @param target Target widget to apply the Path Animator
* @param reverseCallback The reverse callback method to be called when the path animator is applied
*/
public static void reverseAnimate(Widget source, Widget target, Functions.Func reverseCallback) {
reverseAnimate(source.getElement(), target.getElement(), reverseCallback);
}
/**
* Helper method to reverse animate the source element to target element with reverse callback
*
* @param sourceElement Source element to apply the Path Animator
* @param targetElement Target element to apply the Path Animator
* @param reverseCallback The reverse callback method to be called when the path animator is applied
*/
public static void reverseAnimate(Element sourceElement, Element targetElement, Functions.Func reverseCallback) {
MaterialPathAnimator animator = new MaterialPathAnimator();
animator.setSourceElement(sourceElement);
animator.setTargetElement(targetElement);
animator.setReverseCallback(reverseCallback);
animator.reverseAnimate();
}
/**
* Get the source element
*/
public Element getSourceElement() {
return sourceElement;
}
/**
* Set the source element
*/
public void setSourceElement(Element sourceElement) {
this.sourceElement = sourceElement;
}
/**
* Get the target element
*/
public Element getTargetElement() {
return targetElement;
}
/**
* Set the target element
*/
public void setTargetElement(Element targetElement) {
this.targetElement = targetElement;
}
/**
* Get the callback method when the path animator is applied
*/
public Functions.Func getAnimateCallback() {
return animateCallback;
}
/**
* Set the callback method when the path animator is applied
*/
public void setAnimateCallback(Functions.Func animateCallback) {
this.animateCallback = animateCallback;
}
public void setAnimateOnStartCallback(Functions.Func startAnimateCallback) {
this.startAnimateCallback = startAnimateCallback;
}
/**
* Get the reverse callback method when the path animator is applied
*/
public Functions.Func getReverseCallback() {
return reverseCallback;
}
/**
* Set the reverse callback method when the path animator is applied
*/
public void setReverseCallback(Functions.Func reverseCallback) {
this.reverseCallback = reverseCallback;
}
/**
* Set a callback for when the animation has completed.
*/
public void setCompletedCallback(Functions.Func completedCallback) {
this.completedCallback = completedCallback;
}
@Override
public void setDuration(int duration) {
options.duration = duration / 1000.0;
}
@Override
public int getDuration() {
return (int) (options.duration * 1000);
}
/**
* Duration (in milliseconds) of targetElement to become visible, if hidden initially. The library will automatically try to figure this out from the element's computed styles. Default is 0 seconds.
*/
public void setTargetShowDuration(int targetShowDuration) {
options.targetShowDuration = targetShowDuration / 1000.0;
}
public int getTargetShowDuration() {
return (int) (options.targetShowDuration * 1000);
}
/**
* Extra duration (in milliseconds) of targetElement to provide visual continuity between the animation and the rendering of the targetElement. Default is 1 second
*/
public void setExtraTransitionDuration(int extraTransitionDuration) {
options.extraTransitionDuration = extraTransitionDuration / 1000.0;
}
public int getExtraTransitionDuration() {
return (int) (options.extraTransitionDuration * 1000);
}
public boolean isRelativeToWindow() {
return options.relativeToWindow;
}
/**
* Set to true if your target element is fixed positioned in the window. Default is relative to document (works good with normal elements).
*/
public void setRelativeToWindow(boolean relativeToWindow) {
options.relativeToWindow = relativeToWindow;
}
public ScrollHelper getScrollHelper() {
return scrollHelper;
}
@Override
public void setBackgroundColor(Color backgroundColor) {
getStylerMixin().setBackgroundColor(backgroundColor);
}
@Override
public void setStyleProperty(PathStyleProperty property) {
getStylerMixin().setStyleProperty(property);
}
@Override
public void setStyleProperty(String property, String value) {
getStylerMixin().setStyleProperty(property, value);
}
@Override
public void clearStyleProperty(String property) {
getStylerMixin().clearStyleProperty(property);
}
@Override
public void clearStyles() {
getStylerMixin().clearStyles();
}
public Element getBridgeElement() {
return getStylerMixin().getBridgeElement();
}
@Override
public void setShadow(Integer shadowDepth) {
getStylerMixin().setShadow(shadowDepth);
}
protected PathStylerMixin getStylerMixin() {
if (stylerMixin == null) {
stylerMixin = new PathStylerMixin<>(this);
}
return stylerMixin;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy