cocos2dx学习之路----第十一篇(对声音的操作处理)

这一篇就来谈谈关于声音的加载和播放等问题吧。

在cocos里面对声音的加载有两种声音引擎选择供我们使用。一种是比较简单的声音引擎,而另外一种会比较复杂一点。

一般来说,如果我们只是要简单的播放音乐音效,就可以选择用第一种,而如果要对音乐音效的流程进行控制,比如播放完成之后要进行怎样的处理,它将有一个完成音乐或音效播放后的回调方法。

我们先来看下第一种:

这个类命名为SimpleAudioEngine。顾名思义,简单的声音播放引擎。

对于这个类的接口我就不一一列出来了,不过至少需要知道以下的这些接口:

virtual unsigned int playEffect(const char* filePath, bool loop = false,float pitch = 1.0f, float pan = 0.0f, float gain = 1.0f);//播放音效
virtual void pauseEffect(unsigned int soundId);//暂停音效播放
virtual void resumeEffect(unsigned int soundId);//恢复音效播放
virtual void stopEffect(unsigned int soundId);//停止音效播放
virtual void preloadEffect(const char* filePath);//预加载一个压缩的声音文件
virtual void unloadEffect(const char* filePath);//卸载内部加载的音效缓存

这些接口是音效处理的接口。对于音乐文件的处理,这一种中提供了两种音乐类型分别做处理,即背景音乐和音效。一般来说,背景音乐播放的时间会比音效长。而这里的背景音乐与音效的区别就是背景音乐只能同时播放一个,而音效可以多个。并且在win32中,cocos内部目前并没有对背景音乐的预加载进行处理。所以,个人觉得可以把背景音乐也归为音效一类。

在测试中,我们也是如此,对于背景音乐的加载也是通过预加载音效文件。

好,我们来看看具体的操作吧,这次先看看我所做测试的源码:

SimpleAudioEngineTest.h:

#ifndef __SIMPLE_AUDIO_ENGINE_TEST_H__
#define __SIMPLE_AUDIO_ENGINE_TEST_H__

#include"cocos2d.h"
#include"SimpleAudioEngine.h"<span style="white-space:pre">	</span>//包含声音引擎头文件

USING_NS_CC;
using namespace CocosDenshion;<span style="white-space:pre">	</span>//使用该声音引擎的命名空间

class SimpleAudioEngineTest :public Layer{
public:
	SimpleAudioEngineTest();

	static Scene *createScene();
	virtual bool init();
	CREATE_FUNC(SimpleAudioEngineTest);

	virtual void onEnter();<span style="white-space:pre">	</span>//重写onEnter方法
	virtual void onExit();<span style="white-space:pre">	</span>//重写onExit方法

private:
	SimpleAudioEngine *_engine;<span style="white-space:pre">	</span>//声音引擎单例指针
	unsigned int _audioID;<span style="white-space:pre">		</span>//声音文件ID
	bool _loop;<span style="white-space:pre">			</span>//是否循环播放
};

#endif

SimpleAudioEngineTest.cpp:

#include"SimpleAudioEngineTest.h"
#include"SoundPlayerTest.h"<span style="white-space:pre">	</span>//包含第二个测试的头文件,以便跳转至该场景

SimpleAudioEngineTest::SimpleAudioEngineTest(){
	//变量初始化
	_engine = nullptr;
	_audioID = 0;
	_loop = false;

}
Scene *SimpleAudioEngineTest::createScene(){
	auto scene = Scene::create();
	auto layer = SimpleAudioEngineTest::create();
	scene->addChild(layer);
	return scene;
}
bool SimpleAudioEngineTest::init(){
	if (!Layer::init()){
		return false;
	}
	auto visibleSize = Director::getInstance()->getVisibleSize();

	//当前测试标签描述
	auto test_label = Label::createWithSystemFont("SimpleAudioEngine Test", "", 30);
	test_label->setPosition(Vec2(visibleSize.width / 2, visibleSize.height - test_label->getContentSize().height));
	this->addChild(test_label);

	//Play
	auto Play_Item = MenuItemFont::create("Play", [&](Ref *sender){
		_audioID = _engine->playEffect("music/background.mp3", _loop);//条目被点击时,调用播放音效的方法,返回一个文件ID
	});
	Play_Item->setPosition(Vec2(-visibleSize.width / 4, visibleSize.height / 4));

	//Stop
	auto Stop_Item = MenuItemFont::create("Stop", [&](Ref *sender){
		if (_audioID != 0){
			_engine->stopEffect(_audioID);//条目被点击时,调用停止播放音效方法,参数为音乐文件ID
			_audioID = 0;
		}

	});
	Stop_Item->setPosition(Vec2(visibleSize.width / 4, visibleSize.height / 4));

	//Pause
	auto Pause_Item = MenuItemFont::create("Pause", [&](Ref *sender){
		if (_audioID != 0){
			_engine->pauseEffect(_audioID);<span style="font-family: SimSun;">//条目被点击时,调用暂停播放音效方法,参数为音乐文件ID</span>

		}

	});
	Pause_Item->setPosition(Vec2(-visibleSize.width / 4, 0));

	//Resume
	auto Resume_Item = MenuItemFont::create("Resume", [&](Ref *sender){
		if (_audioID != 0){
			_engine->resumeEffect(_audioID);<span style="font-family: SimSun;">//条目被点击时,调用恢复播放音效方法,参数为音乐文件ID</span>

		}

	});
	Resume_Item->setPosition(Vec2(visibleSize.width / 4, 0));

	//第二个测试跳转标签
	auto EnterNextTest = MenuItemLabel::create(Label::createWithSystemFont("Click Here Enter Next Test", "", 35),
		[](Ref *sender){
		Director::getInstance()->replaceScene(SoundPlayerTest::createScene());
	});
	EnterNextTest->setPosition(Vec2(0, -visibleSize.height/4));

	auto menu = Menu::create(Play_Item, Stop_Item, Pause_Item, Resume_Item, EnterNextTest, NULL);
	addChild(menu, 10);

	return true;
}

void SimpleAudioEngineTest::onEnter(){
	Layer::onEnter();

	//声音引擎初始化
	_engine = SimpleAudioEngine::getInstance();

	//预加载声音文件
	_engine->preloadEffect("music/background.mp3");

	CCLOG("OnEnter....");
}
void SimpleAudioEngineTest::onExit(){
	if (_engine){
		_engine->unloadEffect("music/background.mp3");//清除内部声音文件缓存
	}
	Layer::onExit();
}

好,我们再来看看测试二。

测试二将会用到另一种声音引擎,这个类叫AudioEngine。它是在3.x之后开出的另外一个声音播放引擎,功能比第一个声音引擎强大的一点就是可以控制声音的播放进度,并且也实现了在win32平台没有实现的音量的控制。下面我们来看看那需要了解的接口有哪些吧:

static int play2d(const std::string& filePath, bool loop = false, float volume = 1.0f, const AudioProfile *profile = nullptr);//播放音乐文件,返回音乐文件ID
static void setVolume(int audioID, float volume);//设置音量大小
static void pause(int audioID);//通过指定音乐文件ID暂停播放音乐
static void resume(int audioID); //通过指定音乐文件ID恢复播放音乐
static void stop(int audioID); //通过指定音乐文件ID停止播放音乐
static void setFinishCallback(int audioID, const std::function<void(int,const std::string&)>& callback);//音乐文件播放完成后调用指定函数
static void preload(const std::string& filePath);//预加载音乐文件至内部缓存
static void uncache(const std::string& filePath);//通过指定文件清除其在内部中的缓存

以上就是测试二所需要了解的接口。对于声音音量的控制,我们还需要添加一个滑动条的UI来控制。

好,现在先看看我实现的参考代码:

SoundPlayerTest.h:

#ifndef __SOUND_PLAYER_TEST_H__
#define __SOUND_PLAYER_TEST_H__

#include"cocos2d.h"
#include"AudioEngine.h" //声音引擎头文件的引入
#include"ui\CocosGUI.h"<span style="white-space:pre">	</span>//UI控件头文件的引入

USING_NS_CC;
using namespace experimental;//使用声音引擎命名空间
using namespace ui;<span style="white-space:pre">	</span>     //使用UI命名空间

class SoundPlayerTest :public Layer{
public:
	SoundPlayerTest();

	static Scene *createScene();
	virtual bool init();
	CREATE_FUNC(SoundPlayerTest);

	virtual void onEnter();
	virtual void onExit();

private:
	int _audioID;
	bool _loop;
};

#endif

SoundPlayerTest.cpp:

#include"SoundPlayerTest.h"

SoundPlayerTest::SoundPlayerTest(){
	//变量初始化
	_audioID = AudioEngine::INVALID_AUDIO_ID;//这个值为AudioEngine中的一个初始值,为-1
	_loop = false;

}
Scene *SoundPlayerTest::createScene(){
	auto scene = Scene::create();
	auto layer = SoundPlayerTest::create();
	scene->addChild(layer);
	return scene;
}
bool SoundPlayerTest::init(){
	if (!Layer::init()){
		return false;
	}
	auto visibleSize = Director::getInstance()->getVisibleSize();

	//当前测试标签描述
	auto test_label = Label::createWithSystemFont("Sound Player Test", "", 30);
	test_label->setPosition(Vec2(visibleSize.width / 2, visibleSize.height - test_label->getContentSize().height));
	this->addChild(test_label);

	//Play
	auto Play_Item = MenuItemFont::create("Play", [&](Ref *sender){
		if (_audioID == AudioEngine::INVALID_AUDIO_ID){
			_audioID = AudioEngine::play2d("music/background.mp3", _loop);//播放音乐文件
		}
		if (_audioID != AudioEngine::INVALID_AUDIO_ID){
			AudioEngine::setFinishCallback(_audioID, [&](int id, const std::string &filePath){
				_audioID = AudioEngine::INVALID_AUDIO_ID; //播放完成时,再次初始化声音文件ID
			});
		}

	});
	Play_Item->setPosition(Vec2(-visibleSize.width / 4, visibleSize.height / 4));

	//Stop
	auto Stop_Item = MenuItemFont::create("Stop", [&](Ref *sender){
		if (_audioID != AudioEngine::INVALID_AUDIO_ID){
			AudioEngine::stop(_audioID);<span style="white-space:pre">	</span>//停止播放音乐文件
			_audioID = AudioEngine::INVALID_AUDIO_ID;
		}

	});
	Stop_Item->setPosition(Vec2(visibleSize.width / 4, visibleSize.height / 4));

	//Pause
	auto Pause_Item = MenuItemFont::create("Pause", [&](Ref *sender){
		if (_audioID != AudioEngine::INVALID_AUDIO_ID){
			AudioEngine::pause(_audioID);//暂停播放音乐文件
		}

	});
	Pause_Item->setPosition(Vec2(-visibleSize.width / 4, 0));

	//Resume
	auto Resume_Item = MenuItemFont::create("Resume", [&](Ref *sender){
		if (_audioID != AudioEngine::INVALID_AUDIO_ID){
			AudioEngine::resume(_audioID);//恢复播放音乐文件
		}

	});
	Resume_Item->setPosition(Vec2(visibleSize.width / 4, 0));

	auto menu = Menu::create(Play_Item, Stop_Item, Pause_Item, Resume_Item, NULL);
	addChild(menu,10);

	//初始化控制音量的滑动条UI
	Slider *slider = Slider::create();
	slider->loadBarTexture("cocosui/sliderTrack.png");//滑动条背景纹理
	slider->loadSlidBallTextures("cocosui/sliderThumb.png", "cocosui/sliderThumb.png");//滑动按钮纹理,第一个为正常,第二个为被点击时
	slider->loadProgressBarTexture("cocosui/sliderProgress.png");//进度纹理
	slider->setPosition(Vec2(visibleSize.width / 2, 100));
	slider->setScale(1.5f);
	slider->setPercent(100);
<span style="white-space:pre">	</span>
	slider->addEventListener([&](Ref *sender,Slider::EventType type)
	{
		auto s = dynamic_cast<Slider*>(sender);
		auto volum = 1.0f * s->getPercent() / s->getMaxPercent();//计算音量的值,音量的值应为:0~1之间
		if (_audioID != AudioEngine::INVALID_AUDIO_ID){
			AudioEngine::setVolume(_audioID, volum);
		}
	});
	addChild(slider);

	return true;
}

void SoundPlayerTest::onEnter(){
	Layer::onEnter();

	//声音引擎初始化
	AudioEngine::lazyInit();

	//加载声音文件
	AudioEngine::preload("music/background.mp3");

	CCLOG("OnEnter....");
}
void SoundPlayerTest::onExit(){
	if (_audioID != AudioEngine::INVALID_AUDIO_ID){
		AudioEngine::uncache("music/background.mp3");//清除音乐文件的缓存
	}
	Layer::onExit();
}

好,以上就是测试的内容。需要注意的是我这里的资源文件,都需要放在Resource文件夹里,因为资源的默认工作目录就在这。我在Resource里创建了music文件夹,里面就是放置所有声音资源文件,而创建一个cocosui文件夹则是放置所有UI相关控件的纹理文件,如下图所示:

这些资源都可以在源代码包中的testcpp找到。

好了,最后我们再来看运行的结果吧~(虽然声音听不到,但是可以感受一下哈~)

好了,如果想了解关于声音的内容,可以去看看官方提供的例子或是它们里面的提供的接口哈~对于声音的内容就谈到这里吧~下一篇就来谈谈关于Node节点中的自我更新接口。在游戏开发中可是占具重要位置的!

时间: 2024-08-05 03:01:44

cocos2dx学习之路----第十一篇(对声音的操作处理)的相关文章

学习之路四十一丶简论重构

四月份的最后一天,写点心得,记录一下. 这个月一直忙着开发一个基于Win32 API的程序,大量运用了句柄等很多API的知识. 尤其随着代码量越来越大,逻辑越来越复杂,代码的清晰,健壮,扩展性成了一个需要重视的问题,也就是要适时的重构了. 一丶重构的时机 上个星期在修改一块重大逻辑的时候,需要修改很多代码,当时我犯了一个错误,一开始想了一个思路,但一上来没写多少就开始想着重构代码,目的是使其代码清晰以及可扩展. 可是随着时间的流失,不仅没有重构好,而且该改的逻辑也没有改好,我很郁闷,为什么会这样

学习java随笔第十一篇:java窗体程序

要开java的窗体程序,就要下载开发窗体的工具. 这里我用的是的myeclipse,可以直接在网上下载安装即可. 我用的是10.0版本的,如果需要汉化的话,可以看一下这篇文章:myeclipse.10安装和汉化方法 java窗体 package java20130629; import javax.swing.*; public class Form extends JFrame { public Form() { this.setLayout(null); JLabel jl=new JLab

Python之路【第九篇】:Python操作 RabbitMQ、Redis、Memcache、SQLAlchemy

Python之路[第九篇]:Python操作 RabbitMQ.Redis.Memcache.SQLAlchemy Memcached Memcached 是一个高性能的分布式内存对象缓存系统,用于动态Web应用以减轻数据库负载.它通过在内存中缓存数据和对象来减少读取数据库的次数,从而提高动态.数据库驱动网站的速度.Memcached基于一个存储键/值对的hashmap.其守护进程(daemon )是用C写的,但是客户端可以用任何语言来编写,并通过memcached协议与守护进程通信. Memc

linux学习之路二 ------登陆篇

上一篇中讲了如何搭建虚拟机学习平台,在这篇中将介绍如何登陆Linux系统和修改超级用户密码 1.打开虚拟机后,进入界面,需要让我们输入账号密码,在前面安装的时候我们设置过.如图 2.输入账号密码,输入密码的时候是不显示的,所以不要认为没有输入.如图 3.验证成功之后,出现[[email protected] ~]$ ,则说明登陆成功,Linux系统有超级用户和普通用户之分,超级用户也就是root用户了,如我们的就是普通用户, 普通用户[[email protected] ~]$中$就是代表了普通

Python学习之路【第一篇】-Python简介和基础入门

1.Python简介 1.1 Python是什么 相信混迹IT界的很多朋友都知道,Python是近年来最火的一个热点,没有之一.从性质上来讲它和我们熟知的C.java.php等没有什么本质的区别,也是一种开发语言,而且已经进阶到主流的二十多种开发语言的top 5(数据源自最新的TIOBE排行榜). 来头不小啊!二十多种主流的开发语言,我该从哪一个开始呢?人生苦短,let‘s python! 1.2 Python的由来和发展趋势 Python的前世源自鼻祖“龟叔”.1989年,吉多·范罗苏姆(Gu

Extjs5.0 学习之路【资源篇】

磨刀不误砍柴工. 先收集资源,然后再开始学习之路. Extjs5.0 文件下载 关于extjs5.0优秀博文: http://blog.csdn.net/sushengmiyan/article/details/38331347 专栏 http://blog.csdn.net/column/details/sushengextjs5.html?&page=2

cocos2dx学习之路----第十二篇(关于Node节点的自我更新函数Update探讨及定时器的使用)

这一篇来谈谈关于Node节点比较重要的函数接口,自我更新接口(update)以及定时器. 好,下面先来谈谈节点的自我更新. 每一个Node节点都有自带一个update函数.而当调用ScheduleUpdate()这一方法便会被开启,在每一帧被调用. 我们可能注意到,在AppDelegate中有个setAnimationInterval()的方法,有导演类调用.而这个方法就是设置渲染的帧率的.默认是以60帧每秒进行渲染.当然,并不是真正的每秒就渲染60次,这个只能是差不多达到这个速率,因为程序在运

Python学习之路——基础10篇

在这一篇中我们主要学习python中求字符串的长度的方法--__len__() 无论是二维的还是一维的都是它,及其简便 根据题目学python: 给定由大写,小写字母和空格组成的字符串,返回 最后 一个单词的长度. 如果输入中不存在单词,返回 00. 注意: "单词"是指不包含空格符号的字符串 例如: 对于字符串"hello World"(不带引号), 那么返回的结果是 55: 对于字符串"abc abc "(不带引号),那么返回的结果就是 33

vue学习指南:第十一篇(详细) - Vue的 路由 第一篇 ( router )

一.路由的配置 路由  vue-router 1. 什么是路由? 路由相当于一个配置对象 路由:就是我们通过不同的url访问不同的内容,通过angular.js 可以实现多视图的单页,现在流行的单页面开发模式. 2. 路由在web 有两种: 第一种 hash hash值前面带 # "  https:   //    user   :   pass   @ sub.example.com : 8080   /p/a/t/h  ?  query=string  #hash " 第二种是 h