C#利用GDI+绘制旋转文字等效果实例

本文实例讲述了C#利用GDI+绘制旋转文字等效果的方法,是非常实用的技巧。分享给大家供大家参考之用。具体如下:

C#中利用GDI+绘制旋转文本的文字,网上有很多资料,基本都使用矩阵旋转的方式实现。但基本都只提及按点旋转,若要实现在矩形范围内旋转文本,资料较少。经过琢磨,可以将矩形内旋转转化为按点旋转,不过需要经过不少的计算过程。利用下面的类可以实现该功能。

具体实现代码如下:

using System;
using System.Collections.Generic;
using System.Drawing;
using System.Drawing.Drawing2D;
namespace RotateText
{
  public class GraphicsText
  {
    private Graphics _graphics;
    public GraphicsText()
    {
    }
    public Graphics Graphics
    {
      get { return _graphics; }
      set { _graphics = value; }
    }
    /// <summary>
    /// 绘制根据矩形旋转文本
    /// </summary>
    /// <param name="s">文本</param>
    /// <param name="font">字体</param>
    /// <param name="brush">填充</param>
    /// <param name="layoutRectangle">局部矩形</param>
    /// <param name="format">布局方式</param>
    /// <param name="angle">角度</param>
    public void DrawString(string s, Font font, Brush brush, RectangleF layoutRectangle, StringFormat format, float angle)
    {
      // 求取字符串大小
      SizeF size = _graphics.MeasureString(s, font);
      // 根据旋转角度,求取旋转后字符串大小
      SizeF sizeRotate = ConvertSize(size, angle);
      // 根据旋转后尺寸、布局矩形、布局方式计算文本旋转点
      PointF rotatePt = GetRotatePoint(sizeRotate, layoutRectangle, format);
      // 重设布局方式都为Center
      StringFormat newFormat = new StringFormat(format);
      newFormat.Alignment = StringAlignment.Center;
      newFormat.LineAlignment = StringAlignment.Center;
      // 绘制旋转后文本
      DrawString(s, font, brush, rotatePt, newFormat, angle);
    }
    /// <summary>
    /// 绘制根据点旋转文本,一般旋转点给定位文本包围盒中心点
    /// </summary>
    /// <param name="s">文本</param>
    /// <param name="font">字体</param>
    /// <param name="brush">填充</param>
    /// <param name="point">旋转点</param>
    /// <param name="format">布局方式</param>
    /// <param name="angle">角度</param>
    public void DrawString(string s, Font font, Brush brush, PointF point, StringFormat format, float angle)
    {
      // Save the matrix
      Matrix mtxSave = _graphics.Transform;
      Matrix mtxRotate = _graphics.Transform;
      mtxRotate.RotateAt(angle, point);
      _graphics.Transform = mtxRotate;
      _graphics.DrawString(s, font, brush, point, format);
      // Reset the matrix
      _graphics.Transform = mtxSave;
    }
    private SizeF ConvertSize(SizeF size, float angle)
    {
      Matrix matrix = new Matrix();
      matrix.Rotate(angle);
      // 旋转矩形四个顶点
      PointF[] pts = new PointF[4];
      pts[0].X = -size.Width / 2f;
      pts[0].Y = -size.Height / 2f;
      pts[1].X = -size.Width / 2f;
      pts[1].Y = size.Height / 2f;
      pts[2].X = size.Width / 2f;
      pts[2].Y = size.Height / 2f;
      pts[3].X = size.Width / 2f;
      pts[3].Y = -size.Height / 2f;
      matrix.TransformPoints(pts);
      // 求取四个顶点的包围盒
      float left = float.MaxValue;
      float right = float.MinValue;
      float top = float.MaxValue;
      float bottom = float.MinValue;
      foreach(PointF pt in pts)
      {
        // 求取并集
        if(pt.X < left)
          left = pt.X;
        if(pt.X > right)
          right = pt.X;
        if(pt.Y < top)
          top = pt.Y;
        if(pt.Y > bottom)
          bottom = pt.Y;
      }
      SizeF result = new SizeF(right - left, bottom - top);
      return result;
    }
    private PointF GetRotatePoint(SizeF size, RectangleF layoutRectangle, StringFormat format)
    {
      PointF pt = new PointF();
      switch (format.Alignment)
      {
        case StringAlignment.Near:
          pt.X = layoutRectangle.Left + size.Width / 2f;
          break;
        case StringAlignment.Center:
          pt.X = (layoutRectangle.Left + layoutRectangle.Right) / 2f;
          break;
        case StringAlignment.Far:
          pt.X = layoutRectangle.Right - size.Width / 2f;
          break;
        default:
          break;
      }
      switch (format.LineAlignment)
      {
        case StringAlignment.Near:
          pt.Y = layoutRectangle.Top + size.Height / 2f;
          break;
        case StringAlignment.Center:
          pt.Y = (layoutRectangle.Top + layoutRectangle.Bottom) / 2f;
          break;
        case StringAlignment.Far:
          pt.Y = layoutRectangle.Bottom - size.Height / 2f;
          break;
        default:
          break;
      }
      return pt;
    }
  }
}

测试代码如下:

using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Data;
using System.Drawing;
using System.Windows.Forms;
namespace RotateText
{
  public partial class FormMain : Form
  {
    private Font _font = new Font("Arial", 12);
    private Brush _brush = new SolidBrush(Color.Black);
    private Pen _pen = new Pen(Color.Black, 1f);
    private string _text = "Crow Soft";
    public FormMain()
    {
      InitializeComponent();
    }
    protected override void OnPaint(PaintEventArgs e)
    {
      base.OnPaint(e);
      GraphicsText graphicsText = new GraphicsText();
      graphicsText.Graphics = e.Graphics;
      // 绘制围绕点旋转的文本
      StringFormat format = new StringFormat();
      format.Alignment = StringAlignment.Center;
      format.LineAlignment = StringAlignment.Center;
      graphicsText.DrawString(_text, _font, _brush, new PointF(100, 80), format, 45f);
      graphicsText.DrawString(_text, _font, _brush, new PointF(200, 80), format, -45f);
      graphicsText.DrawString(_text, _font, _brush, new PointF(300, 80), format, 90f);
      graphicsText.DrawString(_text, _font, _brush, new PointF(400, 80), format, -60f);
      // 绘制矩形内旋转的文本
      // First line
      RectangleF rc = RectangleF.FromLTRB(50, 150, 200, 230);
      RectangleF rect = rc;
      format.Alignment = StringAlignment.Near;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, 30);
      rect.Location += new SizeF(180, 0);
      format.LineAlignment = StringAlignment.Near;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, -30);
      rect.Location += new SizeF(180, 0);
      format.LineAlignment = StringAlignment.Center;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, -90);
      rect.Location += new SizeF(180, 0);
      format.LineAlignment = StringAlignment.Far;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, 70);
      // Second line
      rect = rc;
      rect.Location += new SizeF(0, 100);
      format.Alignment = StringAlignment.Center;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, 40);
      rect.Location += new SizeF(180, 0);
      format.LineAlignment = StringAlignment.Near;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, 30);
      rect.Location += new SizeF(180, 0);
      format.LineAlignment = StringAlignment.Center;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, -70);
      rect.Location += new SizeF(180, 0);
      format.LineAlignment = StringAlignment.Far;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, 60);
      // Third line
      rect = rc;
      rect.Location += new SizeF(0, 200);
      format.Alignment = StringAlignment.Far;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, -30);
      rect.Location += new SizeF(180, 0);
      format.LineAlignment = StringAlignment.Near;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, -30);
      rect.Location += new SizeF(180, 0);
      format.LineAlignment = StringAlignment.Center;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, 90);
      rect.Location += new SizeF(180, 0);
      format.LineAlignment = StringAlignment.Far;
      e.Graphics.DrawRectangle(_pen, rect.Left, rect.Top, rect.Width, rect.Height);
      graphicsText.DrawString(_text, _font, _brush, rect, format, 45);
    }
  }
}

效果如下图:

完整实例点此

希望本文所述对大家的C#程序设计有所帮助

除声明外,跑步客文章均为原创,转载请以链接形式标明本文地址
  C#利用GDI+绘制旋转文字等效果实例

本文地址:  http://www.paobuke.com/develop/c-develop/pbk23483.html

相关内容

C#实现Access通用访问类OleDbHelper完整实例

winform dateTime数据类型转换方法

C# SQlite操作方法小结

C#实现从多列的DataTable里取需要的几列


基于C#生成条形码操作知识汇总附源码下载

12306奇葩验证码引发思考之C#实现验证码程序

C#将隐私信息(银行账户,身份证号码)中间部分特殊字符替换成*

C#基础语法:Base关键字学习笔记

时间: 2024-10-17 11:30:14

C#利用GDI+绘制旋转文字等效果实例的相关文章

C#利用GDI+绘制旋转文字等效果

C#中利用GDI+绘制旋转文本的文字,网上有很多资料,基本都使用矩阵旋转的方式实现.但基本都只提及按点旋转,若要实现在矩形范围内旋转文本,资料较少.经过琢磨,可以将矩形内旋转转化为按点旋转,不过需要经过不少的计算过程.利用下面的类可以实现该功能. [csharp] view plaincopy using System; using System.Collections.Generic; using System.Drawing; using System.Drawing.Drawing2D;

Flash 开发环境搭建和文字滚动效果实例

一.Flash 开发环境搭建 Flash发布的时候可以将资源(即将库中的元件)集成到swf运行文件中.Flash没有代码自动输入补全功能,因此需要一个英文一个英文手工输入,这样要记住很多系统类和方法.而FlashDevelop由代码自动输入补全功能,防止输入过程中出现错误,并提高代码编写效率,使我们将注意力集中到逻辑代码编写中. 1.安装Flash CS3或CS4.CS5. 我安装的是Flash CS4 下载Flash CS4.点击下一步.下一步就可以. 2.安装FlashDevelop, 安装

利用PorterDuffXfermode绘制图片文字

PorterDuffXfermode是Android中用来对图层进行操作的类,类似于数学中的交集.并集,将上层(src)和下层(dst)进行特定的方式进行混合显示. 构造方法:PorterDuffXfermode(PorterDuff.Mode mode) 下图显示对应的PorterDuff.Mode所对应的效果 这次实现的图片文字是用的SrcIn模式,也就是先画文字然后画图,取其交集区域显示上层图层 完整代码展示 package com.yuyigufen.customview; import

利用GDI+制作Flappy Bird

上次介绍用GDI+写了个验证码图片生成器,这次再来介绍下用GDI+写之前流行过一段时间的小游戏:Flappy Bird.通过写这个游戏再来熟悉下GDI+的一些简单利用. 这是一个粗糙的游戏画面,大家不要介意啊,毕竟这是美工做的事: 先来分析一下这个游戏要怎么写.游戏过程是:1.小鸟不停的往下掉,而且越掉越快:2.障碍物柱子不停地出现并往左移动:3.游戏一开始下面的前进条就不停地转动. 游戏规则:1.小鸟的身体不能触碰障碍物:2.小鸟的身体不能触及底部及上部:3.每当小鸟穿过一个障碍物时统计通过障

D2D引擎与GDI\GDI+绘制效果对比

本例主要是对比D2D和GDI在绘制文字.线条的区别,以及D2D与GDI+在绘制图片时的区别. D2D是基于COM组件开发的,使用前的CoInitialize(NULL)是必须的:另外,GDI+的初始化GdiplusStartup()也别忘了. 废话少说,完整代码如下: // D2DDemo.cpp : 定义应用程序的入口点. // #include "stdafx.h" #include "D2DDemo.h" #include <D2D1.h> #in

利用线性渐变阴影旋转实现晴天效果

1 <!DOCTYPE html> 2 <html lang="en"> 3 <head> 4 <meta charset="utf-8"> 5 <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1.0,minimum-scale=1.0,user-scalable=no

CSS3 利用 text-shadow 实现文字描边效果

实现效果: 效果代码: <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title></title> </head> <style> h2{color:#fff;text-shadow:1px 0px 0px pink,-1px 0px 0px pink,0px 1px 0px pink,0p

利用OpenCV实现旋转文本图像矫正的原理及OpenCV代码

对图像进行旋转矫正,关键是要获取旋转角度是多少!获取了旋转角度就可以用仿射变换对图像进行矫正,图像旋转的代码可以参考我的博文http://blog.csdn.net/wenhao_ir/article/details/51469085 旋转角度怎么获取?可以对图像作傅里叶变换获取这个角度,具体怎么求,请听我慢慢道来! 文本图像的明显特征就是存在分行间隔,那么行与文字之间这个灰度值变化就不如真正的文字及文字间的变化剧烈,那么相应的这些地方的频谱值也低,即频谱的低谱部分,因为傅里叶变换就是表征图像各

学习笔记:利用GDI+生成简单的验证码图片

小分享:我有几张阿里云优惠券,用券购买或者升级阿里云相应产品最多可以优惠五折!领券地址:https://promotion.aliyun.com/ntms/act/ambassador/sharetouser.html?userCode=ohmepe03 学习笔记:利用GDI+生成简单的验证码图片 1 /// <summary> 2 /// 单击图片时切换图片 3 /// </summary> 4 /// <param name="sender">&