本游戏为本科的毕业设计,开贴旨在用来用来记录。目前所写的角色扮演游戏(RPG)已经有了一个成品,但是部分实现上不便于扩展,故进行第三次重构。

本游戏基于SDL_Engine进行开发(简化版的cocos2d-x)。

首先需要构建框架。

本游戏采用MVC设计模式,GameScene为MVC中的Control控制器,主要起到负责全局的管理,逻辑处理,事件接收与分发等。为了便于以后lua脚本的接入,故GameScene为单例类。先看看代码吧。

GameScene.h

#ifndef __GameScene_H__
#define __GameScene_H__
#include <string>

#include "SDL_Engine/SDL_Engine.h"

using namespace std;
USING_NS_SDL;

class MapLayer;

class GameScene : public Scene
{
private:
	static GameScene* s_pInstance;
public:
	static GameScene* getInstance();
	static void purge();
private:
	MapLayer* m_pMapLayer;
public:
	static const int CHARACTER_LOCAL_Z_ORDER = 9999;//需要比tmx地图总图块大
private:
	GameScene();
	~GameScene();
	bool init();
	bool initializeMapAndPlayers();
public:
	//改变场景
	void changeMap(const string& mapName, const Point& tileCoodinate);
	//设置视图中心点
	void setViewPointCenter(const Point& position, float duration = 0.f);
};
#endif

目前的GameScene类中的内容还是比较少的,仅仅包含了一个MapLayer指针和CHARACTER_LOCAL_Z_ORDER,这个静态成员主要是为了处理角色和tmx地图的图块之间的遮挡关系的,它会在MapLayer类中和GameScene中的设置角色的localZOrder中等用到。

GameScene.cpp

#include "GameScene.h"
#include "MapLayer.h"
#include "StaticData.h"
#include "DynamicData.h"
//static
GameScene* GameScene::s_pInstance = nullptr;

GameScene* GameScene::getInstance()
{
	//仅在场景在运行时才会分配新的对象
	if (s_pInstance == nullptr && Director::getInstance()->isRunning())
	{
		s_pInstance = new GameScene();
		s_pInstance->init();
	}

	return s_pInstance;
}

void GameScene::purge()
{
	SDL_SAFE_RELEASE_NULL(s_pInstance);
}

GameScene::GameScene()
	:m_pMapLayer(nullptr)
{
}

GameScene::~GameScene()
{
	StaticData::purge();
	DynamicData::purge();
}

bool GameScene::init()
{
	m_pMapLayer = MapLayer::create();
	this->addChild(m_pMapLayer);

	//初始化地图和角色
	this->initlizeMapAndPlayers();

	return true;
}

bool GameScene::initializeMapAndPlayers()
{
	//获取地图
	auto dynamicData = DynamicData::getInstance();
	//TODO:暂时使用存档1
	dynamicData->initializeSaveData(1);

	auto mapFilePath = dynamicData->getMapFilePath();
	auto tileCooridinate = dynamicData->getTileCoordinateOfPlayer();
	//改变地图
	this->changeMap(mapFilePath, tileCooridinate);

	return true;
}

void GameScene::changeMap(const string& mapName, const Point &tileCoodinate)
{
	//改变当前地图
	m_pMapLayer->clear();
	m_pMapLayer->init(mapName);
	//改变当前中心点
	this->setViewPointCenter(tileCoodinate);
}

void GameScene::setViewPointCenter(const Point& position, float duration)
{
	Size visibleSize = Director::getInstance()->getVisibleSize();
	const int tag = 10;
	//地图跟随点移动
	float x = (float)MAX(position.x, visibleSize.width / 2);
	float y = (float)MAX(position.y, visibleSize.height / 2);
	//获取地图层的地图
	auto tiledMap = m_pMapLayer->getTiledMap();

	auto tileSize = tiledMap->getTileSize();
	auto mapSize = tiledMap->getMapSize();
	auto mapSizePixel = Size(tileSize.width * mapSize.width, tileSize.height * mapSize.height);
	//不让显示区域超过地图的边界
	x = (float)MIN(x, (mapSizePixel.width - visibleSize.width / 2.f));
	y = (float)MIN(y, (mapSizePixel.height - visibleSize.height / 2.f));
	//实际移动的位置
	Point actualPosition = Point(x, y);
	//屏幕中心位置坐标
	Point centerOfView = Point(visibleSize.width / 2, visibleSize.height / 2);

	Point delta = centerOfView - actualPosition;

	Action* action = nullptr;

	//地图运动
	if (duration < FLT_EPSILON)
	{
		action = Place::create(delta);
	}
	else
	{
		action = MoveTo::create(duration, delta);
	}
	action->setTag(tag);

	if (tiledMap->getActionByTag(tag) != nullptr)
	{
		tiledMap->stopActionByTag(tag);
	}
	tiledMap->runAction(action);
}

changeMap函数用的很是广泛,其主要功能就是起到切换地图的功能。需要注意的是,切换地图只是更换了MapLayer对象中的m_pTiledMap指针的指向,GameScene对象中的MapLayer对象是不变的。而setViewPointCenter函数的主要功能则是是让界面跟随角色进行移动,以保证角色能一直显示在屏幕中间,这个函数不依赖于任何一个角色,故其用途广泛。

MapLayer.h

#ifndef __MapLayer_H__
#define __MapLayer_H__
#include <string>
#include "SDL_Engine/SDL_Engine.h"

using namespace std;
USING_NS_SDL;

class MapLayer : public Layer
{
private:
	TMXTiledMap* m_pTiledMap;
	//地图文件路径
	string m_filepath;
public:
	MapLayer();
	~MapLayer();
	CREATE_FUNC(MapLayer);
	static MapLayer* create(const string& filepath);

	bool init();
	bool init(const string& filepath);
	//清除
	void clear();
	//获取该地图的文件路径
	string& getFilepath();
	//获取地图名字
	string getMapName() const;
	//获取碰撞层
	TMXLayer* getCollisionLayer() const;
	//获取脚本对象
	TMXObjectGroup* getScriptObjectGroup() const;
	TMXTiledMap* getTiledMap() const;
private:
	void resetLocalZOrderOfTile();
};
#endif

MapLayer.cpp

#include "MapLayer.h"
#include "StaticData.h"
#include "GameScene.h"

MapLayer::MapLayer()
	:m_pTiledMap(nullptr)
{
}

MapLayer::~MapLayer()
{
}

MapLayer* MapLayer::create(const string& filepath)
{
	auto layer = new MapLayer();

	if (layer && layer->init(filepath))
		layer->autorelease();
	else
		SDL_SAFE_DELETE(layer);

	return layer;
}

bool MapLayer::init()
{
	return true;
}

bool MapLayer::init(const string& filepath)
{
	m_filepath = filepath;

	m_pTiledMap = TMXTiledMap::create(filepath);
	this->addChild(m_pTiledMap);

	this->resetLocalZOrderOfTile();

	return true;
}

void MapLayer::clear()
{
	if (m_pTiledMap != nullptr)
	{
		m_pTiledMap->removeFromParent();
		m_pTiledMap = nullptr;
	}
}

string& MapLayer::getFilepath()
{
	return m_filepath;
}

string MapLayer::getMapName() const
{
	auto text = m_pTiledMap->getPropertyForName("name").asString();

	return text;
}

TMXLayer* MapLayer::getCollisionLayer() const
{
	auto layerName = STATIC_DATA_STRING("collision_layer_name");

	return static_cast<TMXLayer*>(m_pTiledMap->getLayer(layerName));
}

TMXObjectGroup* MapLayer::getScriptObjectGroup() const
{
	auto layerName = STATIC_DATA_STRING("script_layer_name");

	return m_pTiledMap->getObjectGroup(layerName);
}

TMXTiledMap* MapLayer::getTiledMap() const
{
	return m_pTiledMap;
}

void MapLayer::resetLocalZOrderOfTile()
{
	//保存所有优先级为2的图块
	vector<int> tileIds;
	auto tilesets = m_pTiledMap->getTilesets();

	for (auto tileset : tilesets)
	{
		auto& properties = tileset->getProperties();
		
		for(auto itMap = properties.cbegin(); itMap != properties.cend(); itMap++)
		{
			auto id = itMap->first;
			auto& valueMap= itMap->second;

			auto it = valueMap.find("priority");

			if (it != valueMap.cend() && it->second.asInt() == 2)
				tileIds.push_back(id + tileset->firstGrid);
		}
	}
	//设置优先级为2的图块localZOrder
	auto& children = this->getCollisionLayer()->getChildren();

	for (auto child : children)
	{
		//获取名字
		auto name = child->getName();

		if (name.empty())
			continue;
		
		int id = SDL_atoi(name.c_str());
		//设置localZOrder
		if (find(tileIds.begin(),tileIds.end(),id) != tileIds.end())
		{
			child->setLocalZOrder(child->getLocalZOrder() + GameScene::CHARACTER_LOCAL_Z_ORDER);
		}
	}
}

值得一提的是resetLocalZOrderOfTile函数,其主要功能就是获取m_tiledMap中引用的图块中的存在property属性,且为2的图块,并进行重新设置其localZOrder,以保证和角色的正确遮挡,比如如下4个图块

rpg 游戏架构 rpg构造_rpg 游戏架构

上面的两个图块的属性如右

rpg 游戏架构 rpg构造_#include_02

角色在是可以通过上面两个图块的,且角色应该被该树遮挡。这里规定,priority为0,则不可通过;为1则人物遮挡该图块;为2则人物被该图块遮挡。人物的localZOrder一般为CHARACTER_LOCAL_Z_ORDER,故一般情况下,人物的绘制会晚于优先级为0的图块,会早于优先级为2的图块(主要涉及到Node中的成员_localZOrder)。

接下来谈一谈MVC中的Model模型。先谈谈静态数据

StaticData.h

#ifndef __StaticData_H__
#define __StaticData_H__
#include <string>
#include "SDL_Engine/SDL_Engine.h"

using namespace std;
USING_NS_SDL;
//定义一些常用的宏
#define STATIC_DATA_PATH "data/static_data.plist"
/*简化使用*/
#define STATIC_DATA_STRING(key) (StaticData::getInstance()->getValueForKey(key)->asString())
#define STATIC_DATA_INT(key) (StaticData::getInstance()->getValueForKey(key)->asInt())
#define STATIC_DATA_FLOAT(key) (StaticData::getInstance()->getValueForKey(key)->asFloat())
#define STATIC_DATA_BOOLEAN(key) (StaticData::getInstance()->getValueForKey(key)->asBool())
#define STATIC_DATA_POINT(key) (StaticData::getInstance()->getPointForKey(key))
#define STATIC_DATA_ARRAY(key) (StaticData::getInstance()->getValueForKey(key)->asValueVector())
#define STATIC_DATA_TOSTRING(key) (StaticData::getInstance()->toString(key))

class StaticData : public Object
{
private:
	static StaticData* s_pInstance;
public:
	static StaticData* getInstance();
	static void purge();
private:
	//键值对
	ValueMap m_valueMap;
private:
	StaticData();
	~StaticData();
	bool init();
public:
	/**
	@brief 根据键获取值
	@key 要查询的键
	@return 返回的值,如果不存在对应的值,则返回空Value
	*/
	Value* getValueForKey(const string& key);
};
#endif

StaticData.cpp

#include "StaticData.h"

StaticData* StaticData::s_pInstance = nullptr;

StaticData* StaticData::getInstance()
{
	if (s_pInstance == nullptr)
	{
		s_pInstance = new StaticData();
		s_pInstance->init();
	}
	return s_pInstance;
}

void StaticData::purge()
{
	SDL_SAFE_RELEASE_NULL(s_pInstance);
}

StaticData::StaticData()
{
}

StaticData::~StaticData()
{
}

bool StaticData::init()
{
	//读取文件并保存键值对
	m_valueMap = FileUtils::getInstance()->getValueMapFromFile(STATIC_DATA_PATH);

	return true;
}

Value* StaticData::getValueForKey(const string& key)
{
	auto iter = m_valueMap.find(key);

	if(iter != m_valueMap.end())
		return &iter->second;

	return nullptr;
}

静态数据即在程序运行期间不会发生改变的数据,其主要功能是提供一些外部文件的读取,比如static_data.plist文件的读取以及以后的角色升级属性文件等也由StaticData类提供。

DynamicData.h

#ifndef __DynamicData_H__
#define __DynamicData_H__
#include <string>
#include "SDL_Engine/SDL_Engine.h"

using namespace std;
USING_NS_SDL;

class DynamicData : public Object
{
private:
	static DynamicData* s_pInstance;
public:
	static DynamicData* getInstance();
	static void purge();
private:
	DynamicData();
	~DynamicData();
private:
	//存档
	ValueMap m_valueMap;
	//存档名称
	string m_fullpath;
	//是否第一次进入游戏
	bool m_bFirstGame;
	//金币数目
	int m_goldNumber;
	//出售比例
	float m_sellRatio;
public:
	//读取存档
	bool initializeSaveData(int idx);
	//获取角色map
	ValueMap& getValueMapOfCharacter();
	//获取地图文件路径
	string getMapFilePath() const;
	//获取角色对应位置
	Point getTileCoordinateOfPlayer() const;
};
#endif

DynamicData.cpp

#include "DynamicData.h"

DynamicData* DynamicData::s_pInstance = nullptr;

DynamicData* DynamicData::getInstance()
{
	if (s_pInstance == nullptr)
	{
		s_pInstance = new DynamicData();
		s_pInstance->init();
	}
	return s_pInstance;
}

void DynamicData::purge()
{
	SDL_SAFE_RELEASE_NULL(s_pInstance);
}

DynamicData::DynamicData()
	:m_bFirstGame(true)
	,m_goldNumber(0)
	,m_sellRatio(0.f)
{
}

DynamicData::~DynamicData()
{
}

bool DynamicData::initializeSaveData(int idx)
{
	auto fileUtil = FileUtils::getInstance();
	//获取存档路径
	string path = fileUtil->getWritablePath();
	//对应的存档完整路径
	string filepath = m_fullpath = StringUtils::format("%ssave%d.plist", path.c_str(), idx);
	//不存在对应存档,则使用默认存档
	if ( !fileUtil->isFileExist(m_fullpath))
	{
		filepath = "data/default_data.plist";
		m_bFirstGame = true;
	}
	else
		m_bFirstGame = false;
	//获得对应存档的键值对
	m_valueMap = fileUtil->getValueMapFromFile(filepath);
	//获取有用的数据
	m_goldNumber = m_valueMap.at("gold").asInt();
	m_sellRatio = m_valueMap.at("sell_ratio").asFloat();
	//TODO:解析背包中的物品并生成Good

	return true;
}

ValueMap& DynamicData::getValueMapOfCharacter()
{
	return m_valueMap.at("character").asValueMap();
}

string DynamicData::getMapFilePath() const
{
	return m_valueMap.at("map").asString();
}

Point DynamicData::getTileCoordinateOfPlayer() const
{
	auto strPoint = m_valueMap.at("tile_position").asString();
	auto position = PointFromString(strPoint);

	return position;
}

DynamicData类主要功能就是负责动态数据的提供和更新,比如主角的金钱数目,主角所处的地图名和位置,背包物品以及主角的属性,穿戴的装备,技能(以后添加)等等。目前的DynamicData类主要就是负责读取存档。如果是第一次进行游戏的话,则使用默认存档,并解析出有用的属性,此为initializeSaveData函数所做的功能,且为了可扩展性,添加了一个idx形参,允许RPG游戏有不同的存档。

StaticData和DynamicData都为单例类,且为了保证其功能的单一性,这两个类都是只与数据相关,都不包含与渲染相关的功能。

本节运行界面如下:

rpg 游戏架构 rpg构造_#include_03

本节代码:链接:https://pan.baidu.com/s/1MiB-b5AZUlDUlpHypxklxg 密码:z529