异步计算字体大小

时间:2012-06-01 16:17:29

标签: wpf fonts textblock

我使用WPF。我的应用程序有一些文本块,里面有可更改的文本。每个宽度为200,高度为150.问题是我有7个像这样的文本块,我希望它们具有相同的字体大小。该文本必须自动更新。我知道可以自动调整它们。但是当一个句子里面有一个句子而另一个只有两个单词时,字体大小是如此不同......我需要异步重新计算大小(例如创建一些像OnTextChange这样的事件)。块内的文本动态变化。怎么写函数?我想传递3个参数:文本,字体(字体系列+字体样式)和文本块大小,并返回适合的字体大小。

2 个答案:

答案 0 :(得分:2)

确定适当字体大小的最佳方法是以任意大小测量文本,然后将其乘以其大小与区域大小的比率。

例如,如果您测量文本并且它是容器大小的一半,则可以将其乘以2并填充容器。您想要选择要使用的宽度或高度比的最小值。

在WPF中,FormattedText类进行文本测量。

public double GetFontSize(string text, Size availableSize, Typeface typeFace)
{
    FormattedText formtxt = new FormattedText(text, CultureInfo.CurrentCulture, FlowDirection.LeftToRight, typeFace, 10, Brushes.Black);

    double ratio = Math.Min(availableSize.Width / formtxt.Width, availableSize.Height / formtxt.Height);

    return 10 * ratio;
}

每当您更改TextBlocks的文本时都会使用此函数,如下所示:

txtBlock.FontSize = GetFontSize(txt.Text, new Size(txt.ActualWidth, txt.ActualHeight), new Typeface(txt.FontFamily, txt.FontStyle, txt.FontWeight, txt.FontStretch));

修改

出于实用的目的,您可能希望能够使文本在此预定义的边界矩形中垂直居中。一个好方法是将TextBlock包装在另一个对象(如Border元素)中。这样,您可以告诉TextBlock在边框的中心对齐,并且可以自动调整大小以适合其内容。

答案 1 :(得分:0)

确定。工作良好。但我还有另外一个问题。我在MainWindow.cs中写了两个方法:

    private void fitFontSize()
    {
        propertiesList.Clear();

        TextUtils.FontProperty fontProps = new TextUtils.FontProperty();
        foreach (TextBlock tb in findVisualChildren<TextBlock>(statusOptionsGrid))
        {
            fontProps.Text = tb.Text;
            fontProps.Size = new Size(tb.ActualWidth, tb.ActualHeight);
            fontProps.FontFamily = tb.FontFamily;
            fontProps.FontStyle = tb.FontStyle;
            fontProps.FontWeight = tb.FontWeight;
            fontProps.FontStretch = tb.FontStretch;
            propertiesList.Add(fontProps);
        }

        MessageBox.Show(TextUtils.recalculateFontSize(propertiesList) + "");
    }

    public IEnumerable<T> findVisualChildren<T>(DependencyObject depObj) where T : DependencyObject
    {
        if (depObj != null)
        {
            for (int i = 0; i < VisualTreeHelper.GetChildrenCount(depObj); i++)
            {
                DependencyObject child = VisualTreeHelper.GetChild(depObj, i);
                if (child != null && child is T)
                {
                    yield return (T)child;
                }

                foreach (T childOfChild in findVisualChildren<T>(child))
                {
                    yield return childOfChild;
                }
            }
        }
    }

之后我创建了一个新的文本处理类。这是代码:

public class TextUtils
{
    public class FontProperty
    {
        public FontFamily FontFamily { get; set; }
        public FontStyle FontStyle { get; set; }
        public FontWeight FontWeight { get; set; }
        public FontStretch FontStretch { get; set; }
        public string Text { get; set; }
        public Size Size { get; set; }

        public Typeface getTypeFace()
        {
            return new Typeface(FontFamily, FontStyle, FontWeight, FontStretch);
        }
    }

    public static double recalculateFontSize(List<FontProperty> propertiesList)
    {
        List<double> fontSizes = new List<double>();
        foreach (FontProperty fp in propertiesList)
        {
            fontSizes.Add(getFontSizeForControl(fp));
        }

        return fontSizes.Min<double>();
    }

    private static double getFontSizeForControl(FontProperty fp)
    {
        string text = fp.Text;
        Size availableSize = fp.Size;
        Typeface typeFace = fp.getTypeFace();

        FormattedText formtxt = new FormattedText(text, CultureInfo.CurrentCulture, FlowDirection.LeftToRight, typeFace, 10, Brushes.Black);
        double ratio = Math.Min(availableSize.Width / formtxt.Width, availableSize.Height / formtxt.Height);

        return 10 * ratio;
    }
}

代码看起来不好但我稍后会更正......

确定。现在我想创建一个新的计时器,每秒检查字体大小。当我尝试在其中使用fitFontSize()方法时,我得到了msg:“调用线程无法访问此对象,因为不同的线程拥有它。” 如何做到这一点,以避免这样的问题? 我尝试(只是尝试)创建调用此方法的新线程。但是findVisualChildren&lt;&gt;()方法存在同样的问题 - 它在fitFontSize()中调用。我没有任何想法如何解决我的问题...