Как я могу сделать изогнутый текст в растровое изображение? - PullRequest
4 голосов
/ 10 мая 2010

В настоящее время я динамически создаю растровое изображение и использую графический объект из растрового изображения, чтобы нарисовать на нем строку следующим образом:

System.Drawing.Graphics graph = System.Drawing.Graphics.FromImage(bmp);
graph.DrawString(text, font, brush, new System.Drawing.Point(0, 0));

Возвращает прямоугольное растровое изображение со строкой, написанной прямо слева направо. Я хотел бы также иметь возможность нарисовать строку в форме радуги. Как я могу это сделать?

Ответы [ 3 ]

17 голосов
/ 22 июня 2012

У меня недавно была эта проблема (я рендерил текст для печати на компакт-дисках), поэтому вот мое решение:

private void DrawCurvedText(Graphics graphics, string text, Point centre, float distanceFromCentreToBaseOfText, float radiansToTextCentre, Font font, Brush brush)
{
    // Circumference for use later
    var circleCircumference = (float)(Math.PI * 2 * distanceFromCentreToBaseOfText);

    // Get the width of each character
    var characterWidths = GetCharacterWidths(graphics, text, font).ToArray();

    // The overall height of the string
    var characterHeight = graphics.MeasureString(text, font).Height;

    var textLength = characterWidths.Sum();

    // The string length above is the arc length we'll use for rendering the string. Work out the starting angle required to 
    // centre the text across the radiansToTextCentre.
    float fractionOfCircumference = textLength / circleCircumference;

    float currentCharacterRadians = radiansToTextCentre + (float)(Math.PI * fractionOfCircumference);

    for (int characterIndex = 0; characterIndex < text.Length; characterIndex++)
    {
        char @char = text[characterIndex];

        // Polar to cartesian
        float x = (float)(distanceFromCentreToBaseOfText * Math.Sin(currentCharacterRadians));
        float y = -(float)(distanceFromCentreToBaseOfText * Math.Cos(currentCharacterRadians));

        using (GraphicsPath characterPath = new GraphicsPath())
        {
            characterPath.AddString(@char.ToString(), font.FontFamily, (int)font.Style, font.Size, Point.Empty,
                                    StringFormat.GenericTypographic);

            var pathBounds = characterPath.GetBounds();

            // Transformation matrix to move the character to the correct location. 
            // Note that all actions on the Matrix class are prepended, so we apply them in reverse.
            var transform = new Matrix();

            // Translate to the final position
            transform.Translate(centre.X + x, centre.Y + y);

            // Rotate the character
            var rotationAngleDegrees = currentCharacterRadians * 180F / (float)Math.PI - 180F;
            transform.Rotate(rotationAngleDegrees);

            // Translate the character so the centre of its base is over the origin
            transform.Translate(-pathBounds.Width / 2F, -characterHeight);

            characterPath.Transform(transform);

            // Draw the character
            graphics.FillPath(brush, characterPath);
        }

        if (characterIndex != text.Length - 1)
        {
            // Move "currentCharacterRadians" on to the next character
            var distanceToNextChar = (characterWidths[characterIndex] + characterWidths[characterIndex + 1]) / 2F;
            float charFractionOfCircumference = distanceToNextChar / circleCircumference;
            currentCharacterRadians -= charFractionOfCircumference * (float)(2F * Math.PI);
        }
    }
}

private IEnumerable<float> GetCharacterWidths(Graphics graphics, string text, Font font)
{
    // The length of a space. Necessary because a space measured using StringFormat.GenericTypographic has no width.
    // We can't use StringFormat.GenericDefault for the characters themselves, as it adds unwanted spacing.
    var spaceLength = graphics.MeasureString(" ", font, Point.Empty, StringFormat.GenericDefault).Width;

    return text.Select(c => c == ' ' ? spaceLength : graphics.MeasureString(c.ToString(), font, Point.Empty, StringFormat.GenericTypographic).Width);
}

screenshot of curved text in program output

1 голос
/ 10 мая 2010

Я думаю, что единственный способ - визуализировать каждого персонажа отдельно и использовать

Graphics.RotateTransform

для поворота текста. Вам нужно будет отработать угол поворота и смещение рендеринга самостоятельно. Вы можете использовать

Graphics.MeasureCharacterRanges

чтобы получить размер каждого символа.

0 голосов
/ 10 мая 2010

К сожалению, в GDI + нет способа прикрепить строки к пути (это то, что вы бы искали).

Так что единственный способ сделать это - сделать это «вручную». Это означает разделение строки на символы и размещение их на основе собственных вычислений пути.

Если вы не хотите вкладывать в это лот работы, вы должны попытаться найти библиотеку (возможно, полную замену GDI +), чтобы сделать это или разочароваться в радуге.

С WPF вы можете рендерить текст по пути ( см. Ссылку с инструкциями )

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...