|
摘 要 该文论述了在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(&pale tte[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. dvnews
上一篇 下一篇
|