创建完cocos2d-x的lua项目后,打开项目的Resources文件夹,找到hello.lua,cocos2d-x的lua项目的测试例子主要就是由这个脚本文件执行的。

require "AudioEngine" --引入声音引擎文件

-- for CCLuaEngine traceback
function __G__TRACKBACK__(msg)
    print("----------------------------------------")
    print("LUA ERROR: " .. tostring(msg) .. "\n")--打印错误信息
    print(debug.traceback())--输出当前运行的traceback信息
    print("----------------------------------------")
end
--主函数
local function main()
    -- avoid memory leak
    -- 避免内存泄露
    --操作垃圾收集器
    collectgarbage("setpause", 100)
    collectgarbage("setstepmul", 5000)
    --定义日志输出函数
    local cclog = function(...)
        print(string.format(...))
    end
    --引入hello2脚本
    require "hello2"
    --调用hello2脚本中的myadd函数
    cclog("result is " .. myadd(3, 5))

    ---------------
    --获取屏幕的可视化大小
    local visibleSize = CCDirector:sharedDirector():getVisibleSize()
    --获取可视区域的起点坐标
    local origin = CCDirector:sharedDirector():getVisibleOrigin()

    -- add the moving dog
    --创建一个可移动的小狗
    local function creatDog()
        local frameWidth = 105
        local frameHeight = 95

        -- create dog animate
        --添加小狗的贴图到内存中
        local textureDog = CCTextureCache:sharedTextureCache():addImage("dog.png")
        --创建截取第一个小狗的精灵帧的矩形
        local rect = CCRectMake(0, 0, frameWidth, frameHeight)
        --创建一个小狗的精灵帧
        local frame0 = CCSpriteFrame:createWithTexture(textureDog, rect)
        --创建截取第二个小狗的精灵帧的矩形
        rect = CCRectMake(frameWidth, 0, frameWidth, frameHeight)
        --创建一个第二个小狗的精灵帧
        local frame1 = CCSpriteFrame:createWithTexture(textureDog, rect)

        --以小狗的第一个精灵帧创建一个小狗精灵
        local spriteDog = CCSprite:createWithSpriteFrame(frame0)
        --自定义一个bool值的属性,标记为false
        spriteDog.isPaused = false
        --设置小狗的位置在 x:0, y:屏幕高度*0.75 处
        spriteDog:setPosition(origin.x, origin.y + visibleSize.height / 4 * 3)
        --创建一个用于存储精灵帧的数组
        local animFrames = CCArray:create()
        --把上面的两个精灵帧加入到数组中
        animFrames:addObject(frame0)
        animFrames:addObject(frame1)
        --以0.5秒的切换速度方式创建一个动画描述
        local animation = CCAnimation:createWithSpriteFrames(animFrames, 0.5)
        --根据上面的动画描述创建一个动画
        local animate = CCAnimate:create(animation);
        --运行该动画
        spriteDog:runAction(CCRepeatForever:create(animate))

        -- moving dog at every frame
        --小狗每帧都在移动
        local function tick()
            if spriteDog.isPaused then return end--如果暂停,则返回,不移动
            local x, y = spriteDog:getPosition()--获取小狗的当前位置
            if x > origin.x + visibleSize.width then--如果小狗的移到屏幕外面
                x = origin.x--重置小狗的位置
            else
                x = x + 1--每帧往右移动一个像素
            end
            --刷新小狗的最新位置
            spriteDog:setPositionX(x)
        end
        --执行定时器
        CCDirector:sharedDirector():getScheduler():scheduleScriptFunc(tick, 0, false)
        --返回创建的小狗
        return spriteDog
    end

    -- create farm
    --创建农田
    local function createLayerFarm()
        --创建一个农场层
        local layerFarm = CCLayer:create()

        -- add in farm background
        --创建一个背景精灵
        local bg = CCSprite:create("farm.jpg")
        --设置背景精灵的位置
        bg:setPosition(origin.x + visibleSize.width / 2 + 80, origin.y + visibleSize.height / 2)
        --把背景精灵添加到农场层中
        layerFarm:addChild(bg)

        -- add land sprite
        --创建农场精灵
        for i = 0, 3 do
            for j = 0, 1 do
                local spriteLand = CCSprite:create("land.png")
                spriteLand:setPosition(200 + j * 180 - i % 2 * 90, 10 + i * 95 / 2)
                layerFarm:addChild(spriteLand)
            end
        end

        -- add crop
        --创建农作物
        local frameCrop = CCSpriteFrame:create("crop.png", CCRectMake(0, 0, 105, 95))
        for i = 0, 3 do
            for j = 0, 1 do
                local spriteCrop = CCSprite:createWithSpriteFrame(frameCrop);
                spriteCrop:setPosition(10 + 200 + j * 180 - i % 2 * 90, 30 + 10 + i * 95 / 2)
                layerFarm:addChild(spriteCrop)
            end
        end

        -- add moving dog
        --创建一个会移动的小狗
        local spriteDog = creatDog()
        --将小狗添加到农场层中
        layerFarm:addChild(spriteDog)

        -- handing touch events
        --初始化触摸点
        local touchBeginPoint = nil
        --开始点击触摸屏
        local function onTouchBegan(x, y)
            --输出点击位置
            cclog("onTouchBegan: %0.2f, %0.2f", x, y)
            --将点击位置赋给触摸点
            touchBeginPoint = {x = x, y = y}
            --点击时小狗暂停移动
            spriteDog.isPaused = true
            -- CCTOUCHBEGAN event must return true
            return true
        end
        --滑动触摸屏
        local function onTouchMoved(x, y)
            --输出滑动位置
            cclog("onTouchMoved: %0.2f, %0.2f", x, y)
            --如果触摸点不为空
            if touchBeginPoint then
                --获取农场层的当前位置
                local cx, cy = layerFarm:getPosition()
                --移动农场层
                layerFarm:setPosition(cx + x - touchBeginPoint.x,
                                      cy + y - touchBeginPoint.y)
                --将滑动的位置赋给触摸点
                touchBeginPoint = {x = x, y = y}
            end
        end
        --离开触摸屏
        local function onTouchEnded(x, y)
            --输出离开屏幕时的位置
            cclog("onTouchEnded: %0.2f, %0.2f", x, y)
            --删除触摸点
            touchBeginPoint = nil
            --小狗重新移动
            spriteDog.isPaused = false
        end
        --处理触摸状态
        local function onTouch(eventType, x, y)
            if eventType == "began" then   --点击屏幕状态
                return onTouchBegan(x, y)
            elseif eventType == "moved" then --滑动屏幕状态
                return onTouchMoved(x, y)
            else  --离开屏幕状态
                return onTouchEnded(x, y) 
            end
        end
        --注册触摸事件
        layerFarm:registerScriptTouchHandler(onTouch)
        --开启触摸
        layerFarm:setTouchEnabled(true)
        --返回农场层
        return layerFarm
    end


    -- create menu
    --创建菜单
    local function createLayerMenu()
        --创建一个菜单层
        local layerMenu = CCLayer:create()
        --定义三个变量
        local menuPopup, menuTools, effectID
        --执行菜单项的回调函数
        local function menuCallbackClosePopup()
            -- stop test sound effect
            --停止音效的播放
            AudioEngine.stopEffect(effectID)
            --隐藏菜单
            menuPopup:setVisible(false)
        end
        --执行菜单项的回调函数
        local function menuCallbackOpenPopup()
            -- loop test sound effect
            --获取播放音效的路径
            local effectPath = CCFileUtils:sharedFileUtils():fullPathForFilename("effect1.wav")
            --播放音效
            effectID = AudioEngine.playEffect(effectPath)
            --显示菜单
            menuPopup:setVisible(true)
        end

        -- add a popup menu
        --创建一个菜单项
        local menuPopupItem = CCMenuItemImage:create("menu2.png", "menu2.png")
        --设置菜单项位置
        menuPopupItem:setPosition(0, 0)
        --设置执行该菜单项所对应的回调函数
        menuPopupItem:registerScriptTapHandler(menuCallbackClosePopup)
        --创建一个菜单
        menuPopup = CCMenu:createWithItem(menuPopupItem)
        --设置菜单位置
        menuPopup:setPosition(origin.x + visibleSize.width / 2, origin.y + visibleSize.height / 2)
        --隐藏菜单
        menuPopup:setVisible(false)
        --将该菜单添加到菜单层
        layerMenu:addChild(menuPopup)

        -- add the left-bottom "tools" menu to invoke menuPopup
        --创建一个菜单项
        local menuToolsItem = CCMenuItemImage:create("menu1.png", "menu1.png")
        --设置菜单项位置
        menuToolsItem:setPosition(0, 0)
        --设置执行该菜单项所对应的回调函数
        menuToolsItem:registerScriptTapHandler(menuCallbackOpenPopup)
        --创建一个菜单
        menuTools = CCMenu:createWithItem(menuToolsItem)
        --设置菜单位置
        local itemWidth = menuToolsItem:getContentSize().width
        local itemHeight = menuToolsItem:getContentSize().height
        menuTools:setPosition(origin.x + itemWidth/2, origin.y + itemHeight/2)
        --将该菜单添加到菜单层
        layerMenu:addChild(menuTools)
        --返回菜单层
        return layerMenu
    end

    -- play background music, preload effect

    -- uncomment below for the BlackBerry version
    -- local bgMusicPath = CCFileUtils:sharedFileUtils():fullPathForFilename("background.ogg")
    --获取背景音乐的文件路径
    local bgMusicPath = CCFileUtils:sharedFileUtils():fullPathForFilename("background.mp3")
    --播放背景音乐
    AudioEngine.playMusic(bgMusicPath, true)
    --获取音效的文件路径
    local effectPath = CCFileUtils:sharedFileUtils():fullPathForFilename("effect1.wav")
    --将音效缓存到内存中
    AudioEngine.preloadEffect(effectPath)

    -- run
    --创建一个场景
    local sceneGame = CCScene:create()
    --将农村层添加到该场景中
    sceneGame:addChild(createLayerFarm())
    --将菜单层添加到该场景中
    sceneGame:addChild(createLayerMenu())
    --运行该场景
    CCDirector:sharedDirector():runWithScene(sceneGame)
end
--保护模式下调用主函数,并制定错误处理函数句柄__G__TRACKBACK__
xpcall(main, __G__TRACKBACK__)



而这个脚本是在AppDelegate.cpp中被执行的,里面初始化了cocos2d-x解析lua的引擎。

bool AppDelegate::applicationDidFinishLaunching()
{
    // initialize director
    CCDirector *pDirector = CCDirector::sharedDirector();
    pDirector->setOpenGLView(CCEGLView::sharedOpenGLView());

    // turn on display FPS
    pDirector->setDisplayStats(true);

    // set FPS. the default value is 1.0/60 if you don't call this
    pDirector->setAnimationInterval(1.0 / 60);

    // register lua engine
	//初始化lua脚本引擎
    CCLuaEngine* pEngine = CCLuaEngine::defaultEngine();
	//将lua脚本引擎添加到脚本引擎管理器中
    CCScriptEngineManager::sharedManager()->setScriptEngine(pEngine);
	//取得lua的全局指针,所有的lua函数都需要使用这个指针来做为参数进行调用。  
    CCLuaStack *pStack = pEngine->getLuaStack();
	//获取lua_State
    lua_State *tolua_s = pStack->getLuaState();
	//开放扩展函数给lua调用
    tolua_extensions_ccb_open(tolua_s);
	//如果是ios、android、win32平台则开放WebSocket类给lua调用
#if (CC_TARGET_PLATFORM == CC_PLATFORM_IOS || CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID || CC_TARGET_PLATFORM == CC_PLATFORM_WIN32)
    pStack = pEngine->getLuaStack();
    tolua_s = pStack->getLuaState();
    tolua_web_socket_open(tolua_s);
#endif
    
#if (CC_TARGET_PLATFORM == CC_PLATFORM_BLACKBERRY)//黑莓平台,不清楚
    CCFileUtils::sharedFileUtils()->addSearchPath("script");
#endif
	//获取待执行的脚本路径
    std::string path = CCFileUtils::sharedFileUtils()->fullPathForFilename("hello.lua");
	//执行脚本
    pEngine->executeScriptFile(path.c_str());

    return true;
}



其中CCLuaEngine是继承自CCScriptEngineProtocl,这个协议定义了lua引擎的功能,其中CCScriptEngineManager用来管理当期的脚本文件(目前没发现这个管理类有什么用)。

void CCScriptEngineManager::setScriptEngine(CCScriptEngineProtocol *pScriptEngine)
{
    removeScriptEngine();//移除上一个的脚本引擎
    m_pScriptEngine = pScriptEngine;//将当前的脚本引擎赋给m_pScriptEngine
}

void CCScriptEngineManager::removeScriptEngine(void)
{
    if (m_pScriptEngine)//如果脚本引擎的实例存在,则delete掉
    {
        delete m_pScriptEngine;
        m_pScriptEngine = NULL;
    }
}