网站首页
手机版

在Windows中显示多幅彩色图像的技术

更新时间:2005-12-20 11:50:47作者:未知

摘 要 该文论述了在Windows同一窗口中显示多幅彩色图像的技术和实现方法,并提供最优化程序压缩原图像的颜色数量。
关键词 调色板 Windows 图像处理
目前的大多数微机中都配置了高性能的TVGA或SVGA图形卡,在这些图形卡中,红、绿、蓝三元色各占六位,颜色总数占18位,故可以显示262144种颜色,在更高级的图形卡中,甚至可达24位的真彩色。使用调色板技术,可以在一个窗口显示256种颜色,每一副图像都具有独立的256个调色板,显示时更换调色板便可满足各幅图像的颜色要求。
然而,在某些应用中则要求同一窗口中显示两幅甚至更多的彩色图像,如果按照常规设计,窗口中只能正确地显示其中一幅图像,而其它的图像则由于其调色板被更换,颜色就会混乱。在我们开发多媒体查询系统——山东省旅游资源查询子系统时,系统要求以一幅山东省彩色地图为背景,然后在各个旅游景点设置一个触摸按钮,当用户触摸该按钮时,在窗口的右下部分显示该景点的彩色图像。如果图像不经过特殊处理,则在显示该景点的彩色图像的同时也更换了该窗口的调色板,使背景图像的颜色失真。为了解决这个问题,我们对图像做了特殊处理,使背景图像和各个景点图像的调色板不发生冲突。同时还要考虑到Windows占用了前20个调色板,在一般情况下不允许更改。根据系统的实际情况,我们对调色板做了如下布置:第0~19号调色板为Windows系统保留;第20~148号调色板为背景图像使用,一旦背景图像使用后,就不再更改;第149~255号调色板为各景点图像使用,当显示不同的景点图像时,随时更改这些调色板。定义一个PALETTEENTRY类型的数组palette[256]用来保存各颜色分量。该类型是Windows定义的一种结构:
typedef struct {
BYTE peRed;
/*调色板项的红色饱和度*/
BYTE peGreen;/*调色板项的绿色饱和度*/
BYTE peBlue;
/*调色板项的蓝色饱和度*/
BYTE peFlags;/*NULL、PC-EXPLICIT、PC-NOCOLLAPSE一般取PC-EXPLICIT*/
} PALETTEENTRY;
这样把背景图像的第20~148号调色板和景点图像的第149~255号调色板分别放入pal
ette[20]~palette[255]中,然后按下列步骤实现该调色板。
1.定义HDC hdc; HPALETTE w-hp; LOGPALETTE *pal;
2.给pal赋值
pal=(NPLOGPALETTE)LocalAlloc(LMEM-FIXED,
sizeof(LOGPALETTE)+256*sizeof(PALETTEENTRY));
pal->palVersion=0X300;/*
pal->palNumEntries=256;
memcpy(%pal->palPalEntry[0],&palette[0],
256*sizeof(PALETTEENTRY);
3.实现该调色板
hdc=GetDC(hWnd);
w-hp=CreatePalette((LPLOGPALETTE)pal);
w-hp=SelectPalette(hdc,w-hp,0);
RealizePalette(hdc);
LocalFree(HANDLE)pal);
其中hWnd为要显示图像的窗口句柄。按上述步骤实现该调色板后,读入要显示的图像,然后映射到hdc中即可。
本文提供的程序cpcolor.c可以实现把一个BMP格式的图像从256色压缩成(color2-color1+1)色,并把颜色号限制在color1到color2范围内。程序执行格式为:
cpcolor图像文件名 颜色下限 颜色上限本程序使用最优化方法,使用效果良好。
/* 源程序cpcolor.c */
#include <stdio.h>
#include <math.h>
#include <alloc.h>
unsigned char palette[256][4];
long TAB[256];
unsigned char TT[256],BB[256];
int width,depth,bytes;
/* 该函数打开图像文件并读图像的宽、高和各调色板的颜色分量,并把文件指针指向图像的开始处*/
FILE *get-bitmap-file(char*fname)
{
unsigned char ch;
int i,j,n;
FILE *fp;
fp=fopen(fname,"rb+");
if (fp==NULL) return NULL;
fseek(fp,18L,SEEK-SET);
fread(&width,2,1,fp);fseek(fp,2L,SEEK-CUR);
fread(&depth,2,1,fp);fseek(fp,2L,SEEK-CUR);
n=width/4;
if (width%4 !=0) n++;
bytes=n*4;
fseek(fp,54L,SEEK-SET);
fread(&palette[0][0],4,256,fp);
return fp;
}
/*该函数实现颜色的压缩*/
void zh_fan-tu(FILE *fp,int color1,int color2)
{
unsigned char *p,*q,cc,ch;
long len;
int max-no, i,j,m,n,r0,b0,g0,r1,b1,g1;
double dd,dmin;
char s[4];
max-no=color2-color1+1;
for (i=0;i<256;i++) TAB[i]=0;
p=(char *)malloc(bytes+1);
for (i=0;i<depth;i++) {
fread(p,bytes,1,fp);
q=p;
for (j=0;j<bytes;j++,q++) if(j>=width) break;
else {
ch=(unsigned char)*q;
TAB[ch]++;
}
}
for (i=0;i<256;i++) tt[i]=(unsigned char)i;
for (i=0;i<255;i++) for (j=i+1;j<256;j++)
if (TAB[i]<TAB[j]){
len=TAB[i];TAB[i]=TAB[摘 要 该文论述了在Windows同一窗口中显示多幅彩色图像的技术和实现方法,并提供最优化程序压缩原图像的颜色数量。
关键词 调色板 Windows 图像处理
目前的大多数微机中都配置了高性能的TVGA或SVGA图形卡,在这些图形卡中,红、绿、蓝三元色各占六位,颜色总数占18位,故可以显示262144种颜色,在更高级的图形卡中,甚至可达24位的真彩色。使用调色板技术,可以在一个窗口显示256种颜色,每一副图像都具有独立的256个调色板,显示时更换调色板便可满足各幅图像的颜色要求。

然而,在某些应用中则要求同一窗口中显示两幅甚至更多的彩色图像,如果按照常规设计,窗口中只能正确地显示其中一幅图像,而其它的图像则由于其调色板被更换,颜色就会混乱。在我们开发多媒体查询系统——山东省旅游资源查询子系统时,系统要求以一幅山东省彩色地图为背景,然后在各个旅游景点设置一个触摸按钮,当用户触摸该按钮时,在窗口的右下部分显示该景点的彩色图像。如果图像不经过特殊处理,则在显示该景点的彩色图像的同时也更换了该窗口的调色板,使背景图像的颜色失真。为了解决这个问题,我们对图像做了特殊处理,使背景图像和各个景点图像的调色板不发生冲突。同时还要考虑到Windows占用了前20个调色板,在一般情况下不允许更改。根据系统的实际情况,我们对调色板做了如下布置:第0~19号调色板为Windows系统保留;第20~148号调色板为背景图像使用,一旦背景图像使用后,就不再更改;第149~255号调色板为各景点图像使用,当显示不同的景点图像时,随时更改这些调色板。定义一个PALETTEENTRY类型的数组palette[256]用来保存各颜色分量。该类型是Windows定义的一种结构:
typedef struct {
BYTE peRed;
/*调色板项的红色饱和度*/
BYTE peGreen;/*调色板项的绿色饱和度*/
BYTE peBlue;
/*调色板项的蓝色饱和度*/
BYTE peFlags;/*NULL、PC-EXPLICIT、PC-NOCOLLAPSE一般取PC-EXPLICIT*/
} PALETTEENTRY;
这样把背景图像的第20~148号调色板和景点图像的第149~255号调色板分别放入pal
ette[20]~palette[255]中,然后按下列步骤实现该调色板。
1.定义HDC hdc; HPALETTE w-hp; LOGPALETTE *pal;
2.给pal赋值
pal=(NPLOGPALETTE)LocalAlloc(LMEM-FIXED,
sizeof(LOGPALETTE)+256*sizeof(PALETTEENTRY));
pal->palVersion=0X300;/*
pal->palNumEntries=256;
memcpy(%pal->palPalEntry[0],&palette[0],
256*sizeof(PALETTEENTRY);
3.实现该调色板
hdc=GetDC(hWnd);
w-hp=CreatePalette((LPLOGPALETTE)pal);
w-hp=SelectPalette(hdc,w-hp,0);
RealizePalette(hdc);
LocalFree(HANDLE)pal);
其中hWnd为要显示图像的窗口句柄。按上述步骤实现该调色板后,读入要显示的图像,然后映射到hdc中即可。
本文提供的程序cpcolor.c可以实现把一个BMP格式的图像从256色压缩成(color2-color1+1)色,并把颜色号限制在color1到color2范围内。程序执行格式为:
cpcolor图像文件名 颜色下限 颜色上限本程序使用最优化方法,使用效果良好。
/* 源程序cpcolor.c */
#include <stdio.h>
#include <math.h>
#include <alloc.h>
unsigned char palette[256][4];
long TAB[256];
unsigned char TT[256],BB[256];
int width,depth,bytes;
/* 该函数打开图像文件并读图像的宽、高和各调色板的颜色分量,并把文件指针指向图像的开始处*/
FILE *get-bitmap-file(char*fname)
{
unsigned char ch;
int i,j,n;
FILE *fp;
fp=fopen(fname,"rb+");
if (fp==NULL) return NULL;
fseek(fp,18L,SEEK-SET);
fread(&width,2,1,fp);fseek(fp,2L,SEEK-CUR);
fread(&depth,2,1,fp);fseek(fp,2L,SEEK-CUR);
n=width/4;
if (width%4 !=0) n++;
bytes=n*4;
fseek(fp,54L,SEEK-SET);
fread(&palette[0][0],4,256,fp);
return fp;
}
/*该函数实现颜色的压缩*/
void zh_fan-tu(FILE *fp,int color1,int color2)
{
unsigned char *p,*q,cc,ch;
long len;
int max-no, i,j,m,n,r0,b0,g0,r1,b1,g1;
double dd,dmin;
char s[4];
max-no=color2-color1+1;
for (i=0;i<256;i++) TAB[i]=0;
p=(char *)malloc(bytes+1);
for (i=0;i<depth;i++) {
fread(p,bytes,1,fp);
q=p;
for (j=0;j<bytes;j++,q++) if(j>=width) break;
else {
ch=(unsigned char)*q;
TAB[ch]++;
}
}
for (i=0;i<256;i++) tt[i]=(unsigned char)i;
for (i=0;i<255;i++) for (j=i+1;j<256;j++)
if (TAB[i]<TAB[j]){
len=TAB[i];TAB[i]=TAB[j];TAB[j]=len;
cc=TT[i];TT[i]=TT[j];TT[j]=cc;
memcpy(s,&palette[i][0],4);
memcpy(&palette[i][0],&palette[j][0],4);
memcpy(&palette[j][0],s,4);
}
for (i=color2;i>=color1;i--)
memcpy(&palette[i][0],&palette[i-color1][0],4);
for (i=0;i<max-no;i++) BB[TT[i]]=(unsigned char)i;
for (i=max-no;i<256;i++) {
r0=palette[i][2];
g0=palette[i][1];
b0=palette[i][0];
dmin=256.0*256.0*256.0;
m=0;
for (j=0;j<max-no;j++) {
r1=palette[j][2];
g1=palette[j][1];
b1=palette[j][0];
dd=1.0*(r0-r1)*(r0-r1)+1.0*(g0-g1)*(g0-g1)+1.0*(b0-b1)*(b0-b1);
dd=sqrt(dd);
if (dmin>dd) {dmin=dd;m=j;}
}
BB[TT[i]]=(unsigned char)m;
}
for (i=0;i<256;i++) BB[i]+=color1;
}
void w-create-bitmap(FILE *fp)
{
int i,j,n,m,t;
long len;
unsigned char far *p,far *q,cc,ch;
unsigned nn;
fseek(fp,54L,SEEK-SET);
len=ftell(fp);
fwrite(&palette[0][0],4,256,fp);
p=(unsigned char far *)farmalloc(bytes+1);
len=ftell(fp);
for (i=0;i<depth;i++) {
fseek(fp,len,SEEK-SET);
fread(p,bytes,1,fp);
fseek(fp,len,SEEK-SET);
len+=bytes;
q=p;
for (j=0;j<width;j++,q++) {
cc=(unsigned char)(*q);
ch=BB[cc];
*q=ch;
}
fwrite(p,bytes,1,fp);
}
farfree(p);
}
main(int argc,char *args[])
{
FILE *fp;
if (argc<4) {
printf("参数:文件名 颜色下限 颜色上限n");
return;
}
fp=get-bitmap-file(args[1]);
if (fp==NULL) return;
zh-fan-tu(fp,atoi(args[2],atoi(args[3]));
w-create-bitmap(fp);
fclose(fp);
}

参考文献
王旭 张军译.Microsoft Windows 3.1程序员参考手册.北京:清华大学出版社,1994

转贴于 中国论文下载中心 http://paper.studa.com

本文标签: 技术  

为您推荐

FOXPRO在三峡工程信息管理中的应用

三峡工程是世界级的巨型工程,专业门类多、技术复杂、信息管理工作量巨大,必须使用MIS系统对信息进行收集、整理、存储、统计、分析、制表。几年来,我们先后用FOXBASE、FOXPRO编制了《工资管理系统》、《土石方工程量计算程序》、《工程支付管理系统》、《文档管理系统

2011-11-09 16:03

论信息技术在外语教学中的应用

随着信息技术的发展, 计算机多媒体技术和网络被广泛地应用在外语教学中, 改变了传统外语教学模式。现代化外语教学提高了外语教学水平, 从而培养高素质的外语人才, 满足日益增长的社会需求。

2011-11-09 16:02

对计算科学与计算机发展的思考

本文从什么是计算说起, 通过对计算机的发展历史和人类对计算本质认识的回顾, 提出量子计算系统的发展和成熟, 并且提出了人类认识未知世界的规律:“计算工具不断发展—整体思维能力的不断增强—公理系统的不断扩大—旧的神谕被解决—新的神谕不断产生”不断循环。

2011-11-09 16:01

试析高职院校计算机专业教学的改革

提高高职院校计算机专业的教学水平,是高职院校计算机教育工作者应该深入思考的问题。通过分析高职院校计算机专业教育的教学特点,从如何培养计算机应用型人才的角度出发,对高职计算机专业教学中存在的不足进行了总结,并提出了几点改革设想。

2011-11-09 16:00

水利工程计算机应用现状与思考

水利水电工程地质又是所有这些不同行业的工程地质专业中涉及面最广声望最高问题最复杂任务最艰巨的专业,这是众所周知毋庸置疑的。水利工程计算机应用具有广阔的发展前景,无论是数值计算、数据库应用,还是专家系统、网络系统,都大有用武之地;特别是工程地质制图(主

2011-11-09 15:59

论高职计算机应用专业课程优化与整合

课程体系是教育教学目标实现的重要保证,高职计算机应用专业课程体系要以为社会培养计算机应用性专门人才这一根本任务而进行设计,本文对计算机应用专业课程设置进行一定的探讨。

2011-11-09 15:58

加载中...