このページの本文へ

前へ 1 2 次へ

これで作れる! Androidのアプリケーション 第9回

Androidアプリで複数の項目を表示するリストビューを使う

2010年08月27日 12時00分更新

文● 塩田紳二

  • この記事をはてなブックマークに追加
  • 本文印刷

AndroidのListViewとは?

 今回は、Androidの画面パーツ(ウィジェット)の中でも、特に複雑なListViewを見ていくことにしましょう。ListViewは複数の項目を表示してスクロールなどが可能なウィジェットですが、個々の表示にユーザーの指定する任意のViewを使うことができます。

WorldClockのListView

 WorldClockでは、このListViewをItemEditで、タイムゾーンを表示させるために使っています。まずは、ListView自体は、レイアウトファイルであるedititem.xml(res/layoutおよびres/layout-landフォルダ)にあります。Eclipse用のAndroidプラグインにはバグがあり、ListViewのプロパティで「Fast scroll enabled」にしてあると、レイアウトのビジュアル表示がエラーとなって、表示ができなくなります。レイアウトを見たい場合には、アウトラインから「elvTimeZone」を選び、プロパティで上記の項目を「false」を一時的に設定してください。

 プログラムコードであるEditItem.javaでは、onCreateの先頭で、

setContentView(R.layout.edititem);

として、前記のxmlファイルからアクティビティを生成されており、ListViewは82行目で、

tzlist = (ListView) this.findViewById(R.id.elvTimeZone);

として変数tzlistで参照できるようにしてあります。

 ListViewでは、表示する項目を設定する必要がありますが、複数の項目があり、また通常多数なので、たいていは配列などに保存してあるはずです。このため、内容をセットして表示させるにはArrayAdapterを使います。

 ArrayAdapterは、ListViewに表示するデータを配列として渡すと同時にそのビューを提供します。単純なビュー(たとえばテキストを表示するだけ)であれば、直接レイアウトファイル(のリソースID)を渡してもいいのですが、ちょっと複雑な表示をしたい場合には、ビューをArrayAdapterの中で提供します。このEditItemでは、MyListAdapterというクラスを別に作ってあり、これをListViewにセットしています。

MyListAdapter arrayAdapter = new MyListAdapter(this,timeZoneIDs);
tzlist.setAdapter(arrayAdapter);

MyListAdapterは、EditItem.javaの264行目で定義されています(以下のリスト1を参照のこと)。ArrayAdapterは、配列の要素の型を指定して生成します。ここでは、String型としています。

リスト1

private class MyListAdapter extends ArrayAdapter<String> {
  private Boolean mCityNameMode;
  private LayoutInflater myInflater;
  private TextView timezoneid;
  private TextView timezonename;
  private TextView timezoneoffset;
  private TextView timezonedst;
  public MyListAdapter(Context context,String[] objects) {
    super(context, 0, objects);
    mCityNameMode=false;
    myInflater = (LayoutInflater) context.getSystemService(
      Context.LAYOUT_INFLATER_SERVICE);
  }

  @Override
  public View getView(int position,
    View convertView, ViewGroup parent) {
    if (convertView == null ){
      convertView = myInflater.inflate(R.layout.timezonelist, null);
    }
    String timezone = this.getItem(position);
    TimeZone tz = TimeZone.getTimeZone(timezone);
    timezoneid = (TextView)convertView.findViewById(R.id.inListTV);
    timezonename = (TextView)convertView.findViewById(       R.id.inListLongName);
    timezonedst = (TextView)convertView.findViewById(R.id.inListDST);
    timezoneoffset = (TextView)convertView.findViewById(R.id.inListOffset);
    if (mCityNameMode ){
      String[] temp = timezone.split("/");
      String str = temp[temp.length-1];
      str = str.replaceAll("_", " ");
      timezoneid.setText(str);
    } else {
      timezoneid.setText( timezone);
    }
    timezonename.setText(tz.getDisplayName(
      false, java.util.TimeZone.LONG));
    Calendar now = Calendar.getInstance(tz);
    timezonedst.setTextColor(
      tz.inDaylightTime(now.getTime()) ? Color.WHITE : Color.DKGRAY);
    timezoneoffset.setText("GMT"+DateFormat.format("z", now));
    return convertView;
  }
  public void setCityNameMode(Boolean mode){
    mCityNameMode=mode;
  }
}

 ArrayAdapterは、さまざまな型の要素を持つ配列を使うため、ジェネリックを使って、

ArrayAdapter<型>

として定義されます。型には、StringやIntegerなどの要素の型を設定します。今回の場合は、

private class MyListAdapter extends ArrayAdapter<String>

としており、MyListAdapterは、文字列型を要素に持つ配列を扱うArrayAdapterになっています。Javaの中でもジェネリック(型を事前に指定しないで処理を記述すること)は複雑な機能なので、市販の参考書などを参照してください。

 このクラスで定義しているのは、コンストラクタとgetViewメソッドです。setCityNameModeメソッドは、動作を変えるためのメソッドで、ArrayAdapterから継承したメソッドではありません。  ArrayAdapterクラスのコンストラクタは6種類あるのですが、ここでは、

ArrayAdapter(Context context, int textViewResourceId, T[] objects)

という形のものを「super(context,0,objects)」で呼び出しています。「T[]」は、ジェネリックを使うときの記法で、「T型の配列」を表します。前述のようにArrayAdapter<String>としてあるため、TはString型となり、objectsは、Stringの配列になります。ListViewは、このAdapterオブジェクトを介して、表示をします。データの管理などは、Adapter内で行ない、配列に格納されたどの要素を表示するなども、すべてAdapter内で管理します。画面上に表示するためのViewは、getViewメソッドを呼び出して生成させます。ただし、生成されるのは、表示できる数だけで、スクロールしたときなどは、すでに生成されたviewを使い回すようになっています。

 getViewを具体的に見ていきましょう。このメソッドは、

public View getView(int position, View convertView, ViewGroup parent)

という形で呼び出されます。positionは表示するデータの配列内のインデックスです。convertViewは表示用のViewオブジェクト、parentはその親オブジェクトです。

 このメソッドが呼び出されたとき、convertViewが未定義(null)ならば、まだ表示用のビューが作られていないので、これを生成してやります。このプログラムでは、レイアウトファイル(res/layout/timezonelist.xml)でビューを定義してあり、インフレーター(第6回参照)を使ってViewオブジェクトにします。ビューは、必要な数しか作られないので、これは、最初にListViewが表示されるときのみnullとなり、スクロールをさせている間は、ビューは使い回しされます。なので、ビューに値を設定するときには、きちんと設定しないと、前の値が残ってしまいます。

 timezonelist.xmlは(リスト2)のようにTextViewを4つ持っています。TimeZoneのID(inListTV)、表示名(inListLongName)、時差(inListOffset)、サマータイム(inListDST)です。getViewでは、この4つのTextViewにそれぞれ値を設定しています。

リスト2

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout android:id="@+id/LinearLayout01"
  android:layout_width="fill_parent"
  xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_height="wrap_content"
  android:orientation="vertical">
  <TextView
    xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_height="wrap_content"
    android:layout_width="fill_parent" android:id="@+id/inListTV"
    android:text="TimeZoneID" android:textSize="20dip">
    </TextView>
  <LinearLayout android:id="@+id/LinearLayout02"
    android:layout_height="wrap_content"
    android:layout_width="fill_parent"
    android:orientation="horizontal">
    <TextView android:layout_height="wrap_content"
      android:text="TimeZoneName"
      android:id="@+id/inListLongName"
      android:layout_width="fill_parent"
      android:layout_weight="6"
      android:textSize="14dip">
      </TextView>
    <TextView android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:paddingLeft="3px"
      android:paddingRight="8px"
      android:id="@+id/inListOffset"
      android:text="GMT+10:00"
      android:textSize="14dip">
      </TextView>
    <TextView android:layout_width="wrap_content"
      android:layout_height="wrap_content"
      android:id="@+id/inListDST"
      android:layout_gravity="right"
      android:gravity="right"
      android:paddingRight="5px"
      android:text="@string/indicateDST">
      </TextView>
  </LinearLayout>
</LinearLayout>

 TimeZone IDは、「エリア/都市名」という形の文字列で、javaのTimeZoneオブジェクトは、このIDを使ってタイムゾーンを設定します。また表示用には、LongName/ShortNameの2つがありますが、人間にわかりやすい表示としてLongNameを使います。時差やサマータイムの情報は、TimeZone IDを設定したTimeZoneオブジェクトで調べることが可能です。ポイントは、文字列としてのTimeZone IDがあれば、TimeZoneオブジェクトを生成して、情報を得ることが可能だという点です。

 ArrayAdapterを定義したときに設定した配列は、ArrayAdapter内では、getItem(int pos)というメソッドでアクセスが可能です。もちろん、直接配列にアクセスすることもできますが、getItemを使うことで、他の配列を渡してクラスを生成させたときにもコードを変更する必要がありません。ここでは、配列にTimeZone IDを入れてあります。というのは、TimeZone IDからLongNameを得ることはできるのですが、人間に判りやすいLongNameからTimeZone IDを得ることはできないからです。また、TimeZoneオブジェクトは、システムが持つすべてのタイムゾーンを配列にして戻すという機能があり、これから配列を生成させています。

 ですが、人間にわかりやすいようにLongNameやオフセット値などを補助的に表示させる必要があるため、ListViewのgetView内で、これらの情報を求めて、設定しているわけです。

前へ 1 2 次へ

カテゴリートップへ

本記事はアフィリエイトプログラムによる収益を得ている場合があります

この連載の記事

ASCII倶楽部

注目ニュース

  • 角川アスキー総合研究所

プレミアム実機レビュー

ピックアップ
1
KIOXIA(キオクシア) 旧東芝メモリ microSD 128GB UHS-I Class10 (最大読出速度100MB/s) Nintendo Switch動作確認済 国内サポート正規品 メーカー保証5年 KLMEA128G
KIOXIA(キオクシア) 旧東芝メモリ microSD 128GB UHS-I Class10 (最大読出速度100MB/s) Nintendo Switch動作確認済 国内サポート正規品 メーカー保証5年 KLMEA128G
¥2,253
2
Anker PowerLine III Flow USB-C & USB-C ケーブル Anker絡まないケーブル 240W 結束バンド付き USB PD対応 シリコン素材採用 iPhone 17 / 16 / 15 / Galaxy iPad Pro MacBook Pro/Air 各種対応 (1.8m ミッドナイトブラック)
Anker PowerLine III Flow USB-C & USB-C ケーブル Anker絡まないケーブル 240W 結束バンド付き USB PD対応 シリコン素材採用 iPhone 17 / 16 / 15 / Galaxy iPad Pro MacBook Pro/Air 各種対応 (1.8m ミッドナイトブラック)
¥1,390
3
Anker USB Type C ケーブル PowerLine USB-C & USB-A 3.0 ケーブル iPhone 17 / 16 / 15 /Xperia/Galaxy/LG/iPad Pro/MacBook その他 Android 等 USB-C機器対応 テレワーク リモート 在宅勤務 0.9m ホワイト
Anker USB Type C ケーブル PowerLine USB-C & USB-A 3.0 ケーブル iPhone 17 / 16 / 15 /Xperia/Galaxy/LG/iPad Pro/MacBook その他 Android 等 USB-C機器対応 テレワーク リモート 在宅勤務 0.9m ホワイト
¥740
4
UGREEN USB Type Cケーブル PD対応 100W/5A 超急速充電 USB C ナイロン編み 断線防止 iphone17/16/15シリーズ/iPad/MacBook Pro/Galaxy S24/Matebook/iPad/Xperia等USB-C各種対応(1m, ブラック)
UGREEN USB Type Cケーブル PD対応 100W/5A 超急速充電 USB C ナイロン編み 断線防止 iphone17/16/15シリーズ/iPad/MacBook Pro/Galaxy S24/Matebook/iPad/Xperia等USB-C各種対応(1m, ブラック)
¥1,299
5
Anker iPhone充電ケーブル PowerLine II ライトニングケーブル MFi認証 超高耐久 iPhone 14 / 14 Pro Max / 14 Plus / 13 / 13 Pro / 12 / 11 / X/XS/XR / 8 Plus 各種対応 (0.9m ホワイト)
Anker iPhone充電ケーブル PowerLine II ライトニングケーブル MFi認証 超高耐久 iPhone 14 / 14 Pro Max / 14 Plus / 13 / 13 Pro / 12 / 11 / X/XS/XR / 8 Plus 各種対応 (0.9m ホワイト)
¥990
6
KIOXIA(キオクシア)【日本製】USBフラッシュメモリ 32GB USB2.0 国内サポート正規品 KLU202A032GL
KIOXIA(キオクシア)【日本製】USBフラッシュメモリ 32GB USB2.0 国内サポート正規品 KLU202A032GL
¥937
7
NIMASO ガラスフィルム iPad 第11世代(A16) 2025用/iPad 10.9インチ 第10世代 2022用 衝撃吸収 強化 ガラス 保護フィルム 指紋防止 ガイド枠付き NTB22I574
NIMASO ガラスフィルム iPad 第11世代(A16) 2025用/iPad 10.9インチ 第10世代 2022用 衝撃吸収 強化 ガラス 保護フィルム 指紋防止 ガイド枠付き NTB22I574
¥1,359
8
エルパ(ELPA) 扉付タップラン 電源タップ 延長コード 125V 3m 3個口 ホワイト WBT-N3030B(W)
エルパ(ELPA) 扉付タップラン 電源タップ 延長コード 125V 3m 3個口 ホワイト WBT-N3030B(W)
¥841
9
エレコム 電源タップ 6個口 3m 雷ガード 個別スイッチ ほこりシャッター付 耐熱 PSE技術基準適合 ブラック T-K6A-2630BK
エレコム 電源タップ 6個口 3m 雷ガード 個別スイッチ ほこりシャッター付 耐熱 PSE技術基準適合 ブラック T-K6A-2630BK
¥1,590
10
UGREEN LANケーブル CAT8 1M メッシュLANケーブル カテゴリー8 コネクタ 超光速40Gbps/2000MHz CAT8準拠 イーサネットケーブル 爪折れ防止 シールド モデム ルータ PS3 PS4 Xbox等に対応 1M
UGREEN LANケーブル CAT8 1M メッシュLANケーブル カテゴリー8 コネクタ 超光速40Gbps/2000MHz CAT8準拠 イーサネットケーブル 爪折れ防止 シールド モデム ルータ PS3 PS4 Xbox等に対応 1M
¥699

Amazonのアソシエイトとして、ASCII.jpは適格販売により収入を得ています。

デジタル用語辞典

ASCII.jpメール デジタルMac/iPodマガジン