2012年1月4日水曜日

IntentServiceでToastを表示する

IntentServiceではToastを表示できない。
また、不要になったらきっちり死ぬようにしているServiceでもToastが表示できない。
出力されている警告を見ると、Toast表示を行うUIスレッドが死んでるから表示要求が届かなかったとのこと。少なくとも表示メッセージがUIスレッドに届くまではServiceが生存してなきゃいけないらしい。

ということで、Toast表示用のServiceを作った。
public class ToastService extends Service {
    private static final String TAG = ToastService.class.getName();

    public static void showToast(Context context, int strResId) {
        Intent service = new Intent(context, ToastService.class);
        service.putExtra(EXTRA_MESSAGE, strResId);
        context.startService(service);
    }


    private ToastHandler mToastHandler;

    public static final String EXTRA_MESSAGE = "EXTRA_MESSAGE";

    public static final int MESSAGE_DISPLAY = 0;
    public static final int MESSAGE_TIMER = 1;

    private class ToastHandler extends Handler {
        private Toast mToast = null;
        private int curMsgResId;

        public ToastHandler() {
            super();
        }

        @Override
        public void handleMessage(Message msg) {

            switch (msg.what) {
            case MESSAGE_DISPLAY: {
                if (curMsgResId != msg.arg2) {
                    if (mToast != null) {
                        mToast.cancel();
                    }
                    curMsgResId = msg.arg2;
                    mToast = Toast.makeText(
                            getApplication(),
                            curMsgResId,
                            Toast.LENGTH_LONG);
                    mToast.show();
                }
                removeMessages(MESSAGE_TIMER);
                sendMessageDelayed(obtainMessage(MESSAGE_TIMER, msg.arg1, 0), 4 * 1000);
                break;
            }
            case MESSAGE_TIMER: {
                stopSelf(msg.arg1);
                break;
            }
            }
        }
    }

    @Override
    public void onCreate() {
        super.onCreate();
        mToastHandler = new ToastHandler();
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        int msgResId = intent.getIntExtra(EXTRA_MESSAGE, 0);
        Message msg = mToastHandler.obtainMessage(MESSAGE_DISPLAY, startId, msgResId);
        mToastHandler.sendMessage(msg);
        
        return START_NOT_STICKY;
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
    }

    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
}

ViewPagerで表示してるViewを取得する(Support Package r3)

アニメーションとかスクロール周り弄るの面倒くさいから、こういうライブラリがあると助かるんだけど、いざ使おうとなるとかゆい所に手が届かない。

例えば、ViewPagerで保持してるViewを取得して動的に何か表示を変更したい時。

一番簡単なのは、ViewPagerのソースを持ってきて、以下のメソッド追加してあげること。
以下のメソッドは一応position指定できるけど、mItemsに保持してるのはデフォルトだと前後のページのみなので正直微妙。

public View getItemView(int position) {
View v = null;
for (ItemInfo item : mItems) {
if (position == item.position) {
v = (View) item.object;
}
}
return v;
}

2011年10月27日木曜日

WebView#loadDataが文字化けする(Android3.0以降)

Android2.3以前であれば、METAタグにContent-Typeを指定することで解決した。
webView.loadData(
    "<html>"
      + "<head>"
      + "<meta http-quiv=\"Content-Type\" content=\"text/html charset=utf-8\" />"
      + "</head>"
      + "<body>"
      + htmlBodyData
      + "</body>"
      + "</html>",
    "text/html",
    "utf-8");

しかし、Android3.0以降では上記の対処でも文字化けした。
いろいろ調べた結果、WebView#loadDataWithBaseURLを使用することで、とりあえず文字化けを回避できた。
webView.loadDataWithBaseURL(
    null,
    "<html>"
      + "<head>"
      + "<meta http-quiv=\"Content-Type\" content=\"text/html charset=utf-8\" />"
      + "</head>"
      + "<body>"
      + htmlBodyData
      + "</body>"
      + "</html>",
    "text/html",
    "utf-8",
    null);

ソースを軽く眺めると、loadDataとloadDataWithBaseURLでは最初のデータの入れ方からして違っていたので、loadDataがメンテされてないんだろうな…。

2011年10月26日水曜日

RのID等の値が定数でなくなったためswitch文でエラー(ADTr14)

Non-constant Fields in Case Labels

どこかで聞いてたけど、変換の仕方忘れてたのでメモ。

switchステートメントにカーソル合わせて、WindowsはCtrl+1、MacはCmd+1を押す。
すると、ifステートメントに早変わり!

2011年10月3日月曜日

Android端末のAPN切り替え

Managing APN Settings on Google Android
基本的には上記リンク先でいける。

調べてて気になったことを以下に。
・端末(OSバージョン?)によって、ソース内コメントのテーブル定義より拡張されていることがある。GalaxySは'nwkname'というカラムがあるが、Xperia Arcにはなかった。
・またpreferapnによるAPN変更時は、ContentValuesのkeyを'apn_id'とする必要がある。(値は'_id'カラムのものを使用する)指定した条件が一意のものでないと、APNは切り替わらない。
SharedPrefernceにapn_idという使用中のAPN情報を持っているようだ。


com.android.providers.telephony.TelephonyProviderを読むと、もう少し細かい仕組みがわかった。(早くandroid.git.kernel.org復旧しないかな)

指定できるURIは5つある。
content://telephony/carriers/
=>URL_TELEPHONY
content://telephony/carriers/current
=>URL_CURRENT
content://telephony/carriers/#
=>URL_ID
content://telephony/carriers/restore
=>URL_RESTOREAPN
content://telephony/carriers/preferapn
=>URL_PREFERAPN

それぞれ、SQLクエリによって動作が異なる。
URL_TELEPHONY
=>条件通りのCRUDを行う。

URL_CURRENT
=>query()はcurrent列がNULLでないAPNを取得。insertは特殊で、current IS NOT NULLの行をNULL化してnumericが同一の行のcurrent列を1にする。delete/updateは特徴なし。

URL_ID
=>uriのgetLastPathSegmentなどを条件としてCRUDを行うようだけど、動作よくわからず。このUriに対してLastPath叩くと"carriers"が返ってくるし。便利な使い方でもあるんだろか。

URL_RESTOREAPN
=>delete()でのみ使用可能。preferapnの設定や追加したAPNを初期化する。

URL_PREFERAPN
=>query()は現在設定されているAPNを返す。insert/updateはvaluesで指定したapn_idを設定して、APNを切り替える。deleteはpreferredAPNを初期化する。

以下は、調べる時に使ったソースコード。検証用にログ出すだけのメソッド多め。APNクラスはsetter/getterが無駄に長いので省略。

package com.test.apn;

import java.util.LinkedList;
import java.util.List;

import android.content.ContentResolver;
import android.content.ContentValues;
import android.content.Context;
import android.database.Cursor;
import android.net.Uri;
import android.util.Log;

import com.test.apn.entity.APN;

public class ApnManager {

    private static final String TAG = ApnManager.class.getName();
    
    private static final Uri APN_TABLE_URI = Uri.parse("content://telephony/carriers/");
    private static final Uri APN_PREFERRED_URI = Uri.parse("content://telephony/carriers/preferapn");
    private static final Uri APN_CURRENT_URI = Uri.parse("content://telephony/carriers/current");
    private static final Uri APN_ID_URI = Uri.parse("content://telephony/carriers/#");
    private static final Uri APN_RESTORE_URI = Uri.parse("content://telephony/carriers/restore");
    
    private Context mContext;

    public ApnManager(Context context) {
        mContext = context;
    }

    public List<APN> getApnList() {
        ContentResolver resolver = mContext.getContentResolver();
        Cursor c = resolver.query(APN_TABLE_URI, null, null, null, null);
        printAllData(c);
        
        c.moveToFirst();
        List<APN> apns = new LinkedList<APN>();
        while (! c.isAfterLast()) {
            apns.add(createAPNEntity(c));
            c.moveToNext();
        }
        c.close();
        
        return apns;
    }
    
    public APN getCurrentPreferredApn() {
        ContentResolver resolver = mContext.getContentResolver();
        Cursor c = resolver.query(APN_PREFERRED_URI, null, null, null, null);
        printAllData(c);
        
        if (! c.moveToFirst()) {
            return null;
        }
        
        APN apn = createAPNEntity(c);
        c.close();
        return apn;
    }
    
    public void restoreApn() {
        ContentResolver cr = mContext.getContentResolver();
        int restoreCount = cr.delete(APN_RESTORE_URI, null, null);
        Log.d(TAG, "restore = " + restoreCount);
    }
    
    public APN getApn(long id) {
        ContentResolver resolver = mContext.getContentResolver();
        Cursor c = resolver.query(APN_TABLE_URI, null, "_id="+id, null, null);
        if (! c.moveToFirst()) {
            return null;
        }

        APN apn = createAPNEntity(c);
        c.close();
        
        return apn;
    }
    
    public void setDefaultApn(long id) {
        ContentResolver cr = mContext.getContentResolver();
        ContentValues values = new ContentValues();
        values.put("apn_id", id);
        int updateCount = cr.update(APN_PREFERRED_URI, values, null, null);
        Log.d(TAG, "update = " + updateCount);
    }
    
    public void getCurrentApn() {
        ContentResolver resolver = mContext.getContentResolver();
        Cursor c = resolver.query(APN_CURRENT_URI, null, null, null, null);
        Log.d(TAG, "current => ");
        printAllData(c);
        
        c.moveToFirst();
        List<APN> apns = new LinkedList<APN>();
        while (! c.isAfterLast()) {
            apns.add(createAPNEntity(c));
            c.moveToNext();
        }
        c.close();
        
        return;
    }
    
    public void getApnId() {
        ContentResolver resolver = mContext.getContentResolver();
        Cursor c = resolver.query(APN_ID_URI, null, null, null, null);
        printAllData(c);
        
        c.moveToFirst();
        List<APN> apns = new LinkedList<APN>();
        while (! c.isAfterLast()) {
            apns.add(createAPNEntity(c));
            c.moveToNext();
        }
        c.close();
    }
    
    private APN createAPNEntity(Cursor c) {
        APN apn = new APN();
        
        apn.setId(c.getInt(c.getColumnIndex("_id")));
        apn.setName(c.getString(c.getColumnIndex("name")));
        apn.setNumeric(c.getString(c.getColumnIndex("numeric")));
        apn.setMcc(c.getString(c.getColumnIndex("mcc")));
        apn.setMnc(c.getString(c.getColumnIndex("mnc")));
        apn.setApn(c.getString(c.getColumnIndex("apn")));
        apn.setUser(c.getString(c.getColumnIndex("user")));
        apn.setServer(c.getString(c.getColumnIndex("server")));
        apn.setPassword(c.getString(c.getColumnIndex("password")));
        apn.setProxy(c.getString(c.getColumnIndex("proxy")));
        apn.setPort(c.getString(c.getColumnIndex("port")));
        apn.setMmsproxy(c.getString(c.getColumnIndex("mmsproxy")));
        apn.setMmsport(c.getString(c.getColumnIndex("mmsport")));
        apn.setMmsc(c.getString(c.getColumnIndex("mmsc")));
        apn.setType(c.getString(c.getColumnIndex("type")));
        apn.setCurrent(c.getInt(c.getColumnIndex("current")));
        
        return apn;
    }
    
    private void printAllData(Cursor c) {
        if (! c.moveToFirst()) {
            return;
        }
        
        StringBuilder columnLog = new StringBuilder();
        String[] columns = c.getColumnNames();
        for (String column : columns) {
            columnLog.append(column);
            columnLog.append(",");
        }
        Log.d(TAG, columnLog.toString());
        
        int colSize = c.getColumnCount();
        while (! c.isAfterLast()) {
            StringBuilder values = new StringBuilder();
            for (int i = 0; i < colSize; i++) {
                values.append(c.getString(i));
                values.append(",");
            }
            Log.d(TAG, values.toString());
            
            c.moveToNext();
        }
        
        return;
    }
}

2011年9月28日水曜日

workerで動作するApacheでphp-mysqlが動作しない

http://www.bruteberry.com/archives/2011/04/phpinfomysql.php

redmineインストール済みのCentOSにphpMyAdmin入れようとして、DBに接続できなくて結構ハマッたのでメモ。
とりあえずApacheの設定をデフォルトのpreforkに戻して解決。

2011年9月22日木曜日

In-app Billingの設定をCSVインポート/エクスポートできる機能がデベロッパーコンソールに追加

http://developer.android.com/guide/market/billing/billing_admin.html#billing-bulk-add


In-app Billingの設定をCSVインポート/エクスポートできる機能がデベロッパーコンソールに追加されてた。いつの間にか。
これは地味に嬉しい機能!
IDはコンテンツ管理してるリモートサーバでも識別材料とするから、リモートサーバを一次ソースとしてCSVに吐き出せば、デベコンで設定するときのtypoによる事故防止等につながる。

ID手打ちとかアナログ過ぎてアレだと思ってたので、少しは改善へ向かってる模様。
あとは、Google Checkoutみたいにアプリの詳細なダウンロード数だけを閲覧できるような仕組みがあるとよいのだけれど。デベコンのアカウントだけができることが多すぎる。