我有一个多行标签,其文本有可能溢出。如果它这样做,我想减小字体大小,直到它没有溢出,或者直到它达到某个最小大小。这有望使标签改变大小,直到整个字符串可见。
我的问题是我不确定如何测试文本是否已溢出。我尝试测试标签的文本是否以椭圆字符串结尾,但我相信椭圆在技术上并未添加到标签的 textProperty 中。那么有人知道对此进行测试的好方法吗?
简短而令人失望的答案:您根本无法以可靠的方式做到这一点。
稍长一点的答案是,标签本身甚至不知道它是否溢出。每当调整标签大小时,皮肤类 ( LabeledSkinBase
) 负责更新显示的文本。然而,这个类使用 JavaFX utils 类来计算椭圆文本。这里的问题是,如果标签的尺寸需要,相应的方法只返回一个椭圆形的字符串。皮肤本身永远不会被告知文本是否实际上是椭圆形的,它只是将显示的文本更新为返回的结果。
您可以尝试检查皮肤类的显示文本,但它是受保护的。所以你需要做的是子类化LabelSkin
,并实现类似的东西:
package com.sun.javafx.scene.control.skin;
import java.lang.reflect.Field;
import javafx.scene.control.Label;
@SuppressWarnings("restriction")
public class TestLabel extends LabelSkin {
private LabeledText labelledText;
public TestLabel(Label label) throws Exception {
super(label);
for (Field field : LabeledSkinBase.class.getDeclaredFields()) {
if (field.getName().equals("text")) {
field.setAccessible(true);
labelledText = (LabeledText) field.get(this);
break;
}
}
}
public boolean isEllipsoided() {
return labelledText != null && labelledText.getText() != null && !getSkinnable().getText().equals(labelledText.getText());
}
}
如果你使用这个皮肤为你的Label,你应该能够检测到你的文本是否是椭圆的。如果您对循环和反射感到疑惑:Java 不允许我通过其他方式访问文本字段,因此这可能是您真的不应该这样做的一个强有力的指标;-) 不过:它有效!
免责声明:我只检查了 JavaFX 8
minisu 在这个答案中发布了一种检测溢出的方法: https ://stackoverflow.com/a/15178908/9492864
这种方式适用于所有标签,我在带有 JavaFX 8 的按钮上对其进行了测试。例如,您可以将侦听器添加到 needsLayoutProperty:
labeled.needsLayoutProperty().addListener((observable, oldValue, newValue) -> {
String originalString = labeled.getText();
Text textNode = (Text) labeled.lookup(".text"); // "text" is the style class of Text
String actualString = textNode.getText();
boolean clipped = !actualString.isEmpty() && !originalString.equals(actualString);
System.out.println("is " + originalString + " clipped: " + clipped);
});
这对我有用!在 javafx 中
public class LabelHelper{
public static boolean necesitaTooltip(Font font, Label label){
Text text = new Text(label.getText());
text.setFont(font);
Bounds tb = text.getBoundsInLocal();
Rectangle stencil = new Rectangle(
tb.getMinX(), tb.getMinY(), tb.getWidth(), tb.getHeight()
);
Shape intersection = Shape.intersect(text, stencil);
Bounds ib = intersection.getBoundsInLocal();
return ib.getWidth() > label.getPrefWidth();
}
public static void asignarTexto(Label label, String texto){
label.setText(texto);
if (necesitaTooltip(label.getFont(), label)){
Tooltip tp = new Tooltip(texto);
label.setTooltip(tp);
}
}
}
仅调用 asignarTexto(label, texto) 将文本设置为标签并检查标签中的文本是否超出,然后为标签添加工具提示。
Johann 已经提到,您可以使用它(Text)labeled.lookup(".text")
来获取标签的实际显示文本,然后将其与预期的字符串进行比较......但是,在我的情况下,这不起作用。也许是因为我更新标签的频率很高,但实际的字符串总是比预期的少几个字符......
因此,我选择使用该setEllipsisString(String value)
方法将省略号字符串(当出现溢出时附加到标签末尾的内容,默认为“...”)设置为(未使用的)
ASCII控制字符,例如0x03
(适当地命名为“结束text"),然后每次设置标签文本后,我都会检查实际字符串的最后一个字符是否是控制字符。
使用示例Platform.runLater(Runnable)
:
import javafx.scene.control.Label;
import javafx.application.Platform;
import javafx.scene.text.Text;
...
Label label = new Label();
...
label.setEllipsisString("\003");
...
final String newText = "fef foo";
Platform.runLater(() -> {
label.setText(newText);
String actual = ((Text)label.lookup(".text")).getText();
// \003 is octal for the aforementioned "end of text" control char
if (actual.length() > 0 && actual.charAt(actual.length()-1) == '\003') {
// Handle text now that you know it's clipped
}
});
请注意,您可以将控制字符设置为真正的任何内容,并且不必只是一个字符;但是,如果您选择控制字符,请检查它是否不常用。