#java #javafx #scrollpane #vbox
#java #javafx #область прокрутки #vbox
Вопрос:
У меня есть JavaFX VBox внутри области прокрутки:
VBox container = new VBox();
container.setAlignment(Pos.CENTER);
...
scrollPane.setContent(container);
scrollPane.setFitToWidth(true);
scrollPane.setHbarPolicy(ScrollBarPolicy.NEVER);
scrollPane.setVbarPolicy(ScrollBarPolicy.NEVER);
scrollPane.setMinWidth(150);
scrollPane.setPannable(true);
размер этого VBox никогда не меняется, внутри у меня есть несколько меток, одна метка имеет изображение пользователя
, подобное следующему изображению (A)
размер изображения изменен на некоторую высоту, но я не знаю размер этого изображения, поэтому, если ширина изображения больше ширины VBox, это произойдет (часть изображения скрыта) (B)
но я не хочу этого, я хочу что-то вроде следующего изображения: (стороны изображения скрыты, если ширина изображения больше ширины VBox) (C)
http://i.stack.imgur.com/B3DOK.png
Как я могу это сделать?
Я попытался поместить прямоугольник в качестве клипа, в этом прямоугольнике я хочу показать центр изображения, но происходит то же самое.
imageView.setClip(new Rectangle(centerX - recSize, centerY - recSize, recSize*2, recSize*2));
—————с помощью клипа—————-
красный = исходное изображение
синий = видимая часть изображения
http://i.stack.imgur.com/mYbyF.png
Хороший (D)
Нехорошо: (E) (метки не центрируются правильно из-за изображения.)
Извините за ссылки, я не могу размещать изображения напрямую
Ответ №1:
Решение
Установите окно просмотра для изображения, а не для клипа.
imageView.setViewport(
new Rectangle2D(500, 320, 420, 300)
);
Пример
Вот пример. Это не будет точно соответствовать тому, что вы просите, потому что даже со связанными изображениями в вашем вопросе я не совсем понимаю, что вы пытаетесь сделать. Но я думаю, что это должно дать вам достаточно справочной информации, чтобы вы могли научиться выполнять то, что хотите.
В примере создается представление изображения в виде графики на панели прокрутки. Вид изображения применяет окно просмотра к изображению и масштабирует окно просмотра с сохранением соотношения. Это позволяет отображать масштабированную часть изображения гораздо большего размера. Это что-то вроде миниатюры клипа (нажмите на миниатюру, чтобы отобразить полное изображение).
import javafx.application.Application;
import javafx.geometry.*;
import javafx.scene.*;
import javafx.scene.control.*;
import javafx.scene.effect.*;
import javafx.scene.image.*;
import javafx.scene.layout.VBox;
import javafx.scene.paint.Color;
import javafx.scene.text.*;
import javafx.stage.*;
// display a captioned image in a viewport.
// click the image to get an expanded view.
public class LabelWithImage extends Application {
private static final double IMAGE_WIDTH = 150;
@Override
public void start(Stage stage) {
Image image = new Image(IMAGE_LOC);
ImageView imageView = new ImageView(
image
);
imageView.setViewport(
new Rectangle2D(500, 320, 420, 300)
);
imageView.setFitWidth(IMAGE_WIDTH);
imageView.setPreserveRatio(true);
Label labeledImage = createCaptionedImage(
imageView,
"Village Home"
);
addGlowOnMouseOver(labeledImage);
labeledImage.setOnMouseClicked(event -> {
displayFullImage(stage, image);
});
VBox vbox = new VBox( // vbox just there to mimic question askers structure.
labeledImage
);
vbox.setPadding(new Insets(10));
ScrollPane scrollPane = makeScrollable(vbox);
Scene scene = new Scene(
scrollPane
);
stage.setScene(scene);
stage.show();
stage.setMaxWidth(stage.getWidth());
stage.setMaxHeight(stage.getHeight());
}
private void displayFullImage(Stage stage, Image image) {
Stage displayStage = new Stage();
displayStage.initStyle(StageStyle.UTILITY);
displayStage.initModality(Modality.APPLICATION_MODAL);
displayStage.initOwner(stage);
displayStage.setScene(
new Scene(
new Group(
new ImageView(
image
)
)
)
);
displayStage.show();
}
private void addGlowOnMouseOver(Node node) {
Glow glow = new Glow();
DropShadow shadow = new DropShadow(20, Color.GOLD);
glow.setInput(shadow);
node.setOnMousePressed(event -> node.setEffect(null));
node.setOnMouseEntered(event -> node.setEffect(glow));
node.setOnMouseExited(event -> node.setEffect(null));
}
private ScrollPane makeScrollable(Node node) {
ScrollPane scrollPane = new ScrollPane(node);
scrollPane.setHbarPolicy(ScrollPane.ScrollBarPolicy.NEVER);
scrollPane.setVbarPolicy(ScrollPane.ScrollBarPolicy.NEVER);
scrollPane.setPannable(true);
return scrollPane;
}
private Label createCaptionedImage(ImageView imageView, String caption) {
Label labeledImage = new Label(caption);
labeledImage.setFont(Font.font("Athelas", FontPosture.ITALIC, 20));
labeledImage.setStyle("-fx-background-color: cornsilk");
labeledImage.setPadding(new Insets(0, 0, 5, 0));
labeledImage.setGraphic(
imageView
);
labeledImage.setContentDisplay(ContentDisplay.TOP);
return labeledImage;
}
public static void main(String[] args) {
launch(args);
}
private static final String IMAGE_LOC =
"http://www.imgion.com/images/01/beautiful-village-home.jpg";
// image courtesy of http://www.imgion.com which provides
// "free images on large topics to share with your friends and on your blogs."
}
Комментарии:
1. Спасибо! Это работает;), просто вопрос, в вашем коде вы используете строки, подобные этой «.setOnMouseExited(событие -> узел.setEffect(null));». Это новый способ добавления обработчика событий? (Eclipse показывает ошибку), или это было просто для экономии времени?
2. Это лямбда-выражение Java 8 .