深入浅出MFC第2版(PDF格式)-第98部分
按键盘上方向键 ← 或 → 可快速上下翻页,按键盘上的 Enter 键可回到本书目录页,按键盘上方向键 ↑ 可回到本页顶部!
————未阅读完?加入书签已便下次继续阅读!
此函数把点坐标从数组之中一个一个取出,画到窗口上,所以你会看到整个原始绘图
过程的重现,而不是一整张图啪一下子出现。想当然耳,这个函数内会有CreatePen、
SelectObject、MoveTo 、LineTo 等GDI 动作,以及从数组中取坐标点的动作。取
点动作直接利用CArray 的operator' ' 运算子即可办到:
BOOL CStroke::DrawStroke(CDC* pDC)
{
CPen penStroke;
if (!penStroke。CreatePen(PS_SOLID; m_nPenWidth; RGB(0;0;0)))
return FALSE;
CPen* pOldPen = pDC…》SelectObject(&penStroke);
pDC…》MoveTo(m_pointArray'0');
for (int i=1; i 《 m_pointArray。GetSize(); i++)
{
pDC…》LineTo(m_pointArray'i');
}
pDC…》SelectObject(pOldPen);
return TRUE;
}
Serialize:让我们这么想象写档动作:使用者下命令给程序,程序发命令给文
件,文件发命令给线条,线条发命令给点数组,点数组于是把一个个的坐标点
写入磁盘中。请注意,每一线条除了拥有点数组之外,还有一个笔划宽度,读
写文件时可不要忘了这份资料。
void CStroke::Serialize(CArchive& ar)
{
if (ar。IsStoring())
{ // 写档
ar 》 w;
m_nPenWidth = w;
m_pointArray。Serialize(ar);
}
}
485
…………………………………………………………Page 548……………………………………………………………
第篇 深入 MFC 程式設計
肯定你会产生两个疑问:
1。 为什么点数组的读文件写文件动作完全一样,都是Serialize(ar) 呢?
2。 线条串行的Serialize 函数如何能够把命令交派到线条的Serialize 函数呢?
第一个问题的答案很简单,第二个问题的答案很复杂。稍后我对此有所解释。
void CScribbleView::OnDraw(CDC* pDC)
void CScribbleView::OnDraw(CDC* pDC)
{
成员变量: {
CScribbleDoc* pDoc = GetDocument();
CObList m_strokeList; CScribbleDoc* pDoc = GetDocument();
UINT m_nPenWidth; CTypedPtrList&
CPen m_penCur; CTypedPtrList&
strokeList = pDoc…》m_strokeList;
strokeList = pDoc…》m_strokeList;
成员函数: POSITION pos = strokeList。GetHeadPosition();
POSITION pos = strokeList。GetHeadPosition();
while (pos != NULL)
Serialize() while (pos != NULL)
{
InitDocument() {
CStroke* pStroke = strokeList。GetNext(pos);
OnNewDocument() CStroke* pStroke = strokeList。GetNext(pos);
pStroke…》DrawStroke(pDC);
OnOpenDocument() pStroke…》DrawStroke(pDC);
}
NewStroke() } }
}
DeleteContent()
我们将改写OnDraw 如上,令它取得Document 后巡访串行的每一
GetCurrentPen()
元素(线条),并调用每一线条的DrawStroke 把它自己绘出来。
CStroke 对象(线条#0) CStroke 对象(线条#1) CStroke 对象(线条#2)
。。。
成员变量: 成员变量: 成员变量:
UINT m_nPenWidth; UINT m_nPenWidth; UINT m_nPenWidth;
CDWordArray m_pointArray; CDWordArray m_pointArray; CDWordArray m_pointArray;
成员函数: 成员函数: 成员函数:
DrawStroke() DrawStroke() DrawStroke()
Serialize() Serialize() Serialize()
图8…4 Scribble 的Document/View 成员鸟瞰
图8…4 把Scribble Step1 的Document/View 重要成员集中在一起显示,帮助你做大局
观。请注意,虽然本图把「成员函数」和「成员变量」画在每一个对象之中,但你知道,
事实上C++ 类别的成员函数另放在对象内存以外,并不是每一个对象都有一份函数
。。。 码。只有non…static 成员变量,才会每个对象各有一份。这个观念我曾在第2章强调过。
486
…………………………………………………………Page 549……………………………………………………………
第8章 Document…View 深入探討
Scribble Step1 的View :资料重绘与编辑
View 有两个最重要的任务,一是负责资料的显示,另一是负责资料的编辑(透过键盘或
鼠标)。本例的CScribbleView 包括以下特质:
解读CScribbleDoc 中的资料,包括笔宽以及一系列的CPoint 对象,画在View
窗口上。
允许使用者以鼠标左键充当画笔在View 窗口内涂抹,换句话说CSribbleView
必须接受并处理WM_LBUTTONDOWN 、WM_MOUSEMOVE 、WM_LBUTTONUP
三个消息。
当Framework 收到WM_PAINT ,表示画面需要重绘,它会调用OnDraw (注),由
OnDraw 执行真正的绘图动作。什么时候会产生重绘消息WM_PAINT 呢?当使用者改
变窗口大小,或是将窗口图标化之后再恢复原状,或是来自程序(自己或别人)刻意的
制造。除了在必须重绘时重绘之外,做为一个绘图软件,Scribble 还必须「实时」反应
鼠标左键在窗口上移动的轨迹,不能等到WM_PAINT 产生了才有所反应。所以,我们
必须在OnMouseMove 中也做绘图动作,那是针对一个点一个点的绘图,而OnDraw 是
大规模的全部重绘。
注:其实Framework 是先调用OnPaint,OnPaint 再调用OnDraw 。关于OnPaint,第12
章谈到打印机时再说。
绘图前当然必须获得资料内容,调用GetDocument 即可获得,它传回一个CScribbleDoc
对象指针。别忘了View 和Document 以及Frame 窗口早在注册Document Template 时
就建立彼此间的关联了。所以,从CScribbleView 发出的GetDocument 函数当然能够获
得CScribbleDoc 的对象指针。View 可以藉此指针取得Document 的资料,然后显示。
487
…………………………………………………………Page 550……………………………………………………………
CScribbleView 的修改
以下是Step1 程序的View 的设计。其中有鼠标接口,也有资料显示功能OnDraw
SCRIBBLEV IEW 。H (阴影表示与Step0的差异)
#0001 class CScribbleView : public CView
#0002 {
#0003 protected: // create from serialization only
#0004 CScribbleView();
#0005 DECLARE_DYNCREATE(CScribbleView)
#0006
#0007 // Attributes
#0008 public:
#0009 CScribbleDoc* GetDocument();
#0010
#0011 protected:
#0012 CStroke* m_pStrokeCur; // the stroke in progress
#0013 CPoint m_ptPrev; // the last mouse pt in the stroke in progress
#0014
#0015 // Operations
#0016 public:
#0017
#0018 // Overrides
#0019 // ClassWizard generated virtual function overrides
#0020 //{{AFX_VIRTUAL(CScribbleView)
#0021 public:
#0022 virtual void OnDraw(CDC* pDC); // overridden to draw this view
#0023 virtual BOOL PreCreateWindow(CREATESTRUCT& cs);
#0024 protected:
#0025 virtual BOOL OnPreparePrinting(CPrintInfo* pInfo);
#0026 virtual void OnBeginPrinting(CDC* pDC; CPrintInfo* pInfo);
#0027 virtual void OnEndPrinting(CDC* pDC; CPrintInfo* pInfo);
#0028 //}}AFX_VIRT