【游戏客户端】大话slg玩法架构(二)背景地图

 【游戏客户端】大话slg玩法架构(二)背景地图

      大家好,我是Lampard家杰~~ 今天我们继续给大家分享SLG玩法的实现架构,关于SLG玩法的介绍可以参考这篇上一篇文章:【游戏客户端】制作率土之滨Like玩法

       PS:和之前一样,本文也只是分享实现思路,并不会贴具体的代码和资源哟

 (一)架构总览

      SLG玩法的实现思路可以划分为四个部分,分别是滚动容器基类的搭建,背景大地图的实现,建筑的生成与刷新,以及玩法优化预加载相关

      上一篇文章我们分享了滚动基类的实现:【游戏客户端】大话slg玩法架构(一)滚动基类 ,现在我们接着分享背景地图的实现逻辑

      完整全文跳转:【游戏客户端】大话slg玩法架构(全)

(二)背景地图的实现

(1)大地图资源的输出方式

      通过上一篇文章我们知道,我们需要一张大图来铺满滚动容器的InnerContainer来实现背景地图的效果。那这个时候怎么输出这个资源成了一个需要考量的地方

      直接输出一个10000 * 10000的png吗?那加载的时候不得卡鼠。所以必须要采取分块的方式,比如你可以把一块地图的大小设置成1000 * 1000,那么我们就把一张大地图分成了100张小地图

      第二个问题就是,那这个原始的10000*10000分块前的资源要怎么输出,如果单靠美术同学去画一张完整的图工作量是很大的,这个时候我们就可以采取代码重复拼接的方式,比如美术只输出2000 * 2000 (4块资源)组成的地图,我们在代码里每隔2000像素就重复利用一次资源

function getBgRes(Idx, Idy)
    local ResTbl = {
        [1] = {[1] = 资源1, [2] = 资源2},
        [2] = {[1] = 资源3, [2] = 资源4},       
    }

    local ResX = Idx % 2 + 1
    local ResY = Idy % 2 + 1
    return ResTbl[ResX][ResY]
end

    只要处理好资源拼接时候的缝隙,那么我们一方面可以降低大地图的实现成本,另一方面可以减少资源的读取,降低资源缓存

(2)地图块的控件复用

      有了地图的生成方式,后面只需要像摆积木一样一块块摆在滚动容器上就可以了。那么不知道面对这100个地图块,大家会选择怎么创建呢?

      如果直接一个for循环,那么一进游戏就卡成ppt了。有经验的同学可能会采取异步加载的方式分帧去创建,这样确实可以,但是100个地图块生成完之后,光这些地图就已经占据了几百M内存(苹果2G内存瑟瑟发抖)

      而且,哪怕是采取了异步创建的方法,在创建的流程中也是感受到明显卡顿的。其实玩家能看到的无非就是屏幕上那几块地图,哪怕玩家快速的上下左右滑动,我们也只需要多生成一屏的地图就可以了,这样不过十来块的数量。因此我们不需要把100个地图全生成,只需要生成十来个实际用到的,然后监听玩家的滑动,更新当前所视范围地图的纹理并重设它们的位置就可以了

-- 释放
for x = 1, 10 do
    for y = 1, 10 do
        local IsInView = 判断一下这一块地图需不需要显示
        if not IsInView and self.Map[x][y] then
            local Map = self.Map[x][y]
            Map:setFree()
            self.Map[x][y] = nil
        end
    end
end

-- 生成
for x = 1, 10 do
    for y = 1, 10 do
        local IsInView = 判断一下这一块地图需不需要显示
        if IsInView then
            if self.Map[x][y] then
                -- 存在这个地块就设置可视就行
                local Map = self.Map[x][y]
                Map:setVisible(true)
            else
                -- 不存在则找一个空闲的地块控件重设信息
                local Map = getFreeMap()
                self.Map[x][y] = Map 
                Map:resetInfo(x, y)
            end
        end
    end
end

(3)更新地图块的时机

      那什么时候执行这段地图更新代码呢?一开始我是直接每一帧都做一个判断,后来发现这样很多余而且比较耗CPU,其实我们需要监听玩家的滑动操作,滑动距离超过一个地图块的长/高的时候再执行就可以了

	self.SV.OnScrolling  = function()
		local CurPos = 获取当前位置
		if not self.SV.LastPosX or not  self.SV.LastPosY then
			self.SV.LastPosX = CurPos.Idx
			self.SV.LastPosY = CurPos.Idy
		end

		if 超过滑动距离 then
			self.SV.LastPosX = CurPos.Idx
			self.SV.LastPosY = CurPos.Idy
            执行更新地图的方法
		end
	end

      好啦~~地图块的生成介绍大概就到这里,下一篇文章会介绍这个建筑控件的实现

      感谢阅读,记得点赞和关注!!!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Lampard杰

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值