DevC++ easyx实现地图拖动,超过屏幕大小的巨大地图的局部显示在屏幕的方法——用悬浮窗的原理来的实现一个视口

发布时间:2023年12月24日

原来是流量券还有,觉得放以前的帖子效果也就如此如此了。

想着看看整个新的帖子,看看还有啥靠谱的历史代码,能整点新意不?

于是就借着之前的悬浮窗代码往后看,发现这个代码的谱系还有几个后续分支,就挑了一个视口的来讲讲。

基本原理也是和悬浮窗一样。都是复制粘贴,直观讲就是还得拖动,更新屏幕,就逃不掉复制粘贴。

可要说起历史来,就是另一码事情了,原来是想着做一个像素游戏,复刻元气骑士,但是遇到的难题就是如何把超过屏幕大小的地图放进屏幕里?当时也就着一周也想不明白。然后由于还在搓绘图板,这个绘图板就是

EasyxDevC++开发地图编辑和游戏编辑代码工程文件附注释_哔哩哔哩_bilibili

这个视频里面展示的黄黄绿绿的黑白像素画板。

当时还得追溯到高三,想用Unity做游戏,结果教程看一半做不动了,出bug解决不了。unity自己内置的瓦片地图笔刷找不到了。就寄了。后来就想着能不能自己复刻一下这个笔刷功能,大一上查代码,无果,大一下学期才找到一个叫easyx的图形库,又翻了翻他的官网,才找见DevC++里内置好的easyx图形库的安装包。这才解决工具的问题。然后就是立即启动绘图板复刻,untiy既然能出来这个绘图板,就一定有方法实现。

当时是项目启动了两周,在解决悬浮窗问题上用来两周,2023年4月15号启动,4月22号解决小矩形网格生成绘制不重叠像素问题,然后就一阵沉寂,实际上就是4月22号实现功能后,去进行新的开发,就到了悬浮窗开发阶段,产生悬浮窗复制粘贴的想法用了一周,这一周就是想着图片复制粘贴,图片拖动之后,原来位置恢复原状的问题,其实就是怎么恢复原状这个问题卡住好久。悬浮图片是存到一个变量里不会更改,但是恢复原状,又要怎么恢复呢?当时穷举出来就是白色背景直接放白色矩形框,谁也看不出来是新的还是旧的,在后来遇到复杂图形了,要是沿用原位置覆盖,就得想办法获取贴图之前的原位置,就是想办法在粘贴新的图片之前拿到原位置的图,就是复制在将要粘贴的位置的图片,然后再把新的图片粘贴进去,然后才有了上一篇的悬浮窗的原理:“放置原像,扫描保存,粘贴新像”,然后改bug封装适配,又用了一周。

主要当时坑点就是不能确定怎么恢复原状,原来想着是整个程序背景刷新,但是后来觉得消耗内存太大,就想想能不能不刷新全部背景。结果就直接把原图覆盖了,大概就是改bug,测试想法,其中有个bug应该就是鼠标一点击,把图片给覆盖没了,实际上就是悬浮窗上画了一个矩形。然后才有了后续的,这个矩形除了白色还能怎么来的,矩形的覆盖顺序又是怎么确定的,大概穷举了几次,遇到几个图片覆盖没了的bug,才确定一个悬浮窗拖动循环的必须有的内容。

这才是解决拖动悬浮图片的问题的真相。

这次实现一个的视口效果如图

注意两幅图对角线的位置变化了

可以看出来背景就是大地图,这里变化的对角线就是大地图的一部分。

当时开发视口,是解决了图像拖动,不影响背景图片的问题,这次是图像拖动,解决目标图片显示内容的问题。

这个灵光一现,大概是计算拖动坐标得来的,就是鼠标向下移动10个像素,图片也向下移动十个像素,但是对着刚才的屏幕,可以看见,图片上方的像素多了,继续描述就是,我们看到图片上方的东西却多了,目测这多了的部分的像素,才意识到多出来的部分和鼠标拖动的部分应该相同。

实际上就是图片的展示部分反而向上移动了。比原来展示的部分高了10个像素。

这个时候,其实代码方面还是停留在悬浮窗阶段,只是这个悬浮窗的大小和屏幕一样大,相当于复制了一份地图,拖着大地图悬浮窗。

现在可以恍然大悟,,如果我缩小图片的采样范围,然后把采样的小图片贴到屏幕上,只通过这个小窗口来复制大地图部分,根据鼠标移动方向相反的距离继续复制大地图,粘贴到屏幕上,就相当于是有一个矩形相框,只能在相框内拖动相框下面的纸片,然后把相框截屏得到的东西贴到桌面上,如此类似于一个手术遮罩。

这样在映射到代码,就着原来悬浮窗的经验继续穷举方法,就是悬浮窗复制的部分变成大地图的部分,悬浮窗不再移动。采样旧图,不恢复原貌,因为根本没有移动,粘贴新图,刚好覆盖了原貌。

?不多说,先复制粘贴代码,该代码来自:15.1大地图移动.............

看和屏幕大小的悬浮窗。左键长按图片可以拖动。也可以通过对比看出来,这就是之前封装的.h文件,是先有的这个代码对比绘图板需求来封装,然后才复制粘贴,这样移植到.h文件,成为模块的。

#include<stdio.h>
#include<conio.h>
#include<graphics.h>
#include<windows.h>
struct pircle {
	IMAGE img2;
	IMAGE img3;
	const int orilx=0,orily=0;
	int nowlx=0,nowly=0;
	const int a=1500,h=1500;
//	原有图片的左上角坐标
	int m1x=0,m1y=0;

	int  putflag=0;
	int  drawflag=0;


} save;

void check(struct ExMessage m,struct pircle *save,IMAGE ak) {
	printf("putflag = %d\n",save->putflag);
	printf("%d %d\n",m.x,m.y);
	printf("%d %d\n",save->nowlx,save->nowly);

	if(save->putflag==true) {
		BeginBatchDraw();

		putimage(save->nowlx,save->nowly,&save->img3);

		save->nowlx=save->nowlx+m.x-save->m1x;
		save->nowly=save->nowly+m.y-save->m1y;
		save->m1x=m.x;
		save->m1y=m.y;
		getimage(&save->img3,save->nowlx,save->nowly,save->a,save->h);

		putimage(save->nowlx,save->nowly,&save->img2);

		EndBatchDraw();

//			一次绘图出来,没有屏闪了
	}
	switch(m.message) {

		case WM_LBUTTONDOWN:
			if(m.x>save->nowlx&&m.x<save->nowlx+save->a&&m.y>save->nowly&&m.y<save->nowly+save->h) {
				save->putflag=true;
//					启动批复制粘贴
//				getimage(&save->img2,save->nowlx,save->nowly,save->a,save->h);
				save->img2=ak;
				save->m1x=m.x;
				save->m1y=m.y;

			}
			break;
		case WM_LBUTTONUP:

			save->putflag=0;


			break;
	}

}

int main() {
	// 初始化绘图窗口
	initgraph(640, 480);
	IMAGE ak(1500,1500);
	IMAGE b;

	SetWorkingImage(&ak);

	setbkcolor(BLUE);
	cleardevice();
	setlinecolor(BLACK);
	rectangle(0,0,1499,1499);
	line(0, 0, 800, 1400);
	getimage(&b,0,0,150,150);

	SetWorkingImage();

	putimage(0,0,&ak);


	ExMessage m;
	while(1) {
		m=getmessage(EX_MOUSE);

		check(m,&save,ak);
	}
	_getch();
	closegraph();
}

?

然后缩小采样大小,在指定固定位置粘贴

?

?

?对比代码可以知道数字变化,正负变化。

但是还有个bug,就是像刮刮乐一样的两个对角线在视口里,需要完全拖出图片再拖回来才能清空旧的对角线

bug穷举分析,目前没来得及还不能确定。

?先放效果代码

#include<stdio.h>
#include<conio.h>
#include<graphics.h>
#include<windows.h>
struct pircle {
	IMAGE img2;
	IMAGE img3;
	const int orilx=0,orily=0;
	int nowlx=0,nowly=0;
	const int a=300,h=300;
//	原有图片的左上角坐标
	int m1x=0,m1y=0;

	int  putflag=0;
	int  drawflag=0;


} save;

void gameplace(IMAGE *ak) {
	SetWorkingImage(ak);
	setbkcolor(BLUE);
	cleardevice();
	setlinecolor(BLACK);
	rectangle(0,0,1499,1499);
	line(0, 0, 800, 1400);
	
}
struct showplace{
	int x=100;
	int y=100;
	int a=300;
	int h=300;
}show; 

void check(struct ExMessage m,struct pircle *save,IMAGE ak,struct showplace *show) {
	printf("putflag = %d\n",save->putflag);
	printf("%d %d\n",m.x,m.y);
	printf("%d %d\n",save->nowlx,save->nowly);

	if(save->putflag==true) {
		SetWorkingImage();
		getimage(&save->img3,show->x,show->y,show->a,show->h);
		BeginBatchDraw();
		SetWorkingImage(&ak);
		putimage(save->nowlx,save->nowly,&save->img3);

		save->nowlx=save->nowlx-(m.x-save->m1x);
		save->nowly=save->nowly-(m.y-save->m1y);
		save->m1x=m.x;
		save->m1y=m.y;
		getimage(&save->img2,save->nowlx,save->nowly,save->a,save->h);

//		putimage(save->nowlx,save->nowly,&save->img2);
		SetWorkingImage();
		putimage(show->x,show->y,&save->img2);
		EndBatchDraw();

//			一次绘图出来,没有屏闪了
	}
	switch(m.message) {

		case WM_LBUTTONDOWN:
			if(m.x>show->x&&m.x<show->x+show->a&&m.y>show->y&&m.y<show->y+show->h) {
				save->putflag=true;
//					启动批复制粘贴
				SetWorkingImage(&ak);
				getimage(&save->img2,save->nowlx,save->nowly,save->a,save->h);
//				save->img2=ak;
				save->m1x=m.x;
				save->m1y=m.y;
				SetWorkingImage();

			}
			break;
		case WM_LBUTTONUP:

			save->putflag=0;


			break;
	}

}

int main() {
	// 初始化绘图窗口
	initgraph(640, 480);
	IMAGE ak(1500,1500);
	IMAGE b;

gameplace(&ak);


	SetWorkingImage();
	putimage(0,0,&ak);
	setlinecolor(BLACK);
	

	rectangle(show.x,show.y,show.x+show.a,show.y+show.h);
	


	ExMessage m;
	while(1) {
		m=getmessage(EX_MOUSE);

		check(m,&save,ak,&show);
	}
	_getch();
	closegraph();
}

?

?

?没有破案,但是刚刚解决了bug,把其中一行代码注释掉

注意check函数里有一行注释,“奇妙的bug......"

?

#include<stdio.h>
#include<conio.h>
#include<graphics.h>
#include<windows.h>
struct pircle {
	IMAGE img2;
	IMAGE img3;
	const int orilx=0,orily=0;
	int nowlx=0,nowly=0;
	const int a=300,h=300;
//	原有图片的左上角坐标
	int m1x=0,m1y=0;

	int  putflag=0;
	int  drawflag=0;


} save;

void gameplace(IMAGE *ak) {
	SetWorkingImage(ak);
	setbkcolor(BLUE);
	cleardevice();
	setlinecolor(BLACK);
	rectangle(0,0,1499,1499);
	line(0, 0, 800, 1400);
	
}
struct showplace{
	int x=100;
	int y=100;
	int a=300;
	int h=300;
}show; 

void check(struct ExMessage m,struct pircle *save,IMAGE ak,struct showplace *show) {
	printf("putflag = %d\n",save->putflag);
	printf("%d %d\n",m.x,m.y);
	printf("%d %d\n",save->nowlx,save->nowly);

	if(save->putflag==true) {
		SetWorkingImage();
		getimage(&save->img3,show->x,show->y,show->a,show->h);
		BeginBatchDraw();
		SetWorkingImage(&ak);
//		putimage(save->nowlx,save->nowly,&save->img3);
//		取消注释,就有奇妙的bug实现了刮刮乐效果 
		save->nowlx=save->nowlx-(m.x-save->m1x);
		save->nowly=save->nowly-(m.y-save->m1y);
		save->m1x=m.x;
		save->m1y=m.y;
		getimage(&save->img2,save->nowlx,save->nowly,save->a,save->h);

//		putimage(save->nowlx,save->nowly,&save->img2);
		SetWorkingImage();
		putimage(show->x,show->y,&save->img2);
		EndBatchDraw();

//			一次绘图出来,没有屏闪了
	}
	switch(m.message) {

		case WM_LBUTTONDOWN:
			if(m.x>show->x&&m.x<show->x+show->a&&m.y>show->y&&m.y<show->y+show->h) {
				save->putflag=true;
//					启动批复制粘贴
				SetWorkingImage(&ak);
				getimage(&save->img2,save->nowlx,save->nowly,save->a,save->h);
//				save->img2=ak;
				save->m1x=m.x;
				save->m1y=m.y;
				SetWorkingImage();

			}
			break;
		case WM_LBUTTONUP:

			save->putflag=0;


			break;
	}

}

int main() {
	// 初始化绘图窗口
	initgraph(640, 480);
	IMAGE ak(1500,1500);
	IMAGE b;

gameplace(&ak);


	SetWorkingImage();
	putimage(0,0,&ak);
	setlinecolor(BLACK);
	

	rectangle(show.x,show.y,show.x+show.a,show.y+show.h);
	


	ExMessage m;
	while(1) {
		m=getmessage(EX_MOUSE);

		check(m,&save,ak,&show);
	}
	_getch();
	closegraph();
}

?

但是还没解决一开始屏幕上的对角线,只有点击之后,对角线才被覆盖。

按照上次悬浮窗的办法就是先覆盖一次。

复制粘贴就能跑,ctrl+右键实现拖动。刮刮乐效果解决方案还是注释掉check函数里某一行相同的代码。

#include<stdio.h>
#include<conio.h>
#include<graphics.h>
#include<windows.h>
struct pircle {
	IMAGE img2;
	IMAGE img3;
	const int orilx=0,orily=0;
	int nowlx=0,nowly=0;
	const int a=300,h=300;
//	原有图片的左上角坐标
	int m1x=0,m1y=0;

	int  putflag=0;
	int  drawflag=0;

} save;

struct showplace {
	int x=100;
	int y=100;
	const int a=300;
	const int h=300;

} show;
void gameplace(IMAGE *ak) {
	SetWorkingImage(ak);
	setbkcolor(BLUE);
	cleardevice();
	setlinecolor(BLACK);
	rectangle(0,0,1499,1499);
	line(0, 0, 800, 1400);

}


void check(struct ExMessage m,struct pircle *save,IMAGE ak,struct showplace *show) {
	printf("putflag = %d\n",save->putflag);
	printf("%d %d\n",m.x,m.y);
	printf("%d %d\n",save->nowlx,save->nowly);

	if(save->putflag==true) {
		SetWorkingImage();
		getimage(&save->img3,show->x,show->y,show->a,show->h);
		BeginBatchDraw();
		SetWorkingImage(&ak);
		putimage(save->nowlx,save->nowly,&save->img3);

		save->nowlx=save->nowlx-(m.x-save->m1x);
		save->nowly=save->nowly-(m.y-save->m1y);
		save->m1x=m.x;
		save->m1y=m.y;
		getimage(&save->img2,save->nowlx,save->nowly,save->a,save->h);

//		putimage(save->nowlx,save->nowly,&save->img2);
		SetWorkingImage();
		putimage(show->x,show->y,&save->img2);
		EndBatchDraw();

//			一次绘图出来,没有屏闪了
	}
	switch(m.message) {

		case WM_LBUTTONDOWN:
			if(m.x>show->x&&m.x<show->x+show->a&&m.y>show->y&&m.y<show->y+show->h&&m.ctrl) {
				save->putflag=true;
//					启动批复制粘贴
				SetWorkingImage(&ak);
				getimage(&save->img2,save->nowlx,save->nowly,save->a,save->h);
//				save->img2=ak;
				save->m1x=m.x;
				save->m1y=m.y;
				SetWorkingImage();

			}
			break;
		case WM_LBUTTONUP:

			save->putflag=0;


			break;
	}

}

void draw(struct ExMessage m,struct pircle *save,struct showplace *show,IMAGE *ak) {

//	printf("draw = %d\n",save->drawflag);
//		SetWorkingImage(ak);
	if(save->drawflag==true) {
//		save.drawflag=1;
		putpixel(m.x,m.y,RGB(255,155,4));
	}


	switch(m.message) {
		case WM_LBUTTONDOWN:
			if(m.x>show->x&&m.x<show->x+show->a&&m.y>show->y&&m.y<show->y+show->h) {
//
				printf("m.x = %d\tm.y = %d\t%d\t%d\n",m.x,m.y,save->m1x,save->m1y);
				save->drawflag=true;

//				printf("drawflag = %d\n",save->drawflag);
			}

			break;

		case WM_LBUTTONUP:

			save->drawflag=false;
//			printf("%d\n",save->drawflag);
			break;
	}

}


int main() {
	// 初始化绘图窗口
	initgraph(640, 480);
	IMAGE ak(1500,1500);
	IMAGE b(300,300);

	gameplace(&ak);
	getimage(&b,0,0,300,300);

	SetWorkingImage();
	putimage(0,0,&ak);
	putimage(100,100,&b);
	setlinecolor(BLACK);
	

	rectangle(show.x,show.y,show.x+show.a,show.y+show.h);



	ExMessage m;

	while(1) {
		m=getmessage(EX_MOUSE);

		check(m,&save,ak,&show);
		draw(m,&save,&show,&ak);
	}
	_getch();
	closegraph();
}

文章来源:https://blog.csdn.net/a1309602336/article/details/135179205
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。