多年来,我多次遇到过调整文本大小以适应Java GUI上某个区域的问题。我的解决方案通常是通过以下方式解决问题:
重新设计界面以避免问题
更改区域以适合文本的大小
二进制搜索正确大小的字体以适应字符串(当我无法执行前两个字符时)
在最近处理需要快速确定给定区域的正确字体大小的另一个项目时,我的二进制搜索方法太慢了(我怀疑是因为动态内存分配涉及多次按顺序创建和测量字体)并在我的应用程序中引入了明显的延迟。我需要的是一种更快速的计算字体大小的方法,允许渲染给定字符串以适应GUI的定义区域。
答案 0 :(得分:0)
最后我发现有一种更简单,更快捷的方式,只需要在运行时进行一些分配。这种新方法消除了对任何类型搜索的需要,并且仅需要进行1次测量,但是,它确实需要做出一个假设,一个对于大多数应用来说是完全合理的。
使用这个假设,我们可以计算字体尺寸与点大小的比率,并线性推断以找到给定区域所需的字体大小。我写的一些代码如下:
修改: 初始测量的准确性受基本字体大小的限制。使用非常小的字体大小作为基础可以甩掉结果。但是基本字体的大小越大,线性近似越精确越准确。
import java.awt.Font;
import java.awt.FontMetrics;
import java.awt.Graphics;
import java.awt.Shape;
import java.awt.font.GlyphVector;
import java.awt.geom.Rectangle2D;
public class FontUtilities
{
public static Font createFontToFit
(
String value,
double width,
double height,
Font base,
Graphics context
)
{
double measuredWidth;
double measuredHeight;
double baseFontSize;
FontMetrics ruler;
Rectangle2D bounds;
double heightBasedFontSize;
double widthBasedFontSize;
GlyphVector vector;
Shape outline;
if
(
(value == null) ||
(base == null) ||
(context == null) ||
(width != width) ||
(height != height)
)
{
return null;
}
//measure the size of the string in the current font size
baseFontSize = base.getSize2D();
ruler = context.getFontMetrics(base);
vector = base.createGlyphVector(ruler.getFontRenderContext(), value);
//use the bounds measurement on the outline of the text since this is the only
//measurement method that seems to be bug free and consistent in java
outline = vector.getOutline(0, 0);
bounds = outline.getBounds();
measuredWidth = bounds.getWidth();
measuredHeight = bounds.getHeight();
//assume that each of the width and the height of the string
//is proportional to the font size, calculate the ratio
//and extrapolate linearly to determine the needed font size.
//should have 2 font sizes one for matching the width, and one for
//matching the height, return the least of the 2
widthBasedFontSize = (baseFontSize*width)/measuredWidth;
heightBasedFontSize = (baseFontSize*height)/measuredHeight;
if(widthBasedFontSize < heightBasedFontSize)
{
return base.deriveFont(base.getStyle(), (float)widthBasedFontSize);
}
else
{
return base.deriveFont(base.getStyle(), (float)heightBasedFontSize);
}
}
}