JNI筆記

  因爲要作一個可以加紅字體的dialog,而cocos2d中的CCMessageBox是系統內帶的,我沒法修改其字體顏色。事實上是能夠修改的,經過觀察發現CCMessageBox被調用後,在安卓平臺中會調用org.cocos2dx.lib.Cocos2dxHandler類中的showDialog方法,結果發現Cocos2dx使用AlertDialog來實現的,貼上代碼:html

	private void showDialog(Message msg) {
		Cocos2dxActivity theActivity = this.mActivity.get();
		DialogMessage dialogMessage = (DialogMessage)msg.obj;
		
		new AlertDialog.Builder(theActivity)
		.setTitle(dialogMessage.titile)
		.setMessage(dialogMessage.message)
		//.setView(view) 有一個setView成員方法容許咱們定義本身的View
		.setPositiveButton("ok", 
				new DialogInterface.OnClickListener() {
					
					@Override
					public void onClick(DialogInterface dialog, int which) {
						// TODO Auto-generated method stub
						
					}
				}).create().show();
	}

 而後裏面能夠經過setView來實現字體加紅,可是這個org.cocos2dx.lib是公共庫,每一個遊戲都須要依賴它,修改它可能會帶來不知道的隱患,所以我以爲提供一個方法給遊戲調用,能下降耦合。這樣就須要在Cocos2dx中調用java的AlertDialog,因此便引出了想記錄的東西JNI。java


 利用create_project命令建立cocos2dx遊戲工程,在eclipse中import進來,以後在src文件夾中建立一個.java文件,記錄代碼以下:android

package com.My.Dialog;

import android.app.Activity;
import android.app.AlertDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.res.Resources;
import android.os.Handler;
import android.os.Message;
import android.text.Html;
import android.text.Spanned;
import android.text.method.ScrollingMovementMethod;
import android.util.Log;
import android.view.LayoutInflater;
import android.widget.LinearLayout;
import android.widget.TextView;

public class ShowRedDialog {
	private static Handler mHandler;
	private Activity activity;
	
	public void init(Activity act){
		Log.i("%s", "show dialog init");
		activity = act;
		mHandler = new Handler()
		    {
				@Override
				public void handleMessage(Message msg) {
					Log.i("%s", "we are in handler");
					Resources res = activity.getResources();
					//經過在strings.xml中配置layout的位置的方法來添加View
					String layoutResStr = res.getString(com.My.Dialog.R.string.view_layout);
					if(layoutResStr == null){
						Log.e("%s", "layoutResStr can not find !!!");
						new AlertDialog.Builder(activity).setTitle("Error").setMessage("layoutResStr can not find !!!")
															  .setPositiveButton("ok", new DialogInterface.OnClickListener() {
																
																@Override
																public void onClick(DialogInterface dialog, int which) {
																	// TODO Auto-generated method stub
																	
																}
															}).create().show();
						
					}
					int layoutId = res.getIdentifier(layoutResStr, "", "");
					Log.i("%s", "the resource is found!!");
					//反射出layout中的textView
					LayoutInflater inflater = (LayoutInflater)activity.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
					LinearLayout view = (LinearLayout)inflater.inflate(layoutId, null);
					TextView textview = (TextView) view.getChildAt(0);
					//Spanned prasedText = Html.fromHtml("<font color=\"FF0000\"> MESSAGE </font> " + Integer.toString(layoutId));
					//設置textView的文本顯示,利用html來修改文本樣式
					Spanned prasedText = Html.fromHtml("<font color=\"FF0000\"> MESSAGE </font>");
					textview.setText(prasedText);
					textview.setMovementMethod(ScrollingMovementMethod.getInstance());
					Log.i("%s", "ready to show our messagebox");
					//messageBox彈出
					new AlertDialog.Builder(activity)
					.setTitle("Warning")
					.setView(view)
					.setPositiveButton("ok", 
							new DialogInterface.OnClickListener() {
								
								@Override
								public void onClick(DialogInterface dialog, int which) {
									// TODO Auto-generated method stub
									
								}
							}).create().show();
				}
		    };
	}
	
    public static void showMyDialog(String data)
    {
    	Log.i("in java show my dialog %s", data);
    	Message msg = mHandler.obtainMessage();
    	msg.sendToTarget();
    }

}

 這裏還有不少要改進的地方,但是不是如今的重點,例如我想把View分離出來,可讓別人去實現;還有方法到底該怎麼設計更好,變量是靜態仍是非靜態等等。在這裏經過Handler實現C++調用靜態函數,而後靜態函數調用非靜態函數,這只是其中一個實現C++調用非靜態函數的辦法,網上還有好多。我在strings.xml中加入了以下配置:app

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <string name="app_name">JniDialog</string>
    <string name="hello_world">Hello</string>
    <string name="view_layout">com.My.Dialog:layout/godmsgdialog</string>
</resources>

在jni/hellocpp下新建test.h和test.cpp,代碼:eclipse

#ifndef TEST_H
#define TEST_H

void showMyDialog(const char *tmp);

#endif

test.cpp中的代碼:ide

#include <jni.h>
#include "test.h"
#include "platform/android/jni/JniHelper.h"
#include "cocos2d.h"
#include <android/log.h>


#define TAG "myDemo-jni" // 這個是自定義的LOG的標識
#define LOGD(...) __android_log_print(ANDROID_LOG_DEBUG,TAG ,__VA_ARGS__) // 定義LOGD類型
#define LOGI(...) __android_log_print(ANDROID_LOG_INFO,TAG ,__VA_ARGS__) // 定義LOGI類型
#define LOGW(...) __android_log_print(ANDROID_LOG_WARN,TAG ,__VA_ARGS__) // 定義LOGW類型
#define LOGE(...) __android_log_print(ANDROID_LOG_ERROR,TAG ,__VA_ARGS__) // 定義LOGE類型
#define LOGF(...) __android_log_print(ANDROID_LOG_FATAL,TAG ,__VA_ARGS__) // 定義LOGF類型

using namespace cocos2d;

void showMyDialog(const char *tmp){
	LOGI("########## i = %d", "call in JNI");
#if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
	JniMethodInfo t;
	if(JniHelper::getStaticMethodInfo(t, "com/My/Dialog/ShowRedDialog", "showMyDialog", "(Ljava/lang/String;)V")){
		jstring jMsg = t.env->NewStringUTF("do not touch me !!");
		t.env->CallStaticVoidMethod(t.classID, t.methodID, jMsg);
		t.env->DeleteLocalRef(jMsg);
	}
#endif
}

 在這裏遇到了各類問題:函數

一、第一個是JniHelper::getStaticMethodInfo靜態方法在Eclipse中報錯說找不到,解決辦法是加上編譯頭#if (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)和#endif。字體

二、t.env->CallStaticVoidMethod(t.classID, t.methodID, "do not touch me !!");以前是這樣調用這個方法的,結果出錯須要將C++的字符串轉爲jstring,jstring jMsg = t.env->NewStringUTF("do not touch me !!");ui

三、如何打印日誌,經過引入<android/log.h>並添加宏定義而後就能夠在LOGCAT中打印日誌了,而且參考:http://blog.csdn.net/zengraoli/article/details/11644815來修改Android.mk文件。this

四、jni/hellocpp/main.cpp報錯,找不到方法,重啓eclipse便可。。。

五、第四個參數是方法簽名例如:"(Ljava/lang/String;)V",能夠經過在.class文件目錄下打開命令行窗口,輸入命令 javap -s -p ShowRedDialog (-s表示打印簽名信息 -p表示打印全部函數和成員的簽名信息,默認只打印public的簽名信息)。


 修改Android.mk文件,加入要編譯的test.cpp文件:

LOCAL_SRC_FILES := hellocpp/main.cpp \
		   hellocpp/test.cpp \
                   ../../Classes/AppDelegate.cpp \
                   ../../Classes/HelloWorldScene.cpp \

在遊戲中以下修改,添加頭文件:

#if(CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
#include "../proj.android/jni/hellocpp/test.h"
#endif

 添加菜單事件:

void HelloWorld::menuCloseCallback(CCObject* pSender)
{
#if (CC_TARGET_PLATFORM == CC_PLATFORM_WINRT) || (CC_TARGET_PLATFORM == CC_PLATFORM_WP8)
	CCMessageBox("You pressed the close button. Windows Store Apps do not implement a close button.","Alert");
#elif (CC_TARGET_PLATFORM == CC_PLATFORM_ANDROID)
	CCLog("in cocos2d show dialog");
	showMyDialog("dont touch me !!");
#else
    CCDirector::sharedDirector()->end();
#if (CC_TARGET_PLATFORM == CC_PLATFORM_IOS)
    exit(0);
#endif
#endif
}

 以上說的是C++調用JAVA,以後有時間再寫JAVA調用C++。

相關文章
相關標籤/搜索