技术开发 频道

开发Silverlight游戏教程:牵引式地图移动

    【IT168技术文档】

     转载自:深蓝色右手的博客

    在前面诸多的章节里,我就地图构造的实现做了讲解,至此还遗留着一个关键问题:在游戏中是角色在移动还是地图在移动?它们之间的移动(位移)关系是如何实现的?

    那么在接下来的章节中我将围绕这两个问题进行详细的分析解说。

    首先,还得从游戏模式开始说起。目前2D俯视游戏中以即使战略、SLG、RPG(ARPG)等类型的游戏为主流。在即时战略、SLG大地图中,地图的移动原理是:当鼠标处于游戏窗口的8个边缘时,地图即开始移动,我暂且称之为牵引式地图移动模式。
 

     如上图,我们可以打这样一个比方:将游戏窗口比做我们的摄象机(上图中的Windows13窗口),在地图世界里不断的取景,我们从摄象机中看的只有摄象机镜头(游戏窗口)中能看到的区域(其他虚的地图部分窗口中是无法显示的)。但是游戏窗口又相当于是用支架固定着不能移动的镜头,那该如何才能看到景色的各个部位呢?那当然只有去移动背景地图图片让我们需要的景色部分呈现在窗口中。因此,根据上图描述的原理,当鼠标进入这8个区域(蓝色和棕色区域)时即触发地图的移动。

    为更方便大家的理解,我以窗口中左边那块蓝色区域为例:当我的鼠标在游戏窗口内往左移动快接近边缘时,此时地图图片就开始反向向右进行一定速度或加速度移动;同样的,地图中的所有对象均跟着地图图片以同样的方向与速度进行移动,这样给我们视觉上产生一种感觉:我们在通过鼠标牵引游戏窗口向左方去探寻地图及物体对象。有的朋友就要问了:如果一张地图上有1000个对象,难道我每10毫秒都要去移动这1000个对象吗?这样性能上说是完全不科学的!对,在实际开发中如果一张地图拥有大量物体对象的话,我们肯定不会这样做(如果地图是小地图,或者物体不多,这样做是完全可行的,并且更容易实现)。在理解了这个原理后,我们看看在WPF/Silverlight中是如何进行这些操作的。首先需要做的就在地图移动的时候,根据地图移动方向时时(在界面刷新线程CompositionTarget或间隔为10毫秒的DispatcherTimer中)通过Foreach高性能的对所有物体对象(Spirit)的X,Y坐标进行修改;而什么时候才需要将这些物体对象显示出来呢?判断当前游戏窗口中心点对应的地图坐标点;并以该点为中点(圆心)进行一个矩形范围或半径为R的圆形范围(下文我简称为地图中心范围)搜索:如果某物体对象的X,Y坐标在此范围内则动态将它的显示实体加载它进入窗体画布(Carrier.Children.Add(Spirit)),然后再将之布局到它对应的X,Y坐标位置上(Canvas.setLeft(Spirit,X);Canvas.setTop(Spirit,Y);),并且继续根据地图的移动而移动(时时修改Canvas.setLeft(),Canvas.setTop());同样的,在地图移动中地图中心坐标是时时改变的,如果某些物体对象的X,Y坐标超出了地图中心范围,那么我们就将之从窗体画布中移除掉(Carrier.Children.Remove(Spirit)),此时这些物体对象相当于重新回到了等候显示的状态,它们的X,Y坐标同样在后台线程中时时更改,只要某个时候当地图中心再度出现在它们的附近时,它们又会重复以上的步骤再显示出来。

    大致原理有了,如何通过代码来具体实现呢?

    这里我提供两种方法:

    第一种方法为通过载体来实现地图移动。具体为首先向游戏窗体中添加8个完全透明的滚动介质(就好比图中那8块区域,其中4个蓝的,4个棕的)分别布局在地图边缘的8个位置上(它们相对于游戏窗体来说永远是不动的),然后在界面线程中时时判断鼠标是否悬停在它们中的某个上从而进行相应的地图移动。

    这里我以正下方的滚动介质为例,这样来创建它:

 int scrollspeed = 5; //定义滚动速度

Rectangle roller = new Rectangle(); //创建滚动介质

private void InitRoller() {

roller.Width = 800;

roller.Height = 20;

roller.Opacity = 0.3;

roller.Fill = new SolidColorBrush(Colors.Blue);

Carrier.Children.Add(roller);

Canvas.SetZIndex(roller, 10001);

Canvas.SetTop(roller, 490);

}

    这里为了演示需要,我将它的透明度暂且设置为0.3而不是0,目的是为方便大家可以看到它。接下来我们就需要在CompositionTarget的Timer_Tick事件中时时判断鼠标是否在它上面:

 private void Timer_Tick(object sender, EventArgs e) {

//时时判断如果鼠标停留在了该滚动介质上,则地图相应滚动

if (roller.IsMouseOver) {

Canvas.SetTop(Map, Canvas.GetTop(Map) - scrollspeed);

}

}

     这样就创建好了这8个区域中的其中一个(正下方区域),其他7个的创建和实现方法依次类推,很简单就不累述了。好了,我们按CTRL+F5来测试一下,当鼠标停留在窗体正下方的那个蓝色长方形区域时,地图会非常平滑的向上移动,这样就实现了我们视觉上的窗口向下取景效果。

原文地址:http://www.cnblogs.com/alamiye010/archive/2009/06/19/1507095.html

0
相关文章