//*************************
//获取点中y坐标最大值
//*************************
int CPolygonFillView::GetMaxY()
{
int result = points[0].y;
for(int i = 1; i < count; i++)
if(result < points[i].y)
result = points[i].y;
return result;
}
//*************************
//获取点中y坐标最小值
//*************************
int CPolygonFillView::GetMinY()
{
int result = points[0].y;
for(int i = 0; i < count; i++)
if(result > points[i].y)
result = points[i].y;
return result;
}
//*******************************************************
//判断是否为极值点,参数为一条边的y较小点和其在pionts数组中的下标
//********************************************************
bool CPolygonFillView::IsSuperPoint(CPoint &point,int i){
if(i == 0)
return ((points[0].y > points[1].y&&points[0].y > points[count-1].y)
||(points[0].y < points[1].y&& points[0].y < points[count-1].y));
else if(i == count-1)
return ((points[i].y > points[0].y&&points[i].y > points[i-1].y)
||(points[i].y < points[0].y&&points[i].y < points[i-1].y));
else
return ((points[i].y > points[i-1].y&&points[i].y > points[i+1].y)
||(points[i].y < points[i-1].y&&points[i].y < points[i+1].y));
}
//*************************
//用改进的冒泡法对链表排序
//*************************
void CPolygonFillView::SortEdges()
{
if(head == NULL || head->next == NULL) return;
Edge *p,*q = head;
float tymax;
float txmin;
float tm;
bool flag = true;
while(flag){
flag = false;
for(p=q;p->next!=NULL;p=p->next){
if(p->xmin > p->next->xmin){
tymax = p->ymax;
p->ymax = p->next->ymax;
p->next->ymax = tymax;
txmin = p->xmin;
p->xmin = p->next->xmin;
p->next->xmin = txmin;
tm = p->m;
p->m = p->next->m;
p->next->m = tm;
flag = true;
}
q = q->next;
}
}
}
//**************************************
// 从活动边表中清除不满足条件的边
//**************************************
void CPolygonFillView::ClearEdges(int ymax)
{
if(head == NULL) return;
if(head->next==NULL)
{
if(head->ymax = ymax)
head = NULL;
}
else
{
Edge *p = head,*q = head->next,*r = head;
while(head!=NULL && head->ymax == ymax )
{
head = head->next;
r->next = NULL;
r = head;
}
while(q!=NULL)
{
if(q->ymax == ymax){
p->next = q->next;
q->next = NULL;//删除边
q = p->next;//恢复q为恰当的值
}
else{
if(p==NULL) break;
p = p->next;
if(q==NULL) break;
q = q->next;
}
}
}
}
//**************************************
// 向活动边表中添加的边
//**************************************
void CPolygonFillView::AddEdges(int index)
{
if(edgeTable[index].head == NULL) return;
if(head == NULL)
{
head = edgeTable[index].head;
}
else
{
for(Edge *p=head;p->next!=NULL;p=p->next){}
p->next = edgeTable[index].head;
}
}
//*****************************
//初始化边表
//*****************************
void CPolygonFillView::InitEdgeTable()
{
int ymin,t;
Edge *p = NULL;
CPoint cp;
for(int i=0;i<count;i++)
{
cp = points[i].y > points[(i+1)%count].y
? points[(i+1)%count] : points[i];
t = points[i].y > points[(i+1)%count].y
? (i+1)%count : i;
edge[i].m =(float) (points[i].x - points[(i+1)%count].x)/
(points[i].y - points[(i+1)%count].y);
if(IsSuperPoint(cp,t))
{
edge[i].xmin = cp.x;
edge[i].ymax = points[i].y > points[(i+1)%count].y
? points[i].y : points[(i+1)%count].y;
ymin = cp.y;
}
else
{
edge[i].xmin = cp.x + edge[i].m;
edge[i].ymax = points[i].y > points[(i+1)%count].y
? points[i].y : points[(i+1)%count].y;
ymin = cp.y + 1;
}
if((p=edgeTable[ymin].head)==NULL)
{
edgeTable[ymin].head = &edge[i];
}
else
{
if(edge[i].xmin<=p->xmin){
edge[i].next = p;
edgeTable[ymin].head = &edge[i];
}
else{
while(p->next!=NULL&&p->next->xmin<edge[i].xmin){
p = p->next;
}
edge[i].next = p->next;
p->next = &edge[i];
}
}
}
}
//******************
//对多边形进行填充
//******************
void CPolygonFillView::Polygonfill(CClientDC &dc,COLORREF color)
{
int begin = GetMinY(),end = GetMaxY();
Edge *q = NULL;
for(int i=begin;i<=end;i++)
{
AddEdges(i);
SortEdges();
q=head;
if(q==NULL) break;
while(q->next!=NULL)
{
for(int j=q->xmin;j<q->next->xmin;j++)
{
SetPixel(dc,j,i,color);
}
q=q->next->next;
if(q==NULL) break;
}
ClearEdges(i);
q=head;
while(q!=NULL)
{
q->xmin += q->m;
q = q->next;
}
}
}
分享到:
相关推荐
在C++ MFC(Microsoft Foundation Classes)环境中,我们可以构建一个用户界面,允许用户通过鼠标绘制多边形,然后应用扫描线填充算法来填充它们。 MFC是Microsoft开发的一个C++类库,用于构建Windows应用程序。它...
在进行实验时,解压文件“实验三之扫描线算法”,你会得到一个已经实现扫描线算法的MFC应用程序,可以直接运行查看效果。通过调试和分析源代码,可以深入理解算法的各个步骤,以及MFC如何与Windows图形系统交互来...
在本资源中,我们关注的是基于MFC(Microsoft Foundation Classes)框架实现的多边形填充算法,特别是种子填充算法。MFC是微软为Windows应用程序开发提供的一套C++类库,它简化了Windows API的使用。 种子填充算法...
计算机图形学多边形填充算法实验报告 计算机图形学是计算机科学的一个分支,它研究如何使用计算机生成和处理图形信息。多边形填充算法是计算机图形学中一个重要的研究方向,本实验报告将对多边形填充算法的实现和...
在本项目中,我们关注的是使用VC++6.0的MFC(Microsoft Foundation Classes)库来实现计算机图形学的各种基本功能。MFC是微软提供的一套C++类库,用于简化Windows应用程序开发,尤其是图形用户界面的应用。 首先,...
在这个场景中,我们将深入探讨如何使用MFC实现扫描线区域填充算法。 扫描线算法是一种基于图像处理的填充方法,它的基本思想是从图像的顶部到底部逐行扫描,通过判断每个像素点是否属于填充区域来确定其颜色。在MFC...
通过阅读和分析源码,可以深入理解多边形填充算法的细节,并学习如何在实际应用中实现这些算法。同时,由于项目支持VS(Visual Studio),开发者可以在IDE环境下方便地调试和运行代码,进一步加深对算法的理解。
在计算机图形学中,扫描转换(也称为光栅化)是一种将几何形状,如多边形,转换为屏幕上的像素表示的技术。MFC(Microsoft Foundation Classes)是微软提供的一种C++库,用于构建Windows应用程序,它包含了许多方便...
综上所述,在MFC环境中实现扫描线多边形填充算法涉及多个步骤,包括数据结构的使用、排序、边界检测、交点计算以及屏幕绘图。理解这些概念并熟练运用,能够帮助开发者在图形界面应用中实现高质量的多边形填充效果。
在`OnDraw`函数中,我们可以调用`CDC`类提供的绘图方法,如`MoveTo`和`LineTo`来绘制多边形的轮廓,以及自定义的填充算法来填充内部。 资源文件“多边形平滑填充(颜色过渡三角形)”可能包含了实现上述逻辑的代码...
本程序是在上次的画直线、画圆的程序上修改的,添加了扫描线填充算法,使用OpenGL+MFC实现。 填充算法的使用说明: 1、选择菜单中的“种子填充”--〉“鼠标画边界”,然后在绘图区域左键点击若干个点作为多边形的...
在计算机图形学中,多边形填充是一种基本的技术,用于在屏幕上将闭合的多边形区域用颜色或纹理填充。这个过程称为扫描转换,它涉及到将多边形的几何形状转化为像素的集合。本实验重点是理解并实现两种算法:中点...
在计算机图形学中,填充...总的来说,理解和实现这两种填充算法,不仅可以深化对计算机图形学的理解,也能提升在MFC环境下的编程能力。通过实践,我们可以更好地掌握这些技术,并应用于各种图形处理和图像渲染的场景。
这段程序代码展示了如何在 Turbo C (TC) 编译器环境下实现多边形填充算法。这里主要涉及了两种算法:Bresenham 线画法和边缘表算法。 首先,`bresenham_line` 函数实现了 Bresenham 线画法,这是一种用于快速绘制...
### 填充算法在MFC中的实现 #### 背景介绍 本文将详细介绍一个基于Visual C++的MFC应用程序中的图形填充算法实现。该应用程序主要涉及的是使用MFC框架来开发一个简单的图形界面,并在该界面上实现对特定区域进行...
在计算机图形学中,多边形填充是...总的来说,掌握多边形填充算法是图形学和游戏开发的基础,它可以帮助你更好地理解和创建复杂的2D图形效果。无论是简单的练习项目还是大型应用,理解并实现这些算法都是非常有价值的。
本实验“图形学多边形填充实验”着重于理解和应用边标志算法来实现这一功能,并结合MFC(Microsoft Foundation Classes)框架进行编程。 首先,我们要了解**边标志算法**,它是实现多边形填充的一种常用方法。该...
总的来说,"MFC多边形扫描转换算法"涉及到计算机图形学中的基本概念,如多边形光栅化、扫描线算法,以及MFC库中的图形绘制和事件处理机制,这些都是开发图形界面应用程序时不可或缺的技术。开发者可以通过学习和实践...
在计算机图形学中,多边形的扫描填充是一种常见的技术,用于在屏幕上为闭合的多边形区域填充颜色。MFC(Microsoft Foundation Classes)是微软提供的一种C++类库,用于构建Windows应用程序,它提供了丰富的图形用户...