2012年7月31日火曜日

Android SDK のダウンロードとインストール:ubuntu

ia32-libs が入っているか確認する。
dpkg -l | grep ia32

Android SDK を下記サイトからダウンロードしてくる。
http://developer.android.com/sdk/index.html

android-sdk_r20.0.1-linux.tgz を解凍する。
解凍して出来たフォルダ android-sdk-linux をホームに移動する。

~/android-sdk-linux/tools/android をダブルクリックすると
Android SDK Manager が起動する。

必要なものをインストールしておく。
Toolsの
Android SDK Tools
Android SDK Platform-tools

Android 2.2の SDK Platform をインストールした。

toolとか使いやすいようにPATHを通しておく。

.bashrcに追加

export ANDROID_SDK_HOME=[AndroidSDKを配置したパス]
PATH=$PATH:$ANDROID_SDK_HOME/tools:$ANDROID_SDK_HOME/platform-tools

変更を有効にする。
$source .bashrc

eclipse のインストール
ubuntu ソフトウェアセンターからインストールする。

eclipse.intの末尾に追記
-Djava.library.path=/usr/lib/jni

日本語化
pleiades_1.3.4 で日本語化する。
sudo cp -R features/* /usr/lib/eclipse/features/
sudo cp -R plugins/* /usr/lib/eclipse/plugins/

eclipse.intの修正
追記する
-javaagent:/usr/lib/eclipse/plugins/jp.sourceforge.mergedoc.pleiades/pleiades.jar

コメントアウトする
#-showsplash
#org.eclipse.platform

eclipse に Android Plugin をインストール
ヘルプ→新規ソフトウェアのインストールを開く
https://dl-ssl.google.com/android/eclipse/ を入力してインストール

ubuntu12.04 64bitにeclipseをインストールする。

普通にインストールしても起動しないので、設定ファイルを修正する。

設定ファイルを管理者権限で開く
sudo gedit /etc/eclipse.ini

開いたファイルの末尾に次の一行を書き込む
-Djava.library.path=/usr/lib/jni

ソースコードの取得・ビルド

ソースコードの取得
androidのソースコード取得に必要なツールを取得する。
sudo apt-get install git-core
sudo apt-get install curl

ホームにbinディレクトリを作りrepoをコマンドをインストールして、実行権限を与える。
mkdir ~/bin
curl https://dl-ssl.google.com/dl/googlesource/git-repo/repo > ~/bin/repo
PATH=~/bin:$PATH
chmod a+x ~/bin/repo

ソースコードのダウンロード用ディレクトリandroidCoreを作ってそこに移動して、repoコマンドを打つ(masterの場合)
mkdir androidMaster
cd androidMaster/
repo init -u https://android.googlesource.com/platform/manifest

ソースコード取得開始
repo sync -j8

ビルド準備
JDKのインストールオラクルのページからjdk-6u33-linux-x64.binをダウンロードしてくる。

$ chmod a+x jdk-6u33-linux-x64.bin
$ ./jdk-6u33-linux-x64.bin
$ sudo mv jdk1.6.0_33 /usr/lib/jvm/

javaの環境設定をしてくれるツールをインストールする
$ wget http://webupd8.googlecode.com/files/update-java-0.5b
$ chmod a+x update-java-0.5b
$ sudo ./update-java-0.5b

jdk1.6.0_33 を選択する。

必要なツールのインストールとシンボリックリンクの作成
$ sudo apt-get install git-core gnupg flex bison gperf build-essential \ zip curl libc6-dev libncurses5-dev:i386 x11proto-core-dev \ libx11-dev:i386 libreadline6-dev:i386 libgl1-mesa-glx:i386 \ libgl1-mesa-dev g++-multilib mingw32 openjdk-6-jdk tofrodos \ python-markdown libxml2-utils xsltproc zlib1g-dev:i386

$ sudo ln -s /usr/lib/i386-linux-gnu/mesa/libGL.so.1 /usr/lib/i386-linux-gnu/libGL.so

ビルド
source build/envsetup.sh
lunch full-eng
make -j8

備考
バージョン(ブランチ名)を指定して取得する場合
repo init -u https://android.googlesource.com/platform/manifest -b android-4.0.4_r1.2

ブランチ名を調べる方法
repo init ....を打った後
$ ls -1 .repo/manifests.git/logs/refs/remotes/origin/

開発環境
ubuntu 12.04 LTS 64bit
メモリ:8G
プロセッサ:intel Core i3
ディスク容量:135G

2012年7月30日月曜日

ソースコード・ビルドメモ

ubuntuをViertualBoxで利用 ソースコードの取得時に止まるので諦める。
ubuntu-11.04-desktop-amd64.iso を利用

必要なソフトのインストール
$ sudo apt-get install git-core gnupg flex bison gperf build-essential \
zip curl zlib1g-dev libc6-dev lib32ncurses5-dev ia32-libs \
x11proto-core-dev libx11-dev lib32readline5-dev lib32z-dev \
libgl1-mesa-dev g++-multilib mingw32 tofrodos python-markdown \
libxml2-utils xsltproc

シンボリックリンクの作成
$ sudo ln -s /usr/lib32/mesa/libGL.so.1 /usr/lib32/mesa/libGL.so

JDKのインストール
http://source.android.com/source/initializing.html
上記のページに書いてある方法ではインストール出来なくなっていたので、
オラクルのページからダウンロードしてくる。
jdk-6u33-linux-x64.bin

$ chmod a+x jdk-6u33-linux-x64.bin
$ ./jdk-6u33-linux-x64.bin
$ sudo mkdir /usr/lib/jvm
$ sudo mv jdk1.6.0_33 /usr/lib/jvm/
$ cd /usr/lib/jvm
$ sudo ln -s jdk1.6.0_33 java-6-oracle

javaの環境設定をしてくれるツールをインストールする
$ wget http://webupd8.googlecode.com/files/update-java-0.5b
$ chmod a+x update-java-0.5b
$ sudo ./update-java-0.5b

 java-6-oracle

repoのインストール
$ mkdir ~/bin
$ PATH=~/bin:$PATH
$ curl https://dl-ssl.google.com/dl/googlesource/git-repo/repo > ~/bin/repo
$ chmod a+x ~/bin/repo

Androidのソースツリーダウンロード



ubuntu 12.04 LTS 64ビット 


Windows installerを使って30Gの容量でインストールした、ソースのダウンロードは出来たがビルドは容量不足で出来なかった。
結局ハードディスクにインストールしてビルドした。

http://source.android.com/source/initializing.html を確認する。

JDKのインストール
Android 4.0の場合はJava 6をインストールする。
64ビットの場合は、http://developer.android.com/sdk/installing/index.html#troubleshooting
を確認すると,
sudo apt-get install ia32-libs
が必要なようだ。これで sudo apt-get install sun-java6-jdk
インストールで出来なかった。
今回はOAB-Javaでローカルリポジトリを追加して、
Oracle(Sun) Java 6をインストールする。

cd ~/
wget https://github.com/flexiondotorg/oab-java6/raw/0.2.4/oab-java.sh -O oab-java.sh
chmod +x oab-java.sh
sudo ./oab-java.sh
sudo apt-get install sun-java6-jdk これで、java 6 JDK がインストール出来た。


必要なソフトのインストール
$ sudo apt-get install git-core gnupg flex bison gperf build-essential \
 zip curl libc6-dev libncurses5-dev:i386 x11proto-core-dev \
 libx11-dev:i386 libreadline6-dev:i386 libgl1-mesa-glx:i386 \
 libgl1-mesa-dev g++-multilib mingw32 openjdk-6-jdk tofrodos \
 python-markdown libxml2-utils xsltproc zlib1g-dev:i386

シンボリックリンクの作成
$ sudo ln -s /usr/lib/i386-linux-gnu/mesa/libGL.so.1 /usr/lib/i386-linux-gnu/libGL.so

repoコマンドのインストール
mkdir ~/bin
PATH=~/bin:$PATH
curl https://dl-ssl.google.com/dl/googlesource/git-repo/repo > ~/bin/repo
chmod a+x ~/bin/repo

gitの設定
git config --global user.email "androckjp@gmail.com"
git config --global user.name "AndRock"

ソースの取得
mkdir WORKING_DIRECTORY
cd WORKING_DIRECTORY
repo init -u https://android.googlesource.com/platform/manifest -b android-4.0.1_r1

repo sync -j4

j4オプションはビルドするマシンのCPU×2倍位にするといいそうです。

envsetup.sh の実行と lunch の設定
source build/envsetup.sh
lunch generic-eng

make の実効
make -j4


2012年7月25日水曜日

ListVewのタッチイベント:ListActivity,onListItemClick

ListActivityにはonListItemClickメソッドが用意されている。
import java.util.ArrayList;

import android.app.ListActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.ArrayAdapter;
import android.widget.ListView;

public class ListVewTouchEventActivity extends ListActivity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        // Listの作成
        ArrayList<String> list = new ArrayList<String>();
        // Listにデータを入れる
        list.add("ゼウス");
        list.add("アテナ");
        list.add("アポロン");
        list.add("ポセイドン");
        
  //ListView に表示する項目設定
        ArrayAdapter<String> adapter = new ArrayAdapter<String>(this,
    android.R.layout.simple_list_item_1, list);
  setListAdapter(adapter);
        
    }
    
    protected void onListItemClick(ListView listView, View v, int position,long id) {
  super.onListItemClick(listView, v, position, id);
  
  String str = (String) listView.getItemAtPosition(position);
  Log.v("TAG", position + " 行目の " + str + " がクリックされた");
 }

}

07-25 13:05:35.737: V/TAG(4478): 0 行目の ゼウス がクリックされた
07-25 13:05:36.507: V/TAG(4478): 1 行目の アテナ がクリックされた
07-25 13:05:37.307: V/TAG(4478): 2 行目の アポロン がクリックされた
07-25 13:05:42.077: V/TAG(4478): 3 行目の ポセイドン がクリックされた

2012年7月24日火曜日

Threadの利用方法:AsyncTask、ProgressDialog



import android.app.Activity;
import android.app.ProgressDialog;
import android.content.Context;
import android.content.DialogInterface;
import android.content.DialogInterface.OnCancelListener;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.View;

public class AndroidAsyncTaskActivity extends Activity {
 /** Called when the activity is first created. */
 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.main);
 }

 public void buttonClick(View v) {
  Log.d("タグ", "ボタンがクリックした。!");

  new MyAsyncTask(this).execute("Param1");
 }

 public class MyAsyncTask extends AsyncTask<String, Integer, Long> implements OnCancelListener {

  private Context context;
  private ProgressDialog dialog;

  public MyAsyncTask(Context context) {
   // TODO 自動生成されたコンストラクター・スタブ
   this.context = context;
  }

  @Override
  protected void onPreExecute() {
 // タスク開始前処理:UIスレッドで実行される
   //進捗ダイアログの表示
   Log.d("TAG", "onPreExecute");
   dialog = new ProgressDialog(context);
   dialog.setTitle("しばらくお待ちください");
   dialog.setMessage("データ読み込み中...");
   dialog.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
   dialog.setCancelable(true);// ProgressDialog のキャンセルが可能かどうか  
   dialog.setOnCancelListener(this);
   dialog.setMax(100);
   dialog.setProgress(0);
   
    // ProgressDialog の Cancel ボタン  
   dialog.setButton(DialogInterface.BUTTON_NEGATIVE, "キャンセル", new DialogInterface.OnClickListener() {
    public void onClick(DialogInterface dialog, int which) {
     // ProgressDialog をキャンセル
     dialog.cancel();
    }
   });  
   dialog.show();
  }

  @Override
  protected Long doInBackground(String... params) {
// 別スレッドで実行する処理
   Log.d("", "doInBackground ::" + params[0]);//doInBackground ::Param1

   try {
    for (int i = 0; i < 10; i++) {
     if (isCancelled()) {
      Log.d("", "Cancelled!");
      break;
     }
     Thread.sleep(1000);
     publishProgress((i + 1) * 10);
    }
   } catch (InterruptedException e) {
    Log.d("", "InterruptedException in doInBackground");
   }
   return 999L;
  }
  
  // プログレスバー更新処理: UIスレッドで実行される
  @Override
  protected void onProgressUpdate(Integer... values) {
   
   Log.d("", "onProgressUpdate ::" + values[0]);
   dialog.setProgress(values[0]);
  }

  @Override
  protected void onCancelled() {
   Log.d("", "onCancelled");
   dialog.dismiss();
  }

  @Override
  protected void onPostExecute(Long result) {
// タスク終了後処理:UIスレッドで実行される
   Log.d("", "onPostExecute :: " + result);//doInBackground のリターン
   // 進捗ダイアログをクローズ
   dialog.dismiss();
  }

  public void onCancel(DialogInterface dialog) {
   Log.d("", "Dialog onCancell... calling cancel(true)");
   this.cancel(true);
  }

 }

}

07-23 20:52:59.493: D/(12475): onProgressUpdate ::50
07-23 20:53:00.509: D/(12475): onProgressUpdate ::60
07-23 20:53:01.163: D/(12475): Dialog onCancell... calling cancel(true)
07-23 20:53:01.163: D/(12475): InterruptedException in doInBackground
07-23 20:53:01.173: D/(12475): onCancelled

ListView追加読み込み:003 SimpleAdapter



import android.app.Activity;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.AbsListView;
import android.widget.AbsListView.OnScrollListener;
import android.widget.ArrayAdapter;
import android.widget.ListView;

public class AndroidAsyncTaskActivity extends Activity implements OnScrollListener {

 private ArrayAdapter<String> adapter;
 private AsyncTask<String, Integer, Long> mTask;
 private ListView listView;
 private View mFooter;
 

 /** Called when the activity is first created. */
 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.main);

  listView = (ListView) findViewById(R.id.listView1);

  adapter = new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1);
  // アイテムを追加します
  for (int i = 1; i < 10; i++) {
   adapter.add("Item" + i);
  }

  mFooter = getLayoutInflater().inflate(R.layout.listview_footer, null);
  
  listView.addFooterView(mFooter);
  
  // アダプターを設定します
  listView.setAdapter(adapter);

  listView.setOnScrollListener(this);
 }



 public void onScroll(AbsListView view, int firstVisibleItem, int visibleItemCount, int totalItemCount) {
  // TODO 自動生成されたメソッド・スタブ
  if (totalItemCount == firstVisibleItem + visibleItemCount) {
   Log.d("TAG", "末尾");//

   additionalReading();
  }

 }

 private void additionalReading() {
  // TODO 自動生成されたメソッド・スタブ
  // 既に読み込み中ならスキップ
  if (mTask != null && mTask.getStatus() == AsyncTask.Status.RUNNING) {
   Log.d("", "読み込み中");
   return;
  }
  mTask = new MyAsyncTask(this).execute("text");

 }
 public class MyAsyncTask extends AsyncTask<String, Integer, Long> {

  public MyAsyncTask(AndroidAsyncTaskActivity androidAsyncTaskActivity) {
  }
  
  protected Long doInBackground(String... params) {
   Log.d("", "doInBackground ::" + params[0]);// doInBackground

   try {
    Thread.sleep(2000);//確認のために2秒止める
   } catch (InterruptedException e) {
    // TODO 自動生成された catch ブロック
    e.printStackTrace();
   }
   
   return 999L;
  }

  @Override
  protected void onPostExecute(Long result) {
   Log.d("", "onPostExecute :: " + result);// doInBackground のリターン
   //データ追加
   for (int n = 0; n < 10; n++) {
    adapter.add("追加" + n);
   }
  }

 }

 public void onScrollStateChanged(AbsListView view, int scrollState) {
  // TODO 自動生成されたメソッド・スタブ

 }
}

ListView:SimpleAdapter View再利用 getView


使い方があってるか良く分からない。

import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map;

import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;
import android.widget.SimpleAdapter;

public class AndroidListviewTest01Activity extends Activity {

 private ArrayList<Map<String, String>> retDataList;
 private SimpleAdapter adapter;
 private HashMap<String, String> data;

 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.main);

  new ArrayList<Map<String, Object>>();
  retDataList = new ArrayList<Map<String, String>>();

  for (int n = 0; n < 11; n++) {
   data = new HashMap<String, String>();
   data.put("title", n + "行目");
   data.put("comment", "テキスト" + n);
   retDataList.add(data);
  }

  // adapter = new SimpleAdapter(this, retDataList,R.layout.raw, new
  // String[] { "title", "comment" }, new int[] {android.R.id.text1,
  // android.R.id.text2 });
  adapter = new MySimpleAdapter(this, retDataList, R.layout.raw, new String[] { "title", "comment" }, new int[] { android.R.id.text1, android.R.id.text2 });

  // MyAdapter adapter = new MyAdapter(this, list);

  ListView listView = (ListView) findViewById(R.id.listView1);
  listView.setAdapter(adapter);//
 }

}

import java.util.HashMap;
import java.util.List;
import java.util.Map;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.SimpleAdapter;
import android.widget.TextView;

public class MySimpleAdapter extends SimpleAdapter {

 // private Context context;
 private LayoutInflater inflater;
 private List<? extends Map<String, ?>> listData;

 public class ViewHolder {
  TextView line1;
  TextView line2;
 }

 public MySimpleAdapter(Context context, List<? extends Map<String, ?>> data, int resource, String[] from, int[] to) {
  super(context, data, resource, from, to);
  // TODO 自動生成されたコンストラクター・スタブ
  // this.context = context;
  this.inflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
  this.listData = data;
 }

 public View getView(int position, View convertView, ViewGroup parent) {
  ViewHolder holder;

  // ビューを受け取る
  View view = convertView;

  if (view == null) {
   view = inflater.inflate(R.layout.raw, parent, false);
   // LayoutInflater inflater = (LayoutInflater)
   // context.getSystemService(Context.LAYOUT_INFLATER_SERVICE);
   // view = inflater.inflate(R.layout.raw, null);

   holder = new ViewHolder();
   holder.line1 = (TextView) view.findViewById(android.R.id.text1);
   holder.line2 = (TextView) view.findViewById(android.R.id.text2);

   view.setTag(holder);
  } else {
   holder = (ViewHolder) view.getTag();
  }

  String text1 = ((HashMap<?, ?>) listData.get(position)).get("title").toString();
  String text2 = ((HashMap<?, ?>) listData.get(position)).get("comment").toString();
  holder.line1.setText(text1);
  holder.line2.setText(text2);

  return view;
 }
}

2012年7月23日月曜日

ListView追加読み込み:未002 SimpleAdapter

末尾に追加

import java.util.ArrayList;
import java.util.HashMap;
import java.util.Map;

import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.widget.AbsListView;
import android.widget.ListView;
import android.widget.AbsListView.OnScrollListener;
import android.widget.SimpleAdapter;

public class AndroidListviewTest01Activity extends Activity implements OnScrollListener {
    
    Map<String, Object> map0;
    Map<String, Object> map1;
 private ArrayList<Map<String, String>> retDataList;
 private SimpleAdapter adapter2;
     
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
         
         new ArrayList<Map<String, Object>>();
        retDataList = new ArrayList<Map<String, String>>();
          
  for (int n = 0; n < 11; n++) {
   Map<String, String> data = new HashMap<String, String>();
   data.put("title", n + "行目");
   data.put("comment", "111");
   retDataList.add(data);
  }

        adapter2 = new SimpleAdapter(this, retDataList,
                R.layout.raw, new String[] { "title", "comment" }, 
                new int[] {android.R.id.text1, android.R.id.text2 });
         
        ListView listView = (ListView) findViewById(R.id.listView1);
        listView.setAdapter(adapter2);//
        listView.setOnScrollListener(this);
    }
     
 public void onScroll( AbsListView view, int firstVisibleItem,int visibleItemCount, int totalItemCount ) {
  // TODO 自動生成されたメソッド・スタブ
  Log.d("TAG", "スクロールが発生した。2");
    Log.d("TAG", "firstVisibleItem::" + firstVisibleItem );
    Log.d("TAG", "visibleItemCount::" + visibleItemCount );
    Log.d("TAG", "totalItemCount::" + totalItemCount );
     
  if (totalItemCount == firstVisibleItem + visibleItemCount) {
   Log.d("TAG", "end");//
   // mData.add( );
    //List<Map<String, String>> retDataList = new ArrayList<Map<String, String>>();
    Map<String, String> data = new HashMap<String, String>();
    
    data.put("title", "追加");
          data.put("comment", "111");
          retDataList.add(data);
          
          adapter2.notifyDataSetChanged();
  }
 }

 public void onScrollStateChanged(AbsListView arg0, int arg1) {
  // TODO 自動生成されたメソッド・スタブ
  
 }
}


2012年7月22日日曜日

ListView追加読み込み:未001ArrayAdapter

import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.widget.AbsListView;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.AbsListView.OnScrollListener;

public class AndroidListviewTest01Activity extends Activity implements OnScrollListener{

 /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        ListView listView = (ListView) findViewById(R.id.listView1);

        //ListView に表示する項目設定
        ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1);
        // アイテムを追加します
  for (int n = 0; n < 10; n++) {
   adapter.add("Item" + n );
  }

        // アダプタを設定します。
        listView.setAdapter(adapter);
        
        listView.setOnScrollListener(this);
    }

 public void onScroll(AbsListView view, int firstVisibleItem,int visibleItemCount, int totalItemCount) {
  // TODO 自動生成されたメソッド・スタブ
  Log.d("TAG", "スクロールが発生した。2");
  Log.d("TAG", "firstVisibleItem::" + firstVisibleItem );
  Log.d("TAG", "visibleItemCount::" + visibleItemCount );
  Log.d("TAG", "totalItemCount::" + totalItemCount );
  
  if (totalItemCount == firstVisibleItem + visibleItemCount) {
   Log.d("TAG", "end");//
   addListView();
  }
 }

 private void addListView() {
  // TODO 自動生成されたメソッド・スタブ
  
  Log.d("TAG", "end2");//
  ListView listView = (ListView) findViewById(R.id.listView1);
  @SuppressWarnings("unchecked")
  ArrayAdapter<String> tempAdapter = (ArrayAdapter<String>)listView.getAdapter();
  tempAdapter.add("001");
 }

 public void onScrollStateChanged(AbsListView arg0, int arg1) {
  // TODO 自動生成されたメソッド・スタブ
  
 }
}

ListViewで複雑なタッチイベントを取得する:GestureDetector

ListViewの要素変更:ArrayAdapter viewの再利用の記事のソースを複雑なタップイベントが取得出来るようにする。

とりあえずダブルタップを取得してみる。

import java.util.ArrayList;
import java.util.List;

import android.app.ListActivity;
import android.os.Bundle;
import android.util.Log;
import android.view.GestureDetector;
import android.view.View;
import android.view.GestureDetector.SimpleOnGestureListener;
import android.view.MotionEvent;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;

public class AndroidListViewTestActivity extends ListActivity implements OnItemClickListener
{
 private GestureDetector gestureDetector;
 private View.OnTouchListener gestureListener;
 
 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.main);

  gestureDetector = new GestureDetector(new MyGestureDetector());
  gestureListener = new View.OnTouchListener() {
   public boolean onTouch(View v, MotionEvent event) {
    return gestureDetector.onTouchEvent(event);
   }
  };

  getListView().setOnItemClickListener(this);
  getListView().setOnTouchListener(gestureListener);

  List<BindData> list = new ArrayList<BindData>();
  // ここにlistに項目を追加する処理が入る
  list.add(new BindData("タイトル0", null, null));
  list.add(new BindData(null, "line1-1", "line1-2"));

  for (int i = 1; i < 10; i++) {
   list.add(new BindData("タイトル" + i, null, null));
   list.add(new BindData(null, "line2-1", "line2-2"));
   list.add(new BindData(null, "line2-12", "line2-22"));
  }

  MyAdapter adapter = new MyAdapter(this, list);
  setListAdapter(adapter);

 }
 
 public void onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3) {
  // TODO 自動生成されたメソッド・スタブ
  Log.i("TAG", "::" + "リストがタッチされた2" );
 }

 class MyGestureDetector extends SimpleOnGestureListener {

  @Override
  public boolean onDoubleTap(MotionEvent event) {
   Log.d("TAG", "ダブルタップが発生した。");

   return super.onDoubleTap(event);
  }
 }
}

07-22 05:06:34.806: D/TAG(13681): ダブルタップが発生した。
07-22 05:06:34.937: I/TAG(13681): ::リストがタッチされた2

2012年7月21日土曜日

Activityのライフサイクルを確認

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import android.view.View;

public class ActivityTest extends Activity {
    //アクティビティが最初に起動するときに呼び出される。
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        Log.i("タグ", "onCreate()");
    }

    //アクティビティが表示される直前に呼ばれます。
    public void onStart() {
     super.onStart();
      Log.i("タグ", "onStart()");
 }

 //アクティビティがユーザとのやり取りを始められるようになる直前に呼ばれます。
    public void onResume() {
     super.onResume();
     Log.i("タグ", "onResume()");
 }

    //アクティビティがバックグラウンドに隠れるときに呼ばれます。
    public void onPause() {
     super.onPause();
     Log.i("タグ", "onPause()");
 }

    //アクティビティがユーザーから見えなくなったとき
    public void onStop() {
     super.onStop();
     Log.i("タグ", "onStop()");
 }

    //終了状態になったアクティビティが再び実行される直前呼ばれる。
    public void onRestart() {
     super.onRestart();
     Log.i("タグ", "onRestart()");
 }

    //アクティビティが破棄される直前に呼ばれます。
    public void onDestroy() {
     super.onDestroy();
     Log.i("タグ", "onDestroy()");
    }

    //Activityのインスタンスを保存 Bundleにデータ保存する
    public void onSaveInstanceState(Bundle savedInstanceState){
     super.onSaveInstanceState(savedInstanceState);
     Log.i("タグ", "onSaveInstanceState()");
    }
    //Activityのインスタンスを復帰
    public void onRestoreInstanceState(Bundle savedInstanceState){
     super.onRestoreInstanceState(savedInstanceState);
     Log.i("タグ", "onRestoreInstanceState()");
    }


 // ボタンイベント
 public void SetButtonOnClick(View v) {

  switch (v.getId()) {
  case R.id.button1:
   Log.i("タグ", "ボタン1が押された");
   break;
  case R.id.button2:
   Log.i("タグ", "ボタン2が押された");
   Intent subactivity = new Intent(this,com.yamato.test.SecondScreenActivity.class);
      startActivity(subactivity);
   break;
  case R.id.button3:
   Log.i("タグ", "ボタン3が押された");
   finish();
   break;
  }

 }
}

最初に起動させた時
07-21 05:57:31.581: I/タグ(3992): onCreate()
07-21 05:57:31.581: I/タグ(3992): onStart()
07-21 05:57:31.592: I/タグ(3992): onResume()

ホームボタンを押した時
07-21 06:00:18.332: I/タグ(3992): onSaveInstanceState()
07-21 06:00:18.332: I/タグ(3992): onPause()
07-21 06:00:19.042: I/タグ(3992): onStop()

ホームボタンを押した後アプリケーション画面から起動させた時
07-21 06:03:46.612: I/タグ(3992): onRestart()
07-21 06:03:46.612: I/タグ(3992): onStart()
07-21 06:03:46.612: I/タグ(3992): onResume()

finish()が呼びさされた時
07-21 06:06:09.833: I/タグ(3992): onPause()
07-21 06:06:10.451: I/タグ(3992): onStop()
07-21 06:06:10.451: I/タグ(3992): onDestroy()

finish()で終了した後、アプリケーション画面から起動させた時
07-21 06:08:34.592: I/タグ(3992): onCreate()
07-21 06:08:34.592: I/タグ(3992): onStart()
07-21 06:08:34.592: I/タグ(3992): onResume()

別アクティビティーが呼ばれた時
07-21 06:32:38.362: I/タグ(4904): onSaveInstanceState()
07-21 06:32:38.362: I/タグ(4904): onPause()
07-21 06:32:38.842: I/タグ(4904): onStop()

別アクティビティーが呼ばれた後戻るボタンを押した時
07-21 06:34:51.502: I/タグ(4904): onRestart()
07-21 06:34:51.502: I/タグ(4904): onStart()
07-21 06:34:51.502: I/タグ(4904): onResume()

複数ボタンの処理:view button



import android.app.Activity;
import android.os.Bundle;
import android.util.Log;
import android.view.View;

public class ButtonActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
    }

 // ボタンイベント
 public void SetButtonOnClick(View v) {

  switch (v.getId()) {
  case R.id.button1:
   Log.i("buttonTest", "ボタン1が押された");
   break;
  case R.id.button2:
   Log.i("buttonTest", "ボタン2が押された");
   break;
  case R.id.button3:
   Log.i("buttonTest", "ボタン3が押された");
   break;

  }

 }
}


<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <Button
        android:id="@+id/button1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="SetButtonOnClick"
        android:text="Button1" />

    <Button
        android:id="@+id/button2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="SetButtonOnClick"
        android:text="Button2" />

    <Button
        android:id="@+id/button3"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:onClick="SetButtonOnClick"
        android:text="Button3" />

</LinearLayout>

2012年7月15日日曜日

assetsフォルダのフォントファイルの利用:Typeface


フォントサイズが大きすぎてassets フォルダで未圧縮の状態では使えない場合。
assets フォルダに圧縮したフォントファイルをおいて、アプリケーション領域にコピーして使う。

アプリの容量は、4GB にまで拡大されたが、APK ファイル自体のサイズは50MBまで、
拡張ファイル1つあたりの制限は2GB までなので2GBを超えるフォントサイズでは使えないので、WEBからダウンロードする方法を採らないといけないかも。

assetsフォルダの圧縮ファイルを解凍してアプリケーション領域のfilesフォルダにコピーする。
一つのファイルが圧縮されている場合。
try {
   AssetManager am = getResources().getAssets();
   InputStream is = am.open("ipaexm.zip", AssetManager.ACCESS_STREAMING);
   ZipInputStream zis = new ZipInputStream(is);
   ZipEntry  ze = zis.getNextEntry();

   if (ze != null) {
    path = getFilesDir().toString() + "/" + ze.getName();
    FileOutputStream fos = new FileOutputStream(path, false);
    byte[] buf = new byte[1024];
    int size = 0;

    while ((size = zis.read(buf, 0, buf.length)) > -1) {
     fos.write(buf, 0, size);
    }
    fos.close();
    zis.closeEntry();
   }
   zis.close();
  } catch (Exception e) {
   e.printStackTrace();
  }

圧縮ファイルに複数のファイルが含まれているとき
try {
   AssetManager am = getResources().getAssets();
   InputStream is = am.open("ipaexm.zip", AssetManager.ACCESS_STREAMING);
   ZipInputStream zis = new ZipInputStream(is);
   ZipEntry  ze = zis.getNextEntry();

   while (ze != null) { 
    path = getFilesDir().toString() + "/" + ze.getName();
    FileOutputStream fos = new FileOutputStream(path, false);
    byte[] buf = new byte[1024];
    int size = 0;

    while ((size = zis.read(buf, 0, buf.length)) > -1) {
     fos.write(buf, 0, size);
    }
    fos.close();
    zis.closeEntry();
    ze = zis.getNextEntry();
   }
   zis.close();
  } catch (Exception e) {
   e.printStackTrace();
  }

フォントファイルがアプリケーションデータ領域のfilesフォルダに保存されている場合
TextView tv2 = (TextView)findViewById(R.id.textView2);
TextView tv3 = (TextView)findViewById(R.id.textView3);

Typeface typeface = Typeface.createFromFile("/data/data/com.ymato.font/files/ipaexg.ttf");
Typeface typeface2 = Typeface.createFromFile("/data/data/com.ymato.font/files/ipaexm.ttf");

tv2.setTypeface(typeface);//
tv3.setTypeface(typeface2);//

フォントファイルがSDカードに入っている場合
Typeface typeface = Typeface.createFromFile( new File(Environment.getExternalStorageDirectory(), "ipaexm.ttf"));

assetsフォルダには1MB以上の非圧縮ファイルを設置できないので、
大きいサイズのフォントファイルを置いたらエラーが出た
Data exceeds UNCOMPRESS_DATA_MAX (6022748 vs 1048576)

FileOutputStream fos = new FileOutputStream(path, false); ではパーミッションが
-rw------- になるので変更する場合
FileOutputStream fos = openFileOutput( ze.getName() , MODE_WORLD_READABLE);
ファイルの指定方法が違う、上は/data/data/[app name]/files/fileName,
下は、ファイル名のみ、
MODE_PRIVATE このアプリからのみ使用
MODE_APPEND 現在のファイルに追記
MODE_WORLD_READABLE 他のアプリからも読み込み可能
MODE_WORLD_WRITEABLE 他のアプリからも書き込み可能

2012年7月9日月曜日

ListViewの要素変更:ArrayAdapter viewの再利用


AndroidListViewTestActivity.java
import java.util.ArrayList;
import java.util.List;

import android.app.ListActivity;
import android.os.Bundle;

public class AndroidListViewTestActivity extends ListActivity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        List<BindData> list = new ArrayList<BindData>();
        // ここにlistに項目を追加する処理が入る
        list.add(new BindData("タイトル1", null , null));
        list.add(new BindData(null, "line1-1" ,"line1-2"));
  

        list.add(new BindData("タイトル2", null , null));
        list.add(new BindData(null, "line2-1" ,"line2-2"));
        list.add(new BindData(null, "line2-12" ,"line2-22"));

        MyAdapter adapter = new MyAdapter(this, list);
        setListAdapter(adapter);
    }
}

MyAdapter.java
import java.util.List;

import android.content.Context;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.TextView;

class BindData {
    String title;
    String line1;
    String line2;

    public BindData(String string0, String string1, String string2) {
        this.title = string0;
        this.line1 = string1;
        this.line2 = string2;
    }
}

class ViewHolder {
    TextView title;
    TextView line1;
    TextView line2;
}

public class MyAdapter extends ArrayAdapter<BindData> {

    private LayoutInflater inflater;

    public MyAdapter(Context context, List<BindData> objects) {
        super(context, 0, objects);
        this.inflater = (LayoutInflater) context
            .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
    }

    @Override
    public boolean isEnabled(int position) {
        // 選択不可にする
        return true;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        ViewHolder holder;

        if (convertView == null) {
            convertView = inflater.inflate(R.layout.raw, parent, false);
            holder = new ViewHolder();
            holder.title = (TextView) convertView.findViewById(R.id.title);
            holder.line1 = (TextView) convertView.findViewById(R.id.line1);
            holder.line2 = (TextView) convertView.findViewById(R.id.line2);
            convertView.setTag(holder);
        } else {
            holder = (ViewHolder) convertView.getTag();
        }

        BindData data = getItem(position);

        //Log.d("タグ", ":" + position);
        //Log.d("タグ", "::" + data.title);
        //Log.d("タグ", ":::" + isEnabled(position));

        // タイトル
        if ( getItem(position).title != null ) {
            holder.title.setVisibility(View.VISIBLE);
            holder.title.setText(data.title);
            holder.line1.setVisibility(View.GONE);
            holder.line2.setVisibility(View.GONE);
        // line1,2
        } else {
            holder.title.setVisibility(View.GONE);
            holder.line1.setVisibility(View.VISIBLE);
            holder.line1.setText(data.line1);
            holder.line2.setVisibility(View.VISIBLE);
            holder.line2.setText(data.line2);
        }
        return convertView;
    }
}

main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <ListView
        android:id="@id/android:list"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</LinearLayout>

raw.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <!-- グループタイトル -->

    <TextView
        android:id="@+id/title"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content" android:background="#808080" android:textColor="#FFFFFF" android:text="title" android:gravity="center"/>
    <!-- アルバム名 -->

    <TextView
        android:id="@+id/line1"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:text="line1"/>
    <!-- アーティスト名 -->

    <TextView
        android:id="@+id/line2"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content" android:text="line2"/>

</LinearLayout>

[ソースコード公開中]
Google Code

2012年7月6日金曜日

ListViewの要素変更:SimpleAdapter



import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import android.app.ListActivity;
import android.content.Context;
import android.graphics.Color;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.SimpleAdapter;
import android.widget.TwoLineListItem;

public class AndroidListView004Activity extends  ListActivity {

 public class MySimpleAdapter extends SimpleAdapter {
      public MySimpleAdapter(Context context, List<Map<String, String>> items, int resource, String[] from, int[] to) {
          super(context, items, resource, from, to);
      }
  
      @Override
      public View getView(int position, View convertView, ViewGroup parent) {
        View view = super.getView(position, convertView, parent);
         
        TwoLineListItem v = (TwoLineListItem)view;
        v.getText1().setTextSize(30);
         
        if (position % 2 == 0) {
            view.setBackgroundColor(Color.BLACK);
        } else {
            view.setBackgroundColor(Color.DKGRAY);
            
        }
         
        return view;
      }
  }

 /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        List<Map<String, String>> dataList = new ArrayList<Map<String,String>>();

        Map<String, String> data = new HashMap<String, String>();
        data.put("title", "タイトル欄");
        data.put("comment", "COMMENT欄");
        dataList.add(data);

        data = new HashMap<String, String>();
        data.put("title", "タイトル欄2");
        data.put("comment", "COMMENT欄");
        dataList.add(data);

        MySimpleAdapter adapter = new MySimpleAdapter(
                this,
                dataList,
                android.R.layout.simple_list_item_2,
                new String[] { "title", "comment" },
                new int[] { android.R.id.text1, android.R.id.text2 }
            );
        setListAdapter(adapter);
    }
}

main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <ListView
        android:id="@id/android:list"
        android:layout_width="match_parent"
        android:layout_height="match_parent" />

</LinearLayout>

ListViewフォントサイズの変更



import android.app.Activity;
import android.os.Bundle;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ArrayAdapter;
import android.widget.ListView;
import android.widget.TextView;

public class AndroidListViewActivity extends Activity {
 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.main);
  
   String[] data = {"太陽", "水星", "金星"};
  
  ArrayAdapter<String> adapter = new ArrayAdapter<String>(this
             , android.R.layout.simple_list_item_1, data ) {
    
   @Override
            public View getView(int position, View convertView, ViewGroup parent) {
                TextView view = (TextView)super.getView(position, convertView, parent);
                view.setTextSize( 30 );
                return view;
            }
  };
  
  ListView listView = (ListView) findViewById(R.id.listView1);
  
  // アダプターを設定します
  listView.setAdapter(adapter);
 }
}

3行表示:ListView

一行に3つのTextView を表示出来るかやってみた。


res/raw フォルダに raw.xml ファイルを作成する。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" >

    <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="TextView" />

    <TextView
        android:id="@+id/textView2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="TextView" />

    <TextView
        android:id="@+id/textView3"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="TextView" />

</LinearLayout>

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;
import android.widget.SimpleAdapter;

public class AndroidListViewActivity extends Activity {
 /** Called when the activity is first created. */
 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.main);
  // ListView を取得
  ListView listView = (ListView) findViewById(R.id.listView1);

  List<Map<String, String>> retDataList = new ArrayList<Map<String, String>>();

  //for (int n = 0; n < 3; n++) {
   Map<String, String> data = new HashMap<String, String>();
   data.put("title", "一行目");
   data.put("comment", "111");
   data.put("note", "aaa");
   retDataList.add(data);

   data = new HashMap<String, String>();
   data.put("title", "二行目");
   data.put("comment", "222");
   data.put("note", "bbb");
   retDataList.add(data);

  // リストビューに渡すアダプタを生成します。
  SimpleAdapter adapter2 = new SimpleAdapter(this, retDataList,
    R.layout.raw, new String[] { "title", "comment" ,"note"},
    new int[] {R.id.textView1, R.id.textView2 , R.id.textView3});

  // アダプタを設定します。
  listView.setAdapter(adapter2);
 }
}


android.R.layout.simple_list_item_2を変更:ListView


res/raw フォルダに raw.xml ファイルを作成する。

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical" xmlns:android="http://schemas.android.com/apk/res/android">

    <TextView
        android:id="@android:id/text1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="TextView"
        android:textSize="18dp" >
    </TextView>

    <TextView
        android:id="@android:id/text2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="TextView" >
    </TextView>

</LinearLayout>
main.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:orientation="vertical" >

    <ListView
        android:id="@+id/listView1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" >
    </ListView>

</LinearLayout>

アダプターの生成部分

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import android.app.Activity;
import android.os.Bundle;
import android.widget.ListView;
import android.widget.SimpleAdapter;

public class AndroidListView003Activity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        ListView listView = (ListView) findViewById(R.id.listView1);
        
        List<Map<String, String>> retDataList = new ArrayList<Map<String, String>>();
       
        //for (int n = 0; n < 3; n++) {
         Map<String, String> data = new HashMap<String, String>();
         data.put("title", "一行目");
         data.put("comment", "111");
         retDataList.add(data);
       
         data = new HashMap<String, String>();
         data.put("title", "二行目");
         data.put("comment", "222");
         retDataList.add(data);
       
        // リストビューに渡すアダプタを生成します。
         SimpleAdapter adapter2 = new SimpleAdapter(this, retDataList,
            R.layout.raw, new String[] { "title", "comment" }, 
            new int[] {android.R.id.text1, android.R.id.text2 });
       
        // アダプタを設定します。
        listView.setAdapter(adapter2);
    }
}

2012年7月5日木曜日

処理時間の測定

long start = System.currentTimeMillis();
        //測定したい処理
        long end = System.currentTimeMillis();
        Log.d("タグ", "measure: " + (end - start) +" ms");