2013年6月7日金曜日

実験:Locale.getDefault()で得られる情報

実験:Locale.getDefault()で得られる情報

本体設定で端末の言語を変更できる。
言語を変更した後、Locale.getDefault()でLocale情報を取得した場合、言語は当然変わるだろうが、国情報はどうなるのか。

結論:国情報も変わる。

本体設定で言語をEnglishに変更した後、Locale.getDefault()でLocaleを取得し、これでgetCountry()を実行させると、USが返る。

本体設定では「Language」だけを変更するような表記になっているのに、国情報も変わってしまうのだね。むむむ。

ちなみに、getResources().getConfiguration().localeも同様の結果になった。

「オレは英語を話すんだけど、住所地と国籍は日本なんだ」という御仁への配慮は、(以下省略)

2013年5月22日水曜日

Action Barに検索Viewを設ける

Action Barに検索Viewを設ける

検索を行うViewは、EditTextとButtonを併記すれば自作できる。これは簡単だ。
しかし、Androidシステムとして、検索に関する仕掛けが用意されているので、それを用いた方が、利用者の立場から見て親和性の高いViewを構築できる。

この記事は、検索用のViewだけを表示する仕掛けに関してだけである。検索するコマンドは実装していない。

私の所感
SearchView等を使わず、EditTextとButtonを用いて自作する方がずっと簡単だ!
にも関わらず、小難しいSearchViewを作っちゃったってことは、検索に関してマニアックになっている重要視しているってことだね。私としては、このようなマニアックなことは好きなので設計者の意向を尊重してSearchViewを活用してみることにした。

参考にしたサイト
Search Overview
Action Bar..."Action View"とは術語(専門用語)である。このAction Viewの具体例として、例えば、SearchViewがある。
Android Tutorial: Adding Search to Your Apps

簡単なサンプルプログラムを作ってみました。
Action Barに検索用文字列入力欄が表示され、そこに文字列を入力して実行すると、入力した文字列がログに表示されます。

public class MainActivity extends Activity{
    private final String TAG = "MainActivity";

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu){
        getMenuInflater().inflate(R.menu.main, menu);
        SearchView sV;
        SearchManager sM = (SearchManager)getSystemService(Context.SEARCH_SERVICE);
        ComponentName c = getComponentName();
        SearchableInfo si = sM.getSearchableInfo(c);
        MenuItem mi = menu.findItem(R.id.action_Search);
     
/*
 * getActionView()について
 * menuのxmlにおいて、android:actionViewClassで設定したactionViewを取得する。
 * もし、android:actionViewClassが存在しない場合、getActionView()の戻値はnullである。
 */
        sV = (SearchView)mi.getActionView();
        sV.setSearchableInfo(si);
     
        /*
         * setIconifiedByDefault()について
         * falseは、文字列入力領域が表示される。
         * trueは、文字列入力領域は表示されずに、虫メガネアイコンだけが表示される。
         */
        sV.setIconifiedByDefault(false);
        return true;
    }

    @Override
    protected void onNewIntent(Intent intent) {
    /*
     * getIntent()で得られるIntentは、原初の時点でのIntentのままである。
     * setIntent()を使うことにより、最新のIntentに更新される。
     */
        setIntent(intent);
        if(Intent.ACTION_SEARCH.equals(intent.getAction())==true){
            String q = intent.getStringExtra(SearchManager.QUERY);
            Log.d(TAG, "onNewIntent() : query==" + q);
        }
    }
}

AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest
    xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.searchable"
    android:versionCode="1"
    android:versionName="1.0"
    >
    <uses-sdk
        android:minSdkVersion="11"
        android:targetSdkVersion="17"
        >
    </uses-sdk>
    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme"
        >
        <activity
            android:name=".MainActivity"
            android:launchMode="singleTop"
            >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
            <intent-filter>
                <action android:name="android.intent.action.SEARCH" />
            </intent-filter>
            <meta-data
                android:name="android.app.searchable"
                android:resource="@xml/searchable"
                />
        </activity>
    </application>
</manifest>

searchable.xml
<?xml version="1.0" encoding="utf-8"?>
<searchable
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:label="@string/app_name"
    android:hint="@string/SearchHint"
    >
</searchable>

menu/main.xml
<menu xmlns:android="http://schemas.android.com/apk/res/android" >
    <item
        android:id="@+id/action_Search"
        android:actionViewClass="android.widget.SearchView"
        android:showAsAction="always"
        >
    </item>
</menu>

2013年5月9日木曜日

音量の設定が変更されたという通知を受け取る

音量の設定が変更されたという通知を受け取る

副題:本体設定が変更されたことを取得する

標題について調べたところ、下記にコードがありました。
音量設定の取得について

このコードを参考に、下記のとおり私なりに作ってみました。このプログラムの実行中に、音量を変えると、ログが表示されます。
これは、本体設定が変更されたことを取得するためのプログラムなのですね。
一部の機種においては、Settings.System.VOLUME_MUSICの値が11に固定されています。固定されているということは、変更が発生しないということでありますので、その結果、ContentObserver#onChange()が呼ばれないということになってしまいます。この機種においては、"volume_music_speaker"に有効な値が代入されていました。
"volume_music_speaker"は「本体設定」の値の一覧取得プログラムを使って調べました。

public class MainActivity extends Activity{
    final private String TAG = "MainActivity";
    ContentResolver cr;
    MyObserver mo;
    @Override
    protected void onCreate(Bundle savedInstanceState){
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        cr = getContentResolver();
        int i = Settings.System.getInt(cr, "volume_music_speaker", -1);
        Uri u;
        if(i>=0){
            u = Settings.System.getUriFor("volume_music_speaker");
        }
        else{
            u = Settings.System.getUriFor(Settings.System.VOLUME_MUSIC);
        }
        mo = new MyObserver(new Handler());
        cr.registerContentObserver(u, true, mo);
    }

    @Override
    protected void onDestroy() {
        Log.v(TAG, "onDestroy()");
        cr.unregisterContentObserver(mo);
        super.onDestroy();
    }

    private class MyObserver extends ContentObserver{
        final private String TAG = "It's ME!";
        public MyObserver(Handler h){
            super(h);
        }

        @Override
        public void onChange(boolean selfChange, Uri uri) {
            AudioManager am = (AudioManager)getSystemService(Context.AUDIO_SERVICE);
            Log.v(TAG, "onChange(). uri==" + uri +
                ". volume==" + am.getStreamVolume(AudioManager.STREAM_MUSIC));
        }

        @Override
        public void onChange(boolean selfChange){
            onChange(selfChange, null);
        }
    }
}

2013年4月27日土曜日

実験:configuration changeによるActivityの再起動を、コードで実現できるのか

実験:configuration changeによるActivityの再起動過程を、コードで再現できるのか

結論:私がやった限りでは、できなさそうである。

configuration changeが発生すると、Activityが再起動される途中で、onSaveInstanceStateメソッドが呼ばれる。
しかしながら、このメソッドはOSが呼び出すのであって、プログラマがコードで呼び出すことはできない。

onConfigurationChangedメソッド内で、Intentに与えるフラグを操作して、新たにActivityを起動してやれば、onSaveInstanceStateメソッドが呼ばれる。
しかし、この場合、旧Activityのdestroyが、新ActivityのonCreateの後に発生する。この挙動は、私の希望ではない。
finishメソッドを使うと、onSaveInstanceStateメソッドは呼ばれない。

onSaveInstanceStateメソッドは、次のとおり書かれてある。
This method is called before an activity may be killed so that when it comes back some time in the future it can restore its state.
しかしながら、実際には、killされない時にも呼ばれる場合がある。
onSaveInstanceStateメソッド内では、killされることを前提にコードを書いてはいけない。データを保存することに徹した方が良い。

端末がスリープ状態にある時に、Eclipseからデバッグ目的でアプリを起動すると、onSaveInstanceStateメソッド&onDestroy()が繰り返し呼ばれる。この問題に対して、AndroidManifest.xmlにおいてandroid:launchMode="singleTop"を施してみたが、効果は無かった。

Android Configuration Changeに書かれてあるように、onDestroy()の中でgetChangingConfigurations()を使うことも役に立ちそうである。

    @Override
    protected void onDestroy(){
        int iConfig = getChangingConfigurations();
        iConfig &= ActivityInfo.CONFIG_ORIENTATION;
        if(iConfig>0){
            Log.d(TAG, "onDestroy() : ConfigurationChange");
        }

この話題はこれ以上つっこむことなく、ここらで引き揚げることにする。

参考
Android: onSaveInstanceState not being called from activity

2013年4月23日火曜日

RadioButtonの背景色を変更する方法

RadioButtonの背景色を変更する方法

RadioButtonの背景色を変更すると、RadioButtonの形状が崩れることが指摘されています。
Android RadioButton の画像とテキストの間隔を広げる
私は、形状崩れをなんとかしたいと思います。
なお、この現象は少なくともAndroid 4.2.2では発生しません。2.2及び2.3では発生します。

解決方法 その1
xmlにおいて背景色を設定した場合に、形状崩れが発生するのです。実行時に背景色を設定すれば問題は発生しません。

解決方法 その2
RadioGroupを使用せず、javaのコードにおいて、RadioButtonに対する押し下げをonClickメソッドで補足し、他のRadioButtonをsetChecked(false)してやるのです。なお、個々のRadioButtonの上位に個別にLinearLayout等を配置し、このLinearLayoutに対して背景色を設定します。

2013年4月16日火曜日

自分のアプリをZ順の最上位にする

自分のアプリをZ順の最上位にする

自分のアプリのActivityが、画面の前面に無い場合、つまりZ順における最上位では無い場合、詳しく言えば利用者の視点で端末の画面を見た時に他のアプリが表示されていて自分のアプリが見えない場合、自分のアプリのActivityを前面に出してやるにはどうするのか。

自分のアプリのActivityが稼働していない状態であれば、Intentを使えば良いことはわかります。次のとおりです。

Intent it;
it = new Intent(this, MainActivity.class);
it.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(it);//Activityを起動し,上にしてやる

しかしながら、他のアプリが前面に出ていて、自分のアプリのActivityがバックグラウンドにある場合、つまり自分のActivityが生きておりかつ、後ろに隠れている場合、自分のアプリのActivityを前面に出す方法はどうすれば良いでしょうか。
多くのサイトではIntentを使うように案内されています。
確かに、Intentすれば、前面には出ますが、IntentされたActivityは新規に作成されてしまうのです。onCreateメソッドが実行されてしまい、Activityが複数存在することになってしまいます。

このため、何か違うような気がしたのでrequestFocus()やbringToFront()を試しましたがうまくいきませんでした。次のサイトで議論があるように、getWindow().setActive()でもダメなようです。
ブロードキャストの受信で、自分自身を前面に表示させる方法について

結論は、やっぱり、Intentを使うのです。コードは次のとおりです。Serviceにおいて実行させます。

Intent it;
it = new Intent(this, MainActivity.class);//ここでのthisはServiceです。
it.setFlags(Intent.FLAG_ACTIVITY_SINGLE_TOP);
it.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
startActivity(it);//Activityを上にしてやる

これを使えば、ActivityのonCreateメソッドは実行されず、複数のActivityが生成されることなく、自分のアプリをZ順の最上位にすることができます。Intent.FLAG_ACTIVITY_NEW_TASKがあるため、新規に作成されてしまうように思えますが、新規には作成されませんし、これが無いと実行時にエラーになります。
最近の開発環境は良くなっており、エラーになった場合、このIntent.FLAG_ACTIVITY_NEW_TASKを設定しなさいという趣旨のメッセージがログに表示されます。

過去の私の記事において、Intent.FLAG_ACTIVITY_SINGLE_TOPは使う機会が無いという趣旨の記述をしてしまいました。むむむ! 起動モード
そうでもないということですね。

2013年4月15日月曜日

SimpleAdapter : レコード毎に色を変えて表示する

SimpleAdapter : レコード毎に色を変えて表示する

文字列だけで構成されるレコードを一覧にして表示させた場合、レコード間の区切りが見分けにくくなります。
その対処方法として、 レコード毎に色を変えて表示するとか、レコード間の幅を大きくするという方法があります。
javaのコードの概要は下記のとおりです。概要を示したものであるため、このママでは動作しません(見れば分かる)
色分けのフラグをレコード毎に設定し、そのフラグに基づいて背景色を変更すること、そして、背景色の設定にはアルファ値を適当な値で設定することがポイントです。アルファ値を0xffに設定した場合、利用者がそのレコードを選択しても「選択したことを示す背景色」が表示されません。

    ArrayList<Map<String, Object>> al;//List表示用データ
        al = new ArrayList<Map<String, Object>>();
        
            b = cs.moveToLast();
            int i = 0;
            while(b==true){
                String s;
                Map<String, Object> m;
                m = new HashMap<String, Object>();
                s = cs.getString(cs.getColumnIndex("Keyword"));
                m.put("Keyword", s);
                m.put("Background", i%2);//ゼロか1を代入する
                al.add(m);
                i++;
                b = cs.moveToPrevious();
            }
        }
        cs.close();
        
        String sFrom[] = {"Keyword", "Background"};
        int iTo[] = new int[]{ R.id.Keyword_Text, R.id.Layout_Keyword_Background };
        
        SimpleAdapter sa;
        SimpleAdapter.ViewBinder savb;
        savb = new SimpleAdapter.ViewBinder(){
            @Override
            public boolean setViewValue(View v, Object data, String s){
                if(v.getId()==R.id.Layout_Keyword_Background){
                    int i = (Integer)data;
                    if(i==1) v.setBackgroundColor(0x60ffdddd);
                    else v.setBackgroundColor(0x00ffffff);
                    return true;
                }
                return false;
            }
        };
        
        sa = new SimpleAdapter(mainActivity, al, R.layout.list_keyword, sFrom, iTo);
        sa.setViewBinder(savb);
        ListView lv;
        lv = (ListView)mainActivity.findViewById(R.id.ListKeyword);
        lv.setOnItemClickListener(this);
        lv.setAdapter(sa);
        lv.setDividerHeight(10);

list_keyword.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/Layout_Keyword_Background"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:padding="10dp"
    >
    
    <TextView
        android:id="@+id/Keyword_Text"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:textIsSelectable="false"
        >
    </TextView>
</LinearLayout>

面白い動画を見つけました
メ~テレ ぬいぐるみのラパン