第五章-文本编程
来源:互联网 发布:java实现多张图片上传 编辑:程序博客网 时间:2024/06/06 01:49
1.CWnd::CreateSolidCaret创建插入符,ShowCaret()显示插入符。GetTextMetrics(),获得当前字体的一些信息。CWnd::CreateCaret()创建图象插入符
bitmap.LoadBitmap(IDB_BITMAP1);//此处的bitmap为成员变量!!!
CreateCaret(&bitmap);
ShowCaret();
TEXTMETRIC tm;//字体结构体
dc.GetTextMetrics(&tm);//
m_ptOrigin.y+=tm.tmHeight;//获得字体高度。
void CreateSolidCaret( int nWidth, int nHeight );//创建插入符
void CreateCaret( CBitmap* pBitmap );//创建位图插入符
void ShowCaret( );//显示插入符
void HideCaret( );//隐藏插入符
static void PASCAL SetCaretPos( POINT point );//移动插入符号
说明:
1)创建插入符要在窗口创建完成之后,CreateSolidCaret函数创建的插入符被初始化为隐藏,所以需要调用ShowCaret()将其显示。
2)使用CreateCaret函数创建位图插入符的时候,不能使用局部的位图对象关联位图资源。(与资源相关联的C++对象,当它析构的时候会同时把与它相关联的资源销毁。)
获取当前字体信息的度量:CDC::GetTextMetrics
BOOL GetTextMetrics( LPTEXTMETRIC lpMetrics ) const;
说明:
typedef struct tagTEXTMETRIC { /* tm */
int tmHeight;//字体高度。Specifies the height (ascent + descent) of characters.
int tmAscent;//基线以上的字体高度
int tmDescent;//基线以下的字体高度
int tmInternalLeading;
int tmExternalLeading;
int tmAveCharWidth;//字符平均宽度
int tmMaxCharWidth;
int tmWeight;
BYTE tmItalic;
BYTE tmUnderlined;
BYTE tmStruckOut;
BYTE tmFirstChar;
BYTE tmLastChar;
BYTE tmDefaultChar;
BYTE tmBreakChar;
BYTE tmPitchAndFamily;
BYTE tmCharSet;
int tmOverhang;
int tmDigitizedAspectX;
int tmDigitizedAspectY;
} TEXTMETRIC;
获取字符串的高度和宽度(区别字符串的长度):
CDC::GetTextExtent
CSize GetTextExtent( LPCTSTR lpszString, int nCount ) const;
CSize GetTextExtent( const CString& str ) const;
说明:
The CSize class is similar to the Windows SIZE structure。
typedef struct tagSIZE {
int cx;//the x-extent
int cy;//the y-extent
} SIZE;
2.VC中CString::LoadString(ID号),比较方便。
3.路径层的概念:有两种方法创建路径层:
(1)
pDC->BeginPath();
pDC->Rectangle(50,50,50+sz.cx,50+sz.cy);
pDC->EndPath();
pDC->SelectClipPath(RGN_DIFF);
(2)
CSize sz=pDC->GetTextExtent(str);
CRgn rn;
rn.CreateRectRgn(0,50,sz.cx,sz.cy);
pDC->SelectClipRgn(&rn,RGN_DIFF);
路径层有什么作用?可以保护我们先前的文本或者图像不被后来画的覆盖。
BOOL BeginPath( );
//在这作图定义路径层剪切区域
BOOL EndPath( );
BOOL SelectClipPath( int nMode );//调用这个函数来使当前路径层剪切区域与新剪切区域进行互操作。
//在这覆盖作图(包含前定义的路径层区域)定义新的剪切区域
说明:
1)SelectClipPath Selects the current path as a clipping region for the device context, combining the new region with any existing clipping region by using the specified mode. The device context identified must contain a closed path.
////
nMode:RGN_AND,RGN_COPY,RGN_DIFF,RGN_OR,RGN_XOR
RGN_AND The new clipping region includes the intersection (overlapping areas) of the current clipping region and the current path.
RGN_COPY The new clipping region is the current path.
RGN_DIFF The new clipping region includes the areas of the current clipping region, and those of the current path are excluded.
RGN_OR The new clipping region includes the union (combined areas) of the current clipping region and the current path.
RGN_XOR The new clipping region includes the union of the current clipping region and the current path, but without the overlapping areas.
2)应用:当作图的时候,如果想要在整幅图形其中的某个部分和其它部分有所区别,我们可以把这部分图形放到路径层当中,然后指定调用指定互操作模式调用SelectClipPath( int nMode )函数来使路径层和覆盖在其上新绘图剪切区域进行互操作,达到特殊效果。
4.文本字符串一些函数:
COLORREF GetBkColor( ) const;//得到背景颜色
virtual COLORREF SetBkColor( COLORREF crColor );//设置背景颜色
BOOL SetTextBkColor( COLORREF cr );//设置文本背景颜色
virtual COLORREF SetTextColor( COLORREF crColor );//设置文本颜色
virtual BOOL TextOut( int x, int y, LPCTSTR lpszString, int nCount );//输出文本
BOOL TextOut( int x, int y, const CString& str );
CString Left( int nCount ) const;//得到字符串左边nCount个字符
int GetLength( ) const;//得到字符串长度
5.字体CFont::CFont
CFont( );//构造函数
//Constructs a CFont object. The resulting object must be initialized with CreateFont, CreateFontIndirect, CreatePointFont, or CreatePointFontIndirect before it can be used.
选用字体事例代码组:
CClientDC dc(this);
CFont font;//构造字体对象
font.CreatePointFont(300,"华文行楷",NULL);//初始化字体对象,与字体资源相关联
CFont *pOldFont=dc.SelectObject(&font);//将新字体选入DC
...
dc.SelectObject(pOldFont);//恢复原字体
说明:
1)构造字体对象时候,必须初始化。(初始化是将字体对象与字体资源相关联)。
2)初始化对象时候,选用的字体也可以是系统字体,但不一定都有效,据测试选用。
CFont font;//创建字体对象
font.CreatePointFont(300,"华文行楷",NULL);//设置
CFont *pOldFont=dc.SelectObject(&font);//将字体选择到DC中
TEXTMETRIC tm;//创建字体信息对象
dc.GetTextMetrics(&tm);//获得当前字体信息
if(0x0d==nChar)//处理回车键
{
m_strLine.Empty();
m_ptOrigin.y+=tm.tmHeight;
}
else if(0x08==nChar)//处理退格键
{
COLORREF clr=dc.SetTextColor(dc.GetBkColor());
dc.TextOut(m_ptOrigin.x,m_ptOrigin.y,m_strLine);
m_strLine=m_strLine.Left(m_strLine.GetLength()-1);
dc.SetTextColor(clr);
}
else
{
m_strLine+=nChar;
}
CSize sz=dc.GetTextExtent(m_strLine); CPoint pt;//处理光标的位置
pt.x=m_ptOrigin.x+sz.cx;
pt.y=m_ptOrigin.y; SetCaretPos(pt);
dc.TextOut(m_ptOrigin.x,m_ptOrigin.y,m_strLine);//输出字体 dc.SelectObject(pOldFont);//将原先的字体选择回去。有了CEditView和CRichEditView之后,就没有那么辛苦了。
5.模拟卡啦OK变色的步骤。
(1)设置定时器
(2)在定时器中加入代码
平滑变色
CDC::TextOut()是一个字母一个字母的输出,达不到平滑效果。
CDC::DrawText():将文字的输出局限于一个矩形区域,超出矩形区域的文字都被截断。利用这一特点,可每隔些时间增加矩形大小,从而可实现人眼中的平滑效果。
CWnd::SetTimer():设置定时器。按设定的时间定时发送WM_TIMER消息。
具体代码如下:(编完一个之后,注释掉前面的,继续编)
我这里把第五讲分成两个程序,把最后一个卡拉OK输出单独出来了,怕太混乱了。
CXXView:
- private:
- CBitmap cbitmap;
- CPoint m_orgpt;
- CString cstring;
- CPoint cppt;
位图:
OnDraw:
- void CMyTextView::OnDraw(CDC* pDC)
- {
- CMyTextDoc* pDoc = GetDocument();
- ASSERT_VALID(pDoc);
- // TODO: add draw code for native data here
- /*this->CreateSolidCaret(10,20);//创建一个光标
- this->ShowCaret();*/ //显示光标,这两个函数都是CWnd的成员函数
- /*TEXTMETRIC tm; //这个结构体可以获取字体的信息
- pDC->GetTextMetrics(&tm);
- this->CreateSolidCaret(tm.tmAveCharWidth/8,tm.tmHeight);//创建一个大小高度比较正常的光标
- this->ShowCaret();*/
- /*cbitmap.LoadBitmap(BITMAPID);//cbitmap资源一般设置成为成员变量
- this->CreateCaret(&cbitmap);//创建了一个位图光标,一闪一闪
- this->ShowCaret();*/
- /*
- //CString cstring("大家好,欢迎一起学习MFC");
- CString cstring="大家好,欢迎一起学习MFC";
- pDC->TextOut(100,100,cstring);*/
- /*CString cstring;
- cstring.LoadString(IDS_MYBOLG);
- pDC->TextOut(100,100,cstring);*/
- /*CString cstring("Hello,欢迎大家一起学习MFC");
- CSize cs=pDC->GetTextExtent(cstring);//区别跟TextMetric的不同,这里是根据字符串来获取的
- pDC->TextOut(100,100,cstring);
- pDC->BeginPath();
- pDC->Rectangle(100,100,cs.cx+100,cs.cy+100);//如果没有上下的XXPath,则显示一被白色画刷填充的矩形
- pDC->EndPath();
- //SelectClipPath的参数,大家不妨多试几种,看看效果
- //RGN_AND RGN_COPY RGN_DIFF RGN_OR RGN_XOR
- pDC->SelectClipPath(RGN_DIFF);
- for(int i=0;i<300;i+=10)//花了横竖很多线条,交叉着
- {
- pDC->MoveTo(i,0);
- pDC->LineTo(i,300);
- pDC->MoveTo(0,i);
- pDC->LineTo(300,i);
- }*/
- TEXTMETRIC tm;
- pDC->GetTextMetrics(&tm);
- this->CreateSolidCaret(tm.tmAveCharWidth/8,tm.tmHeight);
- this->ShowCaret();
- }
CXXView右键,添加一个WM_Char消息:
生成OnChar方法
- <span style="font-size:10px;">void CMyTextView</span>::OnChar(UINT nChar, UINT nRepCnt, UINT nFlags)
- {
- // TODO: Add your message handler code here and/or call default
- CClientDC ccdc(this);
- TEXTMETRIC tm;
- ccdc.GetTextMetrics(&tm);
- CFont cfont;
- cfont.CreatePointFont(130,"宋体",NULL);
- CFont *pcf=ccdc.SelectObject(&cfont);
- if(0x0D==nChar)//换行
- {
- cstring.Empty();//换行清空
- m_orgpt.y=m_orgpt.y+tm.tmHeight;//横坐标不变,纵坐标向下一个字符的高度
- }
- else if (0x08==nChar)//删除键
- {
- COLORREF bkcolor=ccdc.GetBkColor();//获取背景颜色
- COLORREF oldcolor=ccdc.SetTextColor(bkcolor);//设置字体颜色为背景颜色,并且保留设置前的字体颜色
- ccdc.TextOut(m_orgpt.x,m_orgpt.y,cstring);//用背景颜色输出文字
- cstring=cstring.Left(cstring.GetLength()-1);//去掉最后一个字符
- ccdc.SetTextColor(oldcolor);//重新设置原来的字体颜色
- }
- else
- {
- cstring+=nChar;//插入字符
- }
- CSize cs=ccdc.GetTextExtent(cstring);//获取字体矩形的对象
- CPoint cpt;
- cpt.x=m_orgpt.x+cs.cx;
- cpt.y=m_orgpt.y;
- this->SetCaretPos(cpt);//重新设置光标的位置
- ccdc.TextOut(m_orgpt.x,m_orgpt.y,cstring);//输出字符串
- ccdc.SelectObject(pcf);
- CView::OnChar(nChar, nRepCnt, nFlags);
- }
感觉还是不完美,比如删除,当删除一行之后,应该跳回到上一行的最后,这里都没有判断,请自行测试
下面是单独出来的卡拉OK显示
成员变量:
- private:
- CString cstring;
- int m_width;
构造函数:
- CJustTextView::CJustTextView()
- {
- // TODO: add construction code here
- cstring="";
- m_width=0;
- }
OnDraw函数:
- void CJustTextView::OnDraw(CDC* pDC)
- {
- CJustTextDoc* pDoc = GetDocument();
- ASSERT_VALID(pDoc);
- // TODO: add draw code for native data here
- cstring="hello,welcome to myblog,欢迎大家一起学习MFC";
- pDC->TextOut(0,200,cstring);
- this->SetTimer(1,50,NULL);
- }
CXXView右键,添加一个WM_TIMER消息:
自动成成OnTimer方法
- void CJustTextView::OnTimer(UINT nIDEvent)
- {
- // TODO: Add your message handler code here and/or call default
- CClientDC ccdc(this);
- TEXTMETRIC tm;
- ccdc.GetTextMetrics(&tm);
- CSize cs=ccdc.GetTextExtent(cstring);
- COLORREF oldcolor=ccdc.SetTextColor(RGB(255,0,0));
- if(1==nIDEvent)
- {
- m_width+=5;
- CRect crect;
- crect.left=0;
- crect.top=200;
- crect.right=m_width;
- crect.bottom=crect.top+tm.tmHeight;
- ccdc.DrawText(cstring,&crect,DT_LEFT);
- crect.left=0;
- crect.top=150;
- crect.right=m_width;
- crect.bottom=crect.top+tm.tmHeight;
- ccdc.DrawText(cstring,&crect,DT_RIGHT);
- if(m_width>cs.cx)
- {
- m_width=0;
- ccdc.SetTextColor(RGB(0,255,0));
- ccdc.TextOut(0,200,cstring);
- COLORREF bkcolor=ccdc.GetBkColor();
- ccdc.SetTextColor(bkcolor);
- ccdc.TextOut(0,150,cstring);
- }
- }
- CView::OnTimer(nIDEvent);
- }
- 第五章,文本编程
- 第五章-文本编程
- 第五讲 文本编程
- 孙鑫:第五讲 文本编程
- 孙鑫《vc ++深入详解》第五章文本编程
- VC++深入详解第五章——文本编程
- 《VC++深入详解》学习笔记 第五章 文本编程
- 第五章 文本
- 第五章 文本和字体
- 第一部分 Shell基础编程——第五章 文本过滤
- 编程珠玑第五章
- 编程珠玑第五章
- 第五章 编程练习
- 第五章 编程小事
- 第五章编程题
- 孙鑫VC学习笔记:第五讲 文本编程
- 孙鑫VC学习笔记:第五讲 文本编程
- MFC(文本编程,孙鑫C++第五讲笔记整理)
- 小米无线路由器 + u盘
- python切片与迭代
- 逻辑与(&)和短路与(&&)
- XML学习
- Centos 7 mysql 5.7 给root开启远程访问权限,修改root密码
- 第五章-文本编程
- C++基础(三)
- SSH2框架设计---代码整合
- wex5去除同一行两个相邻button的空隙问题
- scikit-learn文档学习笔记
- java面向对象的几个基本特征
- Node.js安装与配置
- Windows环境下Python开发环境配置
- VTK修炼之道63:纹理映射体绘制_二维纹理映射