Fragmentを使ったtabの中のWebViewの戻るボタンを実装する

Fragmentを使ったタブの実装がドキュメントに書かれている。TabManagerの内部クラスTabInfoにgetterを実装する。

    static final class TabInfo {
        private final String tag;
        private final Class<?> clss;
        private final Bundle args;
        private Fragment fragment;

        TabInfo(String _tag, Class<?> _class, Bundle _args) {
            tag = _tag;
            clss = _class;
            args = _args;
        }

        public Fragment getFragment() {
            return fragment;
        }
    }

activityに以下のコードを記述する。

    @Override
    public boolean dispatchKeyEvent(KeyEvent event) {
        if (event.getAction() == KeyEvent.ACTION_DOWN) {
            WebView webView = (WebView) mTabManager.mLastTab.getFragment().getView().findViewById(R.id.webview);
            if (webView != null && event.getKeyCode() == KeyEvent.KEYCODE_BACK) {
                if (webView.canGoBack()) {
                    webView.goBack();
                    return true;
                }
                else {
                    finish();
                }
            }
        }
        return super.dispatchKeyEvent(event);
    }

HMAC-MD5を計算する

■PHP

hash_hmac('md5', $str, 'key');

■Python

import hmac
from hashlib import sha1
from hashlib import md5
hmac.new("key", "value", md5).hexdigest()
hmac.new("key", "value", sha1).hexdigest()

■Android

public class HmacMD5 {
private static final String ALGORISM = “HmacMD5”;
private static final String S = “key”;
public static String get(String str) {
SecretKeySpec secretKeySpec = new SecretKeySpec(S.getBytes(), ALGORISM);
try {
Mac mac = Mac.getInstance(ALGORISM);
mac.init(secretKeySpec);
byte[] result = mac.doFinal(str.getBytes());
return byteToString(result);
}
catch (NoSuchAlgorithmException e) {
e.printStackTrace();
}
catch (InvalidKeyException e) {
e.printStackTrace();
}
return “”;
}

private static String byteToString(byte [] b) {
StringBuffer buffer = new StringBuffer();
for (int i = 0; i < b.length; i++) { int d = b[i]; d += (d < 0)? 256 : 0; if (d < 16) { buffer.append("0"); } buffer.append(Integer.toString(d, 16)); } return buffer.toString(); } } [/sourcecode]

■iOS

日本語入力に対応するためにstackoverflowから持ってきたコードに少々手を入れた。

+ (NSString *)HMACMD5WithKey:(NSString *)data
{
const char *cKey = [@”key” cStringUsingEncoding:NSUTF8StringEncoding];
const char *cData = [data cStringUsingEncoding:NSUTF8StringEncoding];
const unsigned int blockSize = 64;
char ipad[blockSize];
char opad[blockSize];
char keypad[blockSize];

unsigned int keyLen = strlen(cKey);
CC_MD5_CTX ctxt;
if (keyLen > blockSize) {
CC_MD5_Init(&ctxt);
CC_MD5_Update(&ctxt, cKey, keyLen);
CC_MD5_Final((unsigned char *)keypad, &ctxt);
keyLen = CC_MD5_DIGEST_LENGTH;
}
else {
memcpy(keypad, cKey, keyLen);
}

memset(ipad, 0x36, blockSize);
memset(opad, 0x5c, blockSize);

int i;
for (i = 0; i < keyLen; i++) { ipad[i] ^= keypad[i]; opad[i] ^= keypad[i]; } CC_MD5_Init(&ctxt); CC_MD5_Update(&ctxt, ipad, blockSize); CC_MD5_Update(&ctxt, cData, strlen(cData)); unsigned char md5[CC_MD5_DIGEST_LENGTH]; CC_MD5_Final(md5, &ctxt); CC_MD5_Init(&ctxt); CC_MD5_Update(&ctxt, opad, blockSize); CC_MD5_Update(&ctxt, md5, CC_MD5_DIGEST_LENGTH); CC_MD5_Final(md5, &ctxt); const unsigned int hex_len = CC_MD5_DIGEST_LENGTH*2+2; char hex[hex_len]; for(i = 0; i < CC_MD5_DIGEST_LENGTH; i++) { snprintf(&hex[i*2], hex_len-i*2, "%02x", md5[i]); } NSData *HMAC = [[NSData alloc] initWithBytes:hex length:strlen(hex)]; NSString *hash = [[[NSString alloc] initWithData:HMAC encoding:NSUTF8StringEncoding] autorelease]; [HMAC release]; return hash; } [/sourcecode]

fragmentに値を渡す

■渡す側

FragmentTransaction ft = getActivity().getSupportFragmentManager().beginTransaction();

SearchFragment fragment = (SearchFragment) Fragment.instantiate(getActivity(), SearchFragment.class.getName(), null);
Bundle bundle = new Bundle();
bundle.putString("word", word);
fragment.setArguments(bundle);

ft.replace(R.id.content, fragment).addToBackStack("result");
ft.commit();
getActivity().getSupportFragmentManager().executePendingTransactions();

■受け取り側

mWord = getArguments().getString("word");

SimpleCursorAdapterを元に作ったListViewにfilterを付加する

以下のように実装しているとする。

mListView.setTextFilterEnabled(true);
mAdapter = new SimpleCursorAdapter(
        getActivity().getApplicationContext(),
        R.layout.list,
        null,
        new String[]{Helper.FACE, Helper.TAG},
        new int[]{R.id.list_face, R.id.list_tag}
);
mListView.setListAdapter(mAdapter);

以下のようにして表示されているリストをフィルタリングする。

public void onTextChanged(CharSequence s, int start, int before, int count) {
    mListView.setFilterText(s.toString);
}

これだけでは上手く動作しない。

■解決策

adapterをListViewにセットする前に以下のように記述する必要がある。

mAdapter.setFilterQueryProvider(new FilterQueryProvider() {
    @Override
    public Cursor runQuery(CharSequence constraint) {
        String word = "%" + constraint.toString() + "%";
        Cursor cursor = getActivity().managedQuery(
            Provider.URI,
            new String[]{Helper.ID, Helper.FACE, Helper.TAG, Helper.CREATED},
            "`" + Helper.FACE + "` LIKE ? OR `" + Helper.TAG + "` LIKE ?",
            new String[]{word, word},
            Helper.CREATED + " DESC"
        );
        return cursor;
    }
});

android-support-v4.jarを使ったプロジェクトのテストをする

テスト対象のプロジェクトにおいて以下の操作をする。

propaties > Java Build Path > Order and Export > android-support-v4.jarにチェックを入れる。

.classpath

以下のようにexported属性が付加されている。

<classpathentry exported="true" kind="lib" path="libs/android-support-v4.jar"/>

悪い例

テストプロジェクトにandroid-support-v4.jarをインポートすると以下のようなエラーがでる。

java.lang.RuntimeException: Exception during suite construction
at android.test.suitebuilder.TestSuiteBuilder$FailedToCreateTests.testSuiteConstructionFailed(TestSuiteBuilder.java:239)
at java.lang.reflect.Method.invokeNative(Native Method)
at android.test.AndroidTestRunner.runTest(AndroidTestRunner.java:169)
at android.test.AndroidTestRunner.runTest(AndroidTestRunner.java:154)
at android.test.InstrumentationTestRunner.onStart(InstrumentationTestRunner.java:529)
at android.app.Instrumentation$InstrumentationThread.run(Instrumentation.java:1448)
Caused by: java.lang.reflect.InvocationTargetException
at java.lang.reflect.Constructor.constructNative(Native Method)
at java.lang.reflect.Constructor.newInstance(Constructor.java:415)
at android.test.suitebuilder.TestMethod.instantiateTest(TestMethod.java:87)
at android.test.suitebuilder.TestMethod.createTest(TestMethod.java:73)
at android.test.suitebuilder.TestSuiteBuilder.addTest(TestSuiteBuilder.java:263)
at android.test.suitebuilder.TestSuiteBuilder.build(TestSuiteBuilder.java:185)
at android.test.InstrumentationTestRunner.onCreate(InstrumentationTestRunner.java:373)
at android.app.ActivityThread.handleBindApplication(ActivityThread.java:3285)
at android.app.ActivityThread.access$2200(ActivityThread.java:117)
at android.app.ActivityThread$H.handleMessage(ActivityThread.java:987)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:123)
at android.app.ActivityThread.main(ActivityThread.java:3728)
at java.lang.reflect.Method.invokeNative(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:864)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:622)
at dalvik.system.NativeStart.main(Native Method)

参考

androidでレンダリング後のViewのサイズを取得する

onClickなどのユーザイベント実行時に取得するならば以下の方法で取得ができる。

mView.setOnClickListener(new View.OnClickListener(){
    @Override
    public void onClick(View v) {
        int width = v.getWidth();
        int height = v.getHeight();
    }
});

これを利用して以下のようにすると正しく取得できない。

@Override
public void onCreate(Bundle savedInstanceState) {
    // ...
    int width = mView.getWidth();// 0
    int height = mView.getHeight();// 0
}

onCreate, onStart, onResumeではレンダリングが終了していないためサイズが0となる。

■解決策

以下のようにViewTreeObserverを利用することでレンダリング後のサイズを取得することができる。

@Override
public void onCreate(Bundle savedInstanceState) {
    // ...
    ViewTreeObserver viewTreeObserver = mView.getViewTreeObserver();
    viewTreeObserver.addOnGlobalLayoutListener(new ViewTreeObserver.OnGlobalLayoutListener() {
        @Override
        public void onGlobalLayout() {
            int width = mView.getWidth();
            int height = mView.getHeight();
        }
    });
}

androidでActivityが切り替わるときのアニメーションを変える

activityが切り替わるときにアニメーションをさせたくなくなった。

■実装

res/values/style.xml

以下のようにテーマを定義する。

<resource>
    <style name="NoAnimationTheme" parent="android:Theme">
        <item name="android:windowAnimationStyle">@style/Animation.Activity</item> 
    </style>
    <style name="Animation.Activity" parent="android:Animation.Activity">
        <item name="android:activityOpenEnterAnimation">@anim/activity_open_enter</item>
        <item name="android:activityOpenExitAnimation">@anim/activity_open_exit</item>
        <item name="android:activityCloseEnterAnimation">@anim/activity_close_enter</item>
        <item name="android:activityCloseExitAnimation">@anim/activity_close_exit</item>
    </style>
</resource>

anim/activity_open_enter.xml

startActivityで開くActivityのanimation。

<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:interpolator="@android:anim/accelerate_interpolator">
    <translate
        android:fromXDelta="100%"
        android:toXDelta="0%"
        android:duration="0"
        android:fillAfter="true"
        android:fillEnabled="true"/>
</set>

anim/activity_open_exit.xml

startActivityで閉じるActivityのanimation。

<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:interpolator="@android:anim/accelerate_interpolator">
    <translate
        android:fromXDelta="0%"
        android:toXDelta="-100%"
        android:duration="0"
        android:fillAfter="true"
        android:fillEnabled="true"/>
</set>

anim/activity_close_exit.xml

finishで閉じるActivityのanimation。

<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:interpolator="@android:anim/accelerate_interpolator">
    <translate
        android:fromXDelta="0%"
        android:toXDelta="100%"
        android:duration="0"
        android:fillAfter="true"
        android:fillEnabled="true"/>
</set>

anim/activity_close_open.xml

finishで開くActivityのanimation。

<?xml version="1.0" encoding="utf-8"?>
<set xmlns:android="http://schemas.android.com/apk/res/android"
    android:interpolator="@android:anim/accelerate_interpolator">
    <translate
        android:fromXDelta="-100%"
        android:toXDelta="0%"
        android:duration="0"
        android:fillAfter="true"
        android:fillEnabled="true"/>
</set>
参考

上述の内容などは以下のような参考サイトに情報がある。

AndroidManifest.xml

アプリケーション全体に適用する

以下のようにapplicationタグに記述することでアプリケーション全体に変更したアニメーションを適用できる。

<application
    android:icon="@drawable/icon"
    android:label="@string/app_name"
    android:theme="@style/NoAnimationTheme">
アクティビティに適用する

以下のようにactivityタグに記述することで単一のアクティビティにのみ変更したアニメーションを適用できる。

<activity
    android:label="@string/app_name"
    android:name=".MainActivity">
</activity>

■応用

上述の方法だと特定の条件下でのみアニメーションを変えるといったことができない。条件によってanimationを変えたい場合は、以下のようにすることで対応できる。

public class MainActivity extends Activity {
    /**
     * 起動時に実行される
     */
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        setTheme(R.style.NoAnimationTheme);// super.onCreateの前に実行する必要がある
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
    }
}

androidでギャラリーと連携する

■実装

ギャラリーの起動

以下のようにすることで(画像データに選択肢を絞って)ギャラリーアプリを起動することができる。

Intent intent = new Intent();
intent.setType("image/*");
intent.setAction(Intent.ACTION_GET_CONTENT);
startActivityForResult(intent, REQUESTCODE);

ギャラリーからデータを受け取る

protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    if (requestCode == REQUESTCODE && resultCode == RESULT_OK) {
        //data.getData();//ギャラリーで選択したファイルのuriが格納されている
    }
}

uriからpathの文字列を取得

以下のようにすることでuriからpathの文字列を取得することができる。

private final String getPathFromUri(Uri uri) {
    String[] projection = { MediaStore.Images.Media.DATA };
    Cursor cursor = managedQuery(uri, projection, null, null, null);
    int column_index = cursor.getColumnIndexOrThrow(MediaStore.Images.Media.DATA);
    cursor.moveToFirst();
    return cursor.getString(column_index);
}
参考

androidで顔検出する

int maxFaces = 10;// 検出する顔の最大数
FaceDetector detector = new FaceDetector(bitmap.getWidth(), bitmap.getHeight(), maxFaces);
FaceDetector.Face[] faces = new FaceDetector.Face[maxFaces];
ArrayList<Bitmap> recognized = new ArrayList<Bitmap>();

// bitmapの中から顔を検出してfacesに格納する
int num = detector.findFaces(bitmap, faces);

// facesに格納されたデータから座標情報を取り出してbitmapを切り出す
for (int i = 0; i < num; i++) {
    FaceDetector.Face face = faces[i];
    PointF point = new PointF(0.0f, 0.0f);
    face.getMidPoint(point);

    int x = (int) (point.x - 1.5f * face.eyesDistance());
    int y = (int) (point.y - 2 * face.eyesDistance());
    int width  = (int) (face.eyesDistance() * 3);
    int height = (int) (face.eyesDistance() * 4);
    x = (x < 0)? 0 : x;
    y = (y < 0)? 0 : y;
    width = (width > bitmap.getWidth())? bitmap.getWidth() : width;
    height = (height > bitmap.getHeight())? bitmap.getHeight() : height;
    recognized.add(Bitmap.createBitmap(bitmap, x, y, width, height));
}
return recognized;