c/c++语言开发共享基于Matlab制作伪3D第一视角迷宫小游戏

游戏效果使用键盘上方向键↑向前移动使用键盘左右方向键调整← →朝向游戏原理说明原理很简单,如效果图所示,主要就是以角色视角方向发射大量的直线模拟视线,并计算直线与墙壁交点

游戏效果

基于Matlab制作伪3D第一视角迷宫小游戏

基于Matlab制作伪3D第一视角迷宫小游戏

使用键盘上方向键↑向前移动

使用键盘左右方向键调整← →朝向

游戏原理说明

原理很简单,如效果图所示,主要就是以角色视角方向发射大量的直线模拟视线,并计算直线与墙壁交点,获取每一条视线方向下,角色到墙壁的距离,然后根据近大远小的原理绘制不同长度的竖向直线模拟墙壁。

第一代程序计算交点方法

第一代程序是使用的polyshape对象(二维多边形对象)制作墙壁,polyshape对象重载了函数intersect,可以直接求出直线与多边形的交集,程序编写更加简洁方便,给个检测直线和多边形交点的官方例子:

创建一个矩形多边形和一个线段:

poly1=polyshape([0 0 1 1],[1 0 0 1]);  lineseg=[-1 -1;1.5 1.5];  

计算该多边形与线段的交集,并确定线段的哪些部分在多边形的内部,哪些在其外部。

[in,out] = intersect(poly1,lineseg);  plot(poly1)  hold on  plot(in(:,1),in(:,2),'b',out(:,1),out(:,2),'r')  legend('polygon','inside','outside','location','northwest')  

基于Matlab制作伪3D第一视角迷宫小游戏

我们可以把内部第一个点和最后一个点看作与多边形边缘的交点:

intersectpnt=[in(1,:);in(end,:)]  

intersectpnt =
0 0
1 1

但是!:这样的计算方法较缓慢,而且polyshape对象推出时间较晚(至少需要r2017b),

于是该方案被舍弃!!

于是该方案被舍弃!!

于是该方案被舍弃!!

第二代程序计算交点方法

假设角色当前位置为p0=(x0, y0) ,角色当前面向方向为θ0,将所有的墙壁边缘离散成点集合,其点到角色位置方向向量为:

基于Matlab制作伪3D第一视角迷宫小游戏

每个视线方向向量

基于Matlab制作伪3D第一视角迷宫小游戏

做内积:

基于Matlab制作伪3D第一视角迷宫小游戏

那么inn矩阵的每个数值都是角色到一个墙壁点方向向量与角色某一视线方向做内积,要考虑到视线为单方向射线,因此将内积为负数的值置为无穷大:inn(inn<0)=inf,之后因为明显和视线越垂直算出的距离越短,但这不是我们想要的,因此要添加新的约束,就是内积值和实际的点到角色的距离差值要小于一定阈值1e-5,添加这个约束后,找到inn矩阵每一列的最小值即可找到每一视线方向最近的墙壁点。

距离转换为线段长度

使用的如下公式:

基于Matlab制作伪3D第一视角迷宫小游戏

其中α为常数,l为墙壁点到角色距离,θ为视线和角色面朝方向的夹角。

完整代码

function maze2_5d_v2  % @author : slandarer  % @公众号 : slandarer随笔  % @知乎   : hikari  help maze2_5d    %% ========================================================================  % figure窗口创建  fig=figure();  fig.position=[50,60,1200,600];  fig.name='maze 2.5d by slandarer';  fig.numbertitle='off';  fig.menubar='none';  % 俯视图axes坐标区域  ax2d=axes('parent',fig);  ax2d.xtick=[];ax2d.xcolor='none';  ax2d.ytick=[];ax2d.ycolor='none';  ax2d.xlim=[0,15];  ax2d.ylim=[0,15];  ax2d.color=[0,0,0];  ax2d.position=[0,0,1/2,1];  hold(ax2d,'on')  % 伪3d图axes坐标区域  ax3d=axes('parent',fig);  ax3d.xtick=[];ax2d.xcolor='none';  ax3d.ytick=[];ax2d.ycolor='none';  ax3d.xlim=[0,10];  ax3d.ylim=[0,10];  ax3d.color=[0,0,0];  ax3d.position=[1/2,0,1/2,1];  hold(ax3d,'on')  %% ========================================================================  % 左侧俯视地图初始化  mazemat=[1 1 1 1 1 1 1 1 1 1 1 1 1 1 1;           1 0 0 0 0 0 1 0 1 0 1 0 0 0 1;           1 1 1 1 1 0 1 0 0 0 1 1 1 0 1;           1 0 0 0 1 0 1 0 1 0 1 0 0 0 1;           1 0 1 1 1 0 0 0 1 0 0 0 1 1 1;           1 0 0 0 0 2 1 1 1 0 1 1 1 0 1;           1 0 1 0 1 1 1 0 1 0 0 0 1 0 1;           1 0 1 0 0 0 0 0 1 1 1 0 1 0 1;           1 0 1 1 1 1 1 1 1 0 1 0 0 0 1;           1 0 1 0 0 0 0 0 0 0 1 0 1 0 1;           1 0 1 1 1 1 1 0 1 0 1 0 1 0 1;           1 0 1 0 1 0 1 0 1 0 0 0 1 0 1;           1 0 1 0 1 0 1 0 1 1 1 1 1 0 1;           1 0 0 0 0 0 0 0 1 0 0 0 0 0 1;           1 1 1 1 1 1 1 1 1 1 1 1 1 1 1];  [rowlist,collist]=find(mazemat==1);  sqx=[-1;0;0;-1];sqy=[-1;-1;0;0];  lslist=linspace(-1,0,250)';  nmlist=-ones(size(lslist));  fllist=[[lslist,nmlist];[lslist,nmlist.*0];[nmlist,lslist];[nmlist.*0,lslist]];    block.pntset=zeros(2,0);  for n=1:length(rowlist)       fill(ax2d,sqx+collist(n),sqy+size(mazemat,1)+1-rowlist(n),[1,1,1].*0.9);      block.pntset=[block.pntset;fllist+repmat([collist(n),size(mazemat,1)+1-rowlist(n)],[size(fllist,1),1])];  end  % -------------------------------------------------------------------------  % 角色创建  [trow,tcol]=find(mazemat==2);  rolep.xpos=tcol-0.5;  rolep.ypos=size(mazemat,1)+0.5-trow;  rolep.theta=pi/2;  rolep.trix=cos([pi/3,pi,-pi/3]).*0.15;  rolep.triy=sin([pi/3,pi,-pi/3]).*0.15;  [tx,ty]=rotatedata(rolep.trix,rolep.triy,rolep.theta);  rolep.pfill=fill(ax2d,tx+rolep.xpos,ty+rolep.ypos,[1,1,1]);  rolep.pshape=polyshape(tx+rolep.xpos,ty+rolep.ypos);  rolep.viewrange=size(mazemat,1)*sqrt(2);  %% ========================================================================  % 线条创建  % plot(ax3d,[1,1].*10.*i./length(thetalistv),[5-tlen/2,5+tlen/2],...  % 'linewidth',1.5,'color',[1,1,1]./10.*tlen,'tag','blockline');  % plot(ax2d,[rp.xpos,rp.xpos+cos(thetalist(i))*abs(minlist(i))],[rp.ypos,rp.ypos+sin(thetalist(i))*abs(minlist(i))])  linenum=300;    for n=1:linenum      pline.plotline3(n)=plot(ax3d,[1,1].*10.*n./linenum,[-1,-1],'linewidth',1.5);      pline.plotline2(n)=plot(ax2d,[-1,-1],[-1,-1],'color',lines(1));  end      draw3d(rolep,block,pline,linenum)  %% ========================================================================  % 角色移动函数  set(fig,'keypressfcn',@key)  function key(~,event)      %按键函数      switch event.key          case 'uparrow'              rolep.xpos=rolep.xpos+cos(rolep.theta).*.2;              rolep.ypos=rolep.ypos+sin(rolep.theta).*.2;          case 'leftarrow'              rolep.theta=rolep.theta+pi/20;          case 'rightarrow'              rolep.theta=rolep.theta-pi/20;      end      [tx,ty]=rotatedata(rolep.trix,rolep.triy,rolep.theta);      rolep.pfill.xdata=tx+rolep.xpos;      rolep.pfill.ydata=ty+rolep.ypos;      rolep.pshape=polyshape(tx+rolep.xpos,ty+rolep.ypos);      draw3d(rolep,block,pline,linenum)  end  %% ========================================================================  % 视角检测及伪3d图绘制      function draw3d(rp,bk,pline,ln)      % delete(findobj('tag','blockline'))      thetalistv=linspace(pi/3,-pi/3,ln);      thetalist=thetalistv+rp.theta;      % 内积法计算距离      cutoff=1e-5;      coslist=cos(thetalist);      sinlist=sin(thetalist);      veclist=bk.pntset-[rp.xpos,rp.ypos];      dismat=veclist*[coslist;sinlist];      dismat(dismat<0)=inf;      normlist=vecnorm(veclist')';      diffmat=abs(dismat-repmat(normlist,[1,size(dismat,2)]));      dismat(diffmat>cutoff)=inf;      minlist=min(abs(dismat));      % 图像重绘      for i=1:length(thetalist)          tlen=10/abs(minlist(i))/abs(cos(thetalistv(i))).*0.6;tlen(tlen>10)=10;          pline.plotline3(i).color=[1,1,1]./10.*tlen;          pline.plotline3(i).ydata=[5-tlen/2,5+tlen/2];          pline.plotline2(i).xdata=[rp.xpos,rp.xpos+cos(thetalist(i))*abs(minlist(i))];          pline.plotline2(i).ydata=[rp.ypos,rp.ypos+sin(thetalist(i))*abs(minlist(i))];      end  end  %% ========================================================================  % 数据旋转角度  function [x,y]=rotatedata(x,y,theta)      rotatemat=[cos(theta),-sin(theta);sin(theta),cos(theta)];      xy=rotatemat*[x;y];      x=xy(1,:);y=xy(2,:);  end  end  

基于Matlab制作伪3D第一视角迷宫小游戏

以上就是基于matlab制作伪3d第一视角迷宫小游戏的详细内容,更多关于matlab迷宫游戏的资料请关注<计算机技术网(www.ctvol.com)!!>其它相关文章!

需要了解更多c/c++开发分享基于Matlab制作伪3D第一视角迷宫小游戏,都可以关注C/C++技术分享栏目—计算机技术网(www.ctvol.com)!

本文来自网络收集,不代表计算机技术网立场,如涉及侵权请联系管理员删除。

ctvol管理联系方式QQ:251552304

本文章地址:https://www.ctvol.com/c-cdevelopment/1120912.html

(0)
上一篇 2022年7月13日
下一篇 2022年7月13日

精彩推荐