以下是第五讲的主要内容(文本输入输出,要先设置光标位置,然后再输出文本)
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;//获得字体高度。
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);
路径层有什么作用?可以保护我们先前的文本或者图像不被后来画的覆盖。
4.在View上输入文字的步骤。
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)在定时器中加入代码
具体代码如下:(编完一个之后,注释掉前面的,继续编)
我这里把第五讲分成两个程序,把最后一个卡拉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方法
void CMyTextView::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); }
效果图较多,这里不一一截图了: