我想要一些有关如何在用户按下按钮时在窗格的过渡中实现幻灯片的准则,就像Material Design在滑动菜单中所做的一样。

这是一个视频link,说明了我的需求。

我尝试了ScaleTransition,TranslateTransition,但他们没有成功。

我尝试实现它的方式效率不高。

// swipeMenuPane is builded in SceneBuilder and it is hidden,
// opacity = 0.0 and setX() = -getPrefWidth();
@FXML AnchorPane swipeMenuPane;
@FXML Button menuButton;

menuButton.setOnMouseClicked(e-> {
    swipeMenuPane.setOpacity(1.0);
    swipeTransition.play()
});

TranslateTransition swipeTransition = new TranslateTransition();
swipeTransition.setNode(swipeMenuPane);
swipeTransition.setDuration(Duration.millis(500));
swipeTransition.setToX(swipeMenuPane.getPrefWidth());




-更新-

这是从here下载的示例Gluon应用程序。这是一个gradle项目,我对其进行了修改,以显示按钮而不是默认标签。

我想在用户单击按钮时缩小AnchorPane。

我想念什么?

package com.helloworld;

import com.gluonhq.charm.glisten.animation.ShrinkExpandAnimation;
import javafx.application.Application;
import javafx.scene.Scene;
import javafx.scene.control.Button;
import javafx.scene.layout.AnchorPane;
import javafx.scene.layout.StackPane;
import javafx.stage.Stage;

public class HelloWorld extends Application {

    ShrinkExpandAnimation anim;

    @Override
    public void start(Stage stage) {
        Button btn = new Button("Click Me!");
        btn.setOnMouseClicked(e-> {
            System.out.println("swiping...");
            anim.play();
        });

        AnchorPane pane = new AnchorPane();
        pane.setStyle("-fx-background-color: coral");
        pane.getChildren().add(btn);

        // false to shrink or true to expand
        anim = new ShrinkExpandAnimation(pane, false);

        Scene scene = new Scene(new StackPane(pane), 640, 480);

        stage.setScene(scene);
        stage.show();
    }
}




-更新2-

我设法使用本机JavaFX API实现了与我想要的东西相似的东西,并且没有外部库。

虽然,我遇到了一些问题。


缩小AnchorPane不会缩小/移动其任何子节点,因为它们停留在布局位置。
缩小除AnchorPane之外的其他任何窗格都可以缩小/移动其子节点,但ImageView节点除外。


下两张图片说明了我遇到的第一个问题。

这是AnchorPane(灰色的根窗格)中的AnchorPane(全宽度为珊瑚色;展开)。

javafx - JavaFX:如何在 Pane 的动画效果中创建幻灯片(在透明舞台内)-LMLPHP

这就是单击菜单按钮以缩小/隐藏时发生的情况。如您所见,珊瑚色窗格被缩小/隐藏了,但它的节点却没有(Label,ImageView)

javafx - JavaFX:如何在 Pane 的动画效果中创建幻灯片(在透明舞台内)-LMLPHP

我发布了整个代码以自己重现该问题:

public class SwipeMenuDemo extends Application {

    AnchorPane swapPane;
    Button btnMenu;
    boolean isExpanded = true;

    @Override
    public void start(Stage stage) {
        Label swapPaneLabel = new Label("Expandable Pane");
        swapPaneLabel.setMinWidth(0);

        ImageView swapPaneImage = new ImageView("http://vignette1.wikia.nocookie.net/jfx/images/5/5a/JavaFXIsland600x300.png");
        swapPaneImage.setLayoutY(100);

        Label rootPaneLabel = new Label("Root Pane");
        rootPaneLabel.setStyle("-fx-font-size: 60;");
        rootPaneLabel.setLayoutX(180);
        rootPaneLabel.setLayoutY(180);

        swapPane = new AnchorPane();
        swapPane.setPrefSize(640, 440);
        swapPane.setMinWidth(0);
        swapPane.setLayoutY(40);
        swapPane.setStyle("-fx-background-color: coral; -fx-font-size: 52;");
        swapPane.getChildren().addAll(swapPaneImage, swapPaneLabel);

        btnMenu = new Button("Menu");
        btnMenu.setLayoutX(5);
        btnMenu.setLayoutY(5);
        btnMenu.setOnMouseClicked(e -> {
            if (isExpanded) hideSwapPane().play();
            else showSwapPane().play();
        });

        Button btnClose = new Button("Close");
        btnClose.setLayoutX(590);
        btnClose.setLayoutY(5);
        btnClose.setOnMouseClicked(e -> Platform.exit());

        AnchorPane rootPane = new AnchorPane();
        rootPane.setStyle("-fx-background-color: grey;");
        rootPane.getChildren().addAll(btnMenu, btnClose, rootPaneLabel, swapPane);

        Scene scene = new Scene(rootPane, 640, 480);

        stage.setScene(scene);
        stage.initStyle(StageStyle.UNDECORATED);
        stage.show();
    }

    private Animation hideSwapPane() {
        btnMenu.setMouseTransparent(true);

        Animation collapsePanel = new Transition() {
            {
                setCycleDuration(Duration.millis(2500));
            }

            @Override
            protected void interpolate(double fraction) {
                swapPane.setPrefWidth(640 * (1.0 - fraction));
            }
        };

        collapsePanel.setOnFinished(e-> {
            isExpanded = false;
            btnMenu.setMouseTransparent(false);
        });

        return collapsePanel;
    }

    private Animation showSwapPane() {
        btnMenu.setMouseTransparent(true);

        final Animation expandPanel = new Transition() {
            {
                setCycleDuration(Duration.millis(2500));
            }

            @Override
            protected void interpolate(double fraction) {
                swapPane.setPrefWidth(640 * fraction);
            }
        };

        expandPanel.setOnFinished(e-> {
            isExpanded = true;
            btnMenu.setMouseTransparent(false);
        });

        return expandPanel;
    }
}




-更新3-

根据我的需要,我修改了Felipe Guizar Diaz提供给我的代码,因为我想在透明的舞台窗口上添加阴影效果。

当我单击菜单按钮以显示左窗格时,它会显示在阴影的前面。即使在SceneBuilder中,我也已将具有阴影效果的StackPane放置在所有节点的前面。

当我按下以显示菜单并开始播放打开的过渡时,这是“工件”。

我该如何解决?

javafx - JavaFX:如何在 Pane 的动画效果中创建幻灯片(在透明舞台内)-LMLPHP

最佳答案

我是示例视频的作者。我将重复我在视频评论中所做的回复:
“您应该将其视为android中的导航抽屉,JavaFX中的导航抽屉将是具有两个子项的AnchorPane,首先是一个StackPane,它等效于FrameLayout作为我们的主要内容,其中窗格是根据左侧菜单中的选定项目制作的,最后是ListView作为我们的左侧菜单,其负translateX等于Listview宽度,然后当用户按下按钮时,将translateX的值保存为0的动画。”
您不应该在两个动画的内插方法中使用prefWidth()(折叠面板,展开窗格),因为子级不会调整大小,因此空白设置是AnchorPane的唯一约束。

看看我做过的这个例子。

https://github.com/marconideveloper/leftsidemenuexample

public class FXMLDocumentController implements Initializable {

    @FXML
    private Button menu;
    @FXML
    private AnchorPane navList;
    @Override
    public void initialize(URL url, ResourceBundle rb) {
    //navList.setItems(FXCollections.observableArrayList("Red","Yellow","Blue"));
        prepareSlideMenuAnimation();
    }

    private void prepareSlideMenuAnimation() {
        TranslateTransition openNav=new TranslateTransition(new Duration(350), navList);
        openNav.setToX(0);
        TranslateTransition closeNav=new TranslateTransition(new Duration(350), navList);
        menu.setOnAction((ActionEvent evt)->{
            if(navList.getTranslateX()!=0){
                openNav.play();
            }else{
                closeNav.setToX(-(navList.getWidth()));
                closeNav.play();
            }
        });
    }
}


这是fxml:

<AnchorPane xmlns:fx="http://javafx.com/fxml/1" id="AnchorPane" prefWidth="500" prefHeight="500"    fx:controller="leftslidemenusample.FXMLDocumentController">
    <children>

        <ToolBar AnchorPane.topAnchor="0.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0" minHeight="56.0"   >
            <Button text="menu" fx:id="menu"  />
        </ToolBar>
        <StackPane fx:id="mainContent"  style="-fx-background-color:rgba(0,0,0,0.30)" AnchorPane.bottomAnchor="0.0" AnchorPane.topAnchor="56.0" AnchorPane.leftAnchor="0.0" AnchorPane.rightAnchor="0.0"    >
            <children>

            </children>
        </StackPane>
        <AnchorPane fx:id="navList" style="-fx-background-color:white" AnchorPane.topAnchor="56.0" AnchorPane.bottomAnchor="0.0" prefWidth="180.0" translateX="-180"   >
            <children>
                <Label text="left side menu"/>
            </children>
        </AnchorPane>

    </children>

</AnchorPane>

关于javafx - JavaFX:如何在 Pane 的动画效果中创建幻灯片(在透明舞台内),我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/31601900/

10-10 08:56