2013-04-30 61 views
4

如何更改richtextbox的所有内容的字体而不会丢失格式?更改richtextbox的字体而不会丢失格式化

我想使用

rtb.SelectAll(); 
rtb.SelectionFont = new Font(fontName,...); 

但字体构造必须采取除了字体类型无论是字体样式(粗体,斜体,...)或字体大小。 因此,使用这将改变richtextbox的所有内容的风格/大小。

当然这同样适用于richtextbox中的任何选择。

+1

见(http://stackoverflow.com/q/5325918/719186)。最终,禁令可能是最好的解决方案。 – LarsTech 2013-04-30 18:21:49

+0

你可以在pinvoking更清晰? – Jerry 2013-04-30 18:27:20

回答

6

这是我以前用过的RichTextBox。从这儿发现的堆栈溢出代码和互联网拼接在一起,在大:

public class RichBox : RichTextBox { 
    private const UInt32 CFM_BOLD = 0x00000001; 
    private const UInt32 CFM_ITALIC = 0x00000002; 
    private const UInt32 CFM_UNDERLINE = 0x00000004; 
    private const UInt32 CFM_STRIKE = 0x00000008; 
    private const UInt32 CFM_FACE = 0x20000000; 
    private const UInt32 CFM_SIZE = 0x80000000; 

    private const int WM_PAINT = 0xF; 
    private const int WM_SETREDRAW = 0xB; 
    private const int WM_USER = 0x400; 

    private const int EM_SETCHARFORMAT = (WM_USER + 68); 
    private const int SCF_SELECTION = 0x0001; 
    private const int EM_GETEVENTMASK = WM_USER + 59; 
    private const int EM_SETEVENTMASK = WM_USER + 69; 
    private const int EM_GETSCROLLPOS = WM_USER + 221; 
    private const int EM_SETSCROLLPOS = WM_USER + 222; 

    [StructLayout(LayoutKind.Sequential)] 
    private struct CHARFORMAT { 
    public int cbSize; 
    public uint dwMask; 
    public uint dwEffects; 
    public int yHeight; 
    public int yOffset; 
    public int crTextColor; 
    public byte bCharSet; 
    public byte bPitchAndFamily; 
    [MarshalAs(UnmanagedType.ByValArray, SizeConst = 32)] 
    public char[] szFaceName; 
    public short wWeight; 
    public short sSpacing; 
    public int crBackColor; 
    public int LCID; 
    public uint dwReserved; 
    public short sStyle; 
    public short wKerning; 
    public byte bUnderlineType; 
    public byte bAnimation; 
    public byte bRevAuthor; 
    } 

    [DllImport("kernel32.dll", CharSet = CharSet.Auto)] 
    private static extern IntPtr LoadLibrary(string lpFileName); 

    [DllImport("user32", CharSet = CharSet.Auto)] 
    private static extern int SendMessage(IntPtr hWnd, int msg, int wParam, ref CHARFORMAT lParam); 

    [DllImport("user32.dll")] 
    private static extern IntPtr SendMessage(IntPtr hWnd, Int32 wMsg, Int32 wParam, ref Point lParam); 

    [DllImport("user32.dll")] 
    private static extern IntPtr SendMessage(IntPtr hWnd, Int32 wMsg, Int32 wParam, IntPtr lParam); 

    private bool frozen = false; 
    private Point lastScroll = Point.Empty; 
    private IntPtr lastEvent = IntPtr.Zero; 
    private int lastIndex = 0; 
    private int lastWidth = 0; 

    protected override CreateParams CreateParams { 
    get { 
     var cp = base.CreateParams; 
     if (LoadLibrary("msftedit.dll") != IntPtr.Zero) { 
     cp.ClassName = "RICHEDIT50W"; 
     } 
     return cp; 
    } 
    } 

    [Browsable(false)] 
    [DefaultValue(typeof(bool), "False")] 
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
    public bool FreezeDrawing { 
    get { return frozen; } 
    set { 
     if (value != frozen) { 
     frozen = value; 
     if (frozen) { 
      this.SuspendLayout(); 
      SendMessage(this.Handle, WM_SETREDRAW, 0, IntPtr.Zero); 
      SendMessage(this.Handle, EM_GETSCROLLPOS, 0, ref lastScroll); 
      lastEvent = SendMessage(this.Handle, EM_GETEVENTMASK, 0, IntPtr.Zero); 
      lastIndex = this.SelectionStart; 
      lastWidth = this.SelectionLength; 
     } else { 
      this.Select(lastIndex, lastWidth); 
      SendMessage(this.Handle, EM_SETEVENTMASK, 0, lastEvent); 
      SendMessage(this.Handle, EM_SETSCROLLPOS, 0, ref lastScroll); 
      SendMessage(this.Handle, WM_SETREDRAW, 1, IntPtr.Zero); 
      this.Invalidate(); 
      this.ResumeLayout(); 
     } 
     } 
    } 
    } 

    [Browsable(false)] 
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
    public Font CurrentFont { 
    get { 
     Font result = this.Font; 
     if (this.SelectionLength == 0) { 
     result = SelectionFont; 
     } else { 
     using (RichBox rb = new RichBox()) { 
      rb.FreezeDrawing = true; 
      rb.SelectAll(); 
      rb.SelectedRtf = this.SelectedRtf; 
      rb.Select(0, 1); 
      result = rb.SelectionFont; 
     } 
     } 
     return result; 
    } 
    } 

    [Browsable(false)] 
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
    public string SelectionFontName { 
    get { return CurrentFont.FontFamily.Name; } 
    set { 
     CHARFORMAT cf = new CHARFORMAT(); 
     cf.cbSize = Marshal.SizeOf(cf); 
     cf.szFaceName = new char[32]; 
     cf.dwMask = CFM_FACE; 
     value.CopyTo(0, cf.szFaceName, 0, Math.Min(31, value.Length)); 
     IntPtr lParam = Marshal.AllocCoTaskMem(Marshal.SizeOf(cf)); 
     Marshal.StructureToPtr(cf, lParam, false); 
     SendMessage(this.Handle, EM_SETCHARFORMAT, SCF_SELECTION, lParam); 
    } 
    } 

    [Browsable(false)] 
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
    public float SelectionFontSize { 
    get { return CurrentFont.Size; } 
    set { 
     CHARFORMAT cf = new CHARFORMAT(); 
     cf.cbSize = Marshal.SizeOf(cf); 
     cf.dwMask = CFM_SIZE; 
     cf.yHeight = Convert.ToInt32(value * 20); 
     IntPtr lParam = Marshal.AllocCoTaskMem(Marshal.SizeOf(cf)); 
     Marshal.StructureToPtr(cf, lParam, false); 
     SendMessage(this.Handle, EM_SETCHARFORMAT, SCF_SELECTION, lParam); 
    } 
    } 

    [Browsable(false)] 
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
    public bool SelectionBold { 
    get { return CurrentFont.Bold; } 
    set { 
     CHARFORMAT cf = new CHARFORMAT(); 
     cf.cbSize = Marshal.SizeOf(cf); 
     cf.dwMask = CFM_BOLD; 
     cf.dwEffects = value ? CFM_BOLD : 0; 
     IntPtr lParam = Marshal.AllocCoTaskMem(Marshal.SizeOf(cf)); 
     Marshal.StructureToPtr(cf, lParam, false); 
     SendMessage(this.Handle, EM_SETCHARFORMAT, SCF_SELECTION, lParam); 
    } 
    } 

    [Browsable(false)] 
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
    public bool SelectionItalic { 
    get { return CurrentFont.Italic; } 
    set { 
     CHARFORMAT cf = new CHARFORMAT(); 
     cf.cbSize = Marshal.SizeOf(cf); 
     cf.dwMask = CFM_ITALIC; 
     cf.dwEffects = value ? CFM_ITALIC : 0; 
     IntPtr lParam = Marshal.AllocCoTaskMem(Marshal.SizeOf(cf)); 
     Marshal.StructureToPtr(cf, lParam, false); 
     SendMessage(this.Handle, EM_SETCHARFORMAT, SCF_SELECTION, lParam); 
    } 
    } 

    [Browsable(false)] 
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
    public bool SelectionStrikeout { 
    get { return CurrentFont.Strikeout; } 
    set { 
     CHARFORMAT cf = new CHARFORMAT(); 
     cf.cbSize = Marshal.SizeOf(cf); 
     cf.dwMask = CFM_STRIKE; 
     cf.dwEffects = value ? CFM_STRIKE : 0; 
     IntPtr lParam = Marshal.AllocCoTaskMem(Marshal.SizeOf(cf)); 
     Marshal.StructureToPtr(cf, lParam, false); 
     SendMessage(this.Handle, EM_SETCHARFORMAT, SCF_SELECTION, lParam); 
    } 
    } 

    [Browsable(false)] 
    [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)] 
    public bool SelectionUnderline { 
    get { return CurrentFont.Underline; } 
    set { 
     CHARFORMAT cf = new CHARFORMAT(); 
     cf.cbSize = Marshal.SizeOf(cf); 
     cf.dwMask = CFM_UNDERLINE; 
     cf.dwEffects = value ? CFM_UNDERLINE : 0; 
     IntPtr lParam = Marshal.AllocCoTaskMem(Marshal.SizeOf(cf)); 
     Marshal.StructureToPtr(cf, lParam, false); 
     SendMessage(this.Handle, EM_SETCHARFORMAT, SCF_SELECTION, lParam); 
    } 
    } 
} 

它增加了新的特性,如SelectionBold,SelectionItalic等在那里你可以应用属性,而不是失去文本的其他格式。

+0

完美!谢谢 – Jerry 2013-04-30 20:20:36

+0

嗨LarsTech,有时当我改变包含粗体的大选择的字体时,粗体会丢失。当我点击ctrl + Z一次时,粗体被恢复。你有一个想法,为什么会发生这种情况?谢谢 – Jerry 2013-06-11 21:49:12

+1

@Jerry如果您要更改'SelectionFont'属性,那么是的,您正在告诉文档将所有字体属性替换为新字体。要更改字体但保留属性,请改为使用“SelectionFontName”或“SelectionFontSize”属性。另外,Ctrl-Z是撤消功能。 – LarsTech 2013-06-17 13:25:24

0

您可以传入所需的任何参数的新值,并使用rtb属性来保留其他值。例如,如果你想改变font family但希望保留font size,这是你会怎么做:

rtb.SelectionFont = new Font(fontName, rtb.Font.Size); 

这将SelectionFont家庭改变fontName但保留的字体大小。对于其他重载,您可以遵循相同的模式。

+1

但是,如果有不同的字体大小,这将无法正常工作。 – Jerry 2013-04-30 18:15:07

+0

你是什么意思不同的字体大小?解释你想要达成的目标,然后或许我们可以搞清楚。 – PoweredByOrange 2013-04-30 18:18:27

+0

具有不同字体大小的不同文本。无论如何,即使你的答案导致字体样式改变。 – Jerry 2013-04-30 18:20:33

2

通过使用现有的richtextbox字体属性,您可以传递新的字体名称,同时保持其他值不变。为了改变选中的文本只字体的名称,你需要做的:

if (rtb.SelectionFont !=null) 
    rtb.SelectionFont = new Font(fontName, rtb.SelectionFont.Size, rtb.SelectionFont.Style); 

注意,上面的代码只会工作,如果所有选定的文本具有相同的格式(字体大小,样式等)。这可以通过首先检查SelectionFont属性来检测,如果选择包含样式组合,则它将为null。

现在要更改richtextbox所有内容的字体名称,同时保持其他格式不变,您需要遍历richtextbox的所有字符并逐个应用字体名称。

for (int i = 0; i < rtb.TextLength; i++) 
{ 
    rtb.Select(i, 1); 
    rtb.SelectionFont = new Font(fontName, rtb.SelectionFont.Size, rtb.SelectionFont.Style); 
} 
0
Private Sub changeFont(ByVal fontz As FontStyle, getStr As RichTextBox) 
    Dim currentFont As System.Drawing.Font = txt.SelectionFont 
    Dim newFontStyle As System.Drawing.FontStyle 
    newFontStyle = fontz 
    getStr.SelectionFont = New Font(currentFont.FontFamily, currentFont.Size, newFontStyle) 
End Sub 

这将改变所选文本的字体属性。
样品:[?我如何保持改变任何一个元素时富文本格式(粗体/斜体/等)

changeFont(FontStyle.Italic, [textbox_name]) 
相关问题