{ CPoint prePnt,curPnt; prePnt=m_bR; //获得鼠标所在的前一位置 curPnt=point; //绘制橡皮筋线
m_r=ComputeRadius(m_bO,prePnt);
MidpointCircle(pDC,m_bO.x,m_bO.y,m_r,RGB(255,0,0));//用异或模式重复画圆,擦出所画的圆 // DrawCircle(pDC,m_bO,prePnt); m_r=ComputeRadius(m_bO,curPnt);
MidpointCircle(pDC,m_bO.x,m_bO.y,m_r,RGB(255,0,0)); //用当前位置作为圆周上的点画圆 m_bR=point; } pDC->SetROP2(nDrawmode); //恢复原绘图模式 ReleaseDC(pDC); //释放设备环境 CView::OnMouseMove(nFlags, point); }
-- 42
任务三:编写中点画椭圆法的扫描转换程序
程序实现步骤:
(1) 建立MidPointEllise工程文件;
(2)右击CMidPointElliseView类,建立成员函数
void MidpointEllise(CDC *pDC, int x0, int y0, int a, int b, COLORREF color) (3) 编写成员函数代码,程序如下:
void CMidPointEllipseView::MidpointEllise(CDC *pDC, int x0, int y0, int a, int b, COLORREF color) {
int x,y; float d1,d2; x=0;y=b;
d1=b*b+a*a*(-b+0.25);
pDC->SetPixel(x+x0,y+y0,color);
-- 43
while (b*b*(x+1)
if (d1<0) {
d1+=b*b*(2*x+3); x++; } else {
d1+=(b*b*(2*x+3)+a*a*(-2*y+2)); x++; y--; }
pDC->SetPixel(x0+x,y0+y,color); pDC->SetPixel(x0+x,y0-y,color); pDC->SetPixel(x0-x,y0+y,color); pDC->SetPixel(x0-x,y0-y,color); } // 上半部分
d2=(b*(x+0.5))*(b*(x+0.5))+(a*(y-1))*(a*(y-1))-(a*b)*(a*b); while (y>0) {
if (d2<0) {
d2+=b*b*(2*x+2)+a*a*(-2*y+3); x++; y--; } else {
d2+=a*a*(-2*y+3); y--; }
pDC->SetPixel(x0+x,y0+y,color); pDC->SetPixel(x0+x,y0-y,color); pDC->SetPixel(x0-x,y0+y,color); pDC->SetPixel(x0-x,y0-y,color); } //下半部分 }
(4) 编写OnDraw()函数
void CMidPointEllipseView::OnDraw(CDC* pDC) { CMidPointEllipseDoc* pDoc = GetDocument(); ASSERT_VALID(pDoc); // TODO: add draw code for native data here
MidpointEllise(pDC, 300, 200, 50, 20, RGB(255,0,0));
-- 44
}
(6) 编译、运行程序。
任务四:添加鼠标程序,实现交互式画椭圆
程序实现步骤:
(1)~(3)同上,建立MidPointElliseMouse工程文件。 (4)添加成员变量 protected: int b; int a; int m_ist; CPoint CenterPoint; CPoint RightBottom; CPoint LeftTop;
(5)在构造函数中赋初值
CMidPointElliseMouseView::CMidPointElliseMouseView() { // TODO: add construction code here LeftTop.x=0;LeftTop.y=0; // 左上角坐标初值
RightBottom.x=0;RightBottom.y=0; // 右下角坐标初值 CenterPoint.x=0;CenterPoint.y=0; // 中心点坐标初值 a=0; b=0; //长轴和短轴长度
m_ist=0; //0:表示第一点,1:表示第二点 }
(6) 添加OnLButtonDown()函数
void CMidPointElliseMouseView::OnLButtonDown(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default CDC *pDC=GetDC();
-- 45
pDC->SelectStockObject(NULL_BRUSH); if (!m_ist) //第一点,左上角 { LeftTop=RightBottom=point; //记录第一次单击鼠标位置 m_ist++; } else { RightBottom=point; //记录第二次单击鼠标的位置 m_ist--; // 为新绘图作准备 CenterPoint.x=(LeftTop.x+RightBottom.x)/2; CenterPoint.y=(LeftTop.y+RightBottom.y)/2; a=(int)abs((RightBottom.x-LeftTop.x))/2; b=(int)abs((RightBottom.y-LeftTop.y))/2; MidpointEllise(pDC,CenterPoint.x,CenterPoint.y,a,b,RGB(255,0,0)); } ReleaseDC(pDC); //释放设备环境 CView::OnLButtonDown(nFlags, point); }
(7)添加OnMouseMove()函数
void CMidPointElliseMouseView::OnMouseMove(UINT nFlags, CPoint point) { // TODO: Add your message handler code here and/or call default CDC *pDC=GetDC();
int nDrawmode=pDC->SetROP2(R2_NOT); //设置异或绘图模式,并保存原来绘图模式 pDC->SelectStockObject(NULL_BRUSH); if(m_ist==1) { CPoint prePnt,curPnt; prePnt=RightBottom; //获得鼠标所在的前一位置 curPnt=point; //绘制橡皮筋线
CenterPoint.x=(LeftTop.x+prePnt.x)/2; CenterPoint.y=(LeftTop.y+prePnt.y)/2; a=(int)abs((prePnt.x-LeftTop.x))/2; b=(int)abs((prePnt.y-LeftTop.y))/2; MidpointEllise(pDC,CenterPoint.x,CenterPoint.y,a,b,RGB(255,0,0)); //用异或模式重复画圆,擦出所画的圆 //用当前点作为右下角点,画椭圆
CenterPoint.x=(LeftTop.x+curPnt.x)/2; CenterPoint.y=(LeftTop.y+curPnt.y)/2; a=(int)abs((curPnt.x-LeftTop.x))/2; b=(int)abs((curPnt.y-LeftTop.y))/2;
-- 46