2012年9月27日木曜日

assetsフォルダのデータベース利用2

圧縮されているデータベースを解凍してコピーする。

import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
import android.app.Activity;
import android.content.Context;
import android.content.res.AssetManager;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteDatabase.CursorFactory;
import android.database.sqlite.SQLiteException;
import android.database.sqlite.SQLiteOpenHelper;
import android.os.Bundle;
import android.util.Log;

public class KenpoActivity extends Activity {

 private String DB_PATH;
 private String DB_NAME;

 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  Log.i("kenpoActivity", "onCreate()");

  setContentView(R.layout.minpo);

  DB_PATH = "/data/data/" + this.getPackageName() + "/databases/";//
  DB_NAME = "ropou.db";


  DatabaseHelper dbHelper = new DatabaseHelper(this, DB_NAME, null, 1);

  //データベースが在るか確認する
  boolean DatabaseCheck = databaseExists(DB_PATH,DB_NAME);

  if(DatabaseCheck){
   Log.i("kenpoActivity", "データベース在り" );
  }else{
   //データベースが無い場合assetsフォルダのデータベースをコピー
   Log.i("kenpoActivity", "データベース無し" );

   //空のデータベース作成
   dbHelper.createEmptyDataBase();

   //データベースコピー
   //databaseCopy(DB_PATH,DB_NAME);

   //データベースが圧縮されている場合
   databaseUnZipCopy(DB_PATH,DB_NAME);
  }

 }

 /*
  * データベースの存在確認
  */
 private boolean databaseExists(String path, String name) {
  // TODO 自動生成されたメソッド・スタブ
  SQLiteDatabase checkDb = null;

  try {
   String dbPath = path + DB_NAME;
   checkDb = SQLiteDatabase.openDatabase(dbPath, null, SQLiteDatabase.OPEN_READONLY);
  } catch (SQLiteException e) {
   // データベースはまだ存在していない
   Log.i("kenpoActivity", "データベースが無い");
  }

  if (checkDb != null) {
   Log.i("kenpoActivity", "データベースが在る");
   checkDb.close();
  }

  return checkDb != null ? true : false;
 }

 /*
  * データベースのコピー
  */
 private void databaseCopy(String path, String name) {
  // TODO 自動生成されたメソッド・スタブ

  try {
   InputStream mInput = this.getAssets().open( name );//assetsフォルダのデータベース

   // デフォルトのデータベースパスに作成した空のDB
   String outFileName = path + DB_NAME;

   OutputStream mOutput = new FileOutputStream(outFileName);

   // コピー
   byte[] buffer = new byte[1024];
   int size;
   while ((size = mInput.read(buffer)) > 0) {
    mOutput.write(buffer, 0, size);
   }

   // Close the streams
   mOutput.flush();
   mOutput.close();
   mInput.close();

  } catch (IOException e) {
   // TODO 自動生成された catch ブロック
   e.printStackTrace();
  }

 }

 /*
  * データベースのコピー 圧縮されている場合
  */
 private void databaseUnZipCopy(String path, String name) {
  // TODO 自動生成されたメソッド・スタブ
  try {
   AssetManager am = getResources().getAssets();
   InputStream is = am.open("ropou.zip",AssetManager.ACCESS_STREAMING);
   ZipInputStream zis = new ZipInputStream(is);
   ZipEntry ze = zis.getNextEntry();

   if (ze != null) {
    String fullPath = path + DB_NAME;

    FileOutputStream fos = new FileOutputStream(fullPath, 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();
  }
 }

 /*
  * データベースヘルパー
  */
 public class DatabaseHelper extends SQLiteOpenHelper {

  public DatabaseHelper(Context context, String name, CursorFactory factory, int version) {
   super(context, name, factory, version);
   // TODO 自動生成されたコンストラクター・スタブ
  }

  @Override
  public void onCreate(SQLiteDatabase db) {
   // TODO 自動生成されたメソッド・スタブ

  }

  @Override
  public void onUpgrade(SQLiteDatabase db, int oldVersion, int newVersion) {
   // TODO 自動生成されたメソッド・スタブ

  }

  //空のデータベース作成
  public void createEmptyDataBase(){
   Log.i("kenpoActivity", "空のデータベース作成" );

   this.getReadableDatabase();
  }
 }

}


空のデータベース作成する部分は、databases フォルダがある場合は必要ないが、
アプリをインストールした時には、databases フォルダが無いので必要になる、
作成されるデータベースのパーミッションもちゃんと設定できるので、
空のデータベースを作成する部分は必要

2012年9月12日水曜日

ペアディバイスの取得


前の記事:Bluetoothアダプタを有効にする 

ペアディバイスを取得してListViewに表示
import java.util.Set;

import android.os.Bundle;
import android.app.Activity;
import android.bluetooth.BluetoothAdapter;
import android.bluetooth.BluetoothDevice;
import android.widget.ArrayAdapter;
import android.widget.ListView;

public class MainActivity extends Activity {

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

  // Bluetoothアダプタが利用可能か調べる
  BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();

  //ペアディバイスの表示
  Set pairedDevices = mBluetoothAdapter.getBondedDevices();

  //ペアディバイスをListViewに表示
  //ListView を取得
        ListView listView = (ListView) findViewById(R.id.listView1);
        //ListView のアダプタ設定
  ArrayAdapter mArrayAdapter = new ArrayAdapter(this, android.R.layout.simple_list_item_1);
  
  if (pairedDevices.size() > 0) {
      // Loop through paired devices
      for (BluetoothDevice device : pairedDevices) {
          // Add the name and address to an array adapter to show in a ListView
          mArrayAdapter.add(device.getName() + "\n" + device.getAddress());
       //Log.v("タグ", device.getName() + "\n" + device.getAddress() );
      }
  }
  
  listView.setAdapter(mArrayAdapter);
 }

}


    

    

    



2012年9月10日月曜日

オフラインで地図を使う:osmdroid



1.アプリをインストールする。
アプリ名:Osmdroid
https://play.google.com/store/apps/details?id=org.andnav.osm&feature=search_result#?t=W251bGwsMSwyLDEsIm9yZy5hbmRuYXYub3NtIl0.

2.地図データのダウンロード
日本の地図データをダウンロードしてくる。
http://downloads.cloudmade.com/asia/eastern_asia/japan

都道府県別にもダウンロード出来ます。
都道府県別のデータが容量が少ないのでお勧め。
例:aichi.osm.bz2

3.ダウンロードしたファイルをOsmdroidで利用できるようにする。
OsmAndMapCreator をダウンロードしてきて解凍する。
http://download.osmand.net/latest-night-build/

OsmAndMapCreator.bat をダブルクリックしてソフトを起動させる。
File -> Create .obf from osm file... を選択して、
ダウンロードしてきたファイルを選択すれば変換が始まります。

4.変換して出来上がったファイルをSDカードにコピー
保存先を変更していなければ、
C:\Users\ユーザー名\osmand に変更したデータが保存されている
アンドロイドのSDカードの osmand フォルダに obfファイルとtilesフォルダをコピーする。

5.Osmdroidを起動させる。
移動させたファイルは自動的に読み込まれる。

参考:
ドラッグで範囲指定
地図を動かすときは右ボタンを押しながら移動
拡大は右上のボタン

ダウンロードした地図データから範囲を指定して変換する場合は、
Create .obf from osm file for specifed area... を選択する。

地図データをダウンロードしていなくても直接、範囲を指定して変換する事もできる。

変換には結構時間が掛かる。
変換中にはtemp ファイルが作成されて結構な容量が必要、空き容量は30GB位は必要

OsmAndMapCreator.batをメモ帳で開き、メモリ最大サイズを増やして時間を短縮する。
初期の状態は720M

MapView の準備

OpenStreetMap
http://download.geofabrik.de/osm/asia/

mapsforge

map ファイルのダウンロード

download.mapsforge.org
More www.mapsforge.org

osmdroid





osm ファイルダウンロード
http://downloads.cloudmade.com/asia/eastern_asia/japan#downloads_breadcrumbs


javascript
OpenStreetMap.js
OpenLayers.js


2012年9月5日水曜日

BluetoothChatの使い方

ダウンロードしてきたファイルを解凍して、
プロジェクトをEclipseに読み込んで
BluetoothChatService.java のUUIDを変更する。

変更後
// Unique UUID for this application private static final UUID MY_UUID = UUID.fromString("00001101-0000-1000-8000-00805F9B34FB");

エラー対策:Class requires API level 5 (current min is 1)


Class requires API level 5 (current min is 1): android.bluetooth.BluetoothAdapter

パッケージ・エクスプローラからAndroidプロジェクトを選択して、
右クリック->Androidツール->Cleare Lint Marlers を選択する。


2012年9月2日日曜日

Bluetoothアダプタを有効にする

Bluetoothアダプタが利用可能か調べて
利用可能な場合、有効になっているか調べて、
有効になっていない場合有効にする。

import android.os.Bundle;
import android.app.Activity;
import android.bluetooth.BluetoothAdapter;
import android.content.Intent;
import android.util.Log;

public class MainActivity extends Activity {

 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  
  //Bluetoothアダプタが利用可能か調べる
  BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
  if (mBluetoothAdapter == null) {
   Log.v("タグ", "Bluetoothアダプタが利用出来ません");
  }else{
   Log.v("タグ", "Bluetoothアダプタが利用出来ます");
   
   //Bluetoothアダプタが有効か調べる
   if (mBluetoothAdapter.isEnabled()) {
    Log.v("タグ", "Bluetoothアダプタが有効です");
   }else {
    Log.v("タグ", "Bluetoothアダプタを有効にします");
    
    Intent enableBtIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
       startActivityForResult(enableBtIntent, 1);
   }
   
  }
 }

}


Bluetoothアダプタが利用可能か調べる


import android.os.Bundle;
import android.app.Activity;
import android.bluetooth.BluetoothAdapter;
import android.util.Log;

public class MainActivity extends Activity {

 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  
  //Bluetoothアダプタが利用可能か調べる
  BluetoothAdapter mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter();
  if (mBluetoothAdapter == null) {
   Log.v("タグ", "Bluetoothアダプタが利用出来ません");
  }else
   Log.v("タグ", "Bluetoothアダプタが利用出来ます");//
 }

}
    <uses-permission android:name="android.permission.BLUETOOTH"/>
    <uses-permission android:name="android.permission.BLUETOOTH_ADMIN"/>

Bluetoothアダプタが利用できないとBluetooth機能が使えない。

2012年8月31日金曜日

ヤフオクで売ってるボードでADKをやってみる

ボードの情報
http://mybd.sitemix.jp/pic_24FJ64GB002_2.html

回路図(必要な所だけ)


後は、 PIC24FJ64GB002でAndroid Accessory:ADK
の記事を参考に進める。


試す場合は自己責任でお願いします。
パソコンやアンドロイド端末が壊れる可能性があります。

PIC24FJ64GB002でAndroid Accessory:ADK

手元にある部品だけでとりあえず回路を組んでみました。
動作確認

参考にしたサイト:
side2(サイドツー)PIC24FJ64GB002でADKを試す(その2)
PIC24FJ64GB002のADK開発環境
第4回名古屋Android勉強会資料

試す場合は自己責任でお願いします。
パソコンやアンドロイド端末が壊れる可能性があります。

1.まず、pickit2でPIC24FJ64GB002にプログラムを書き込むのに必要な回路だけ組む
pickit3は持っていないのでpickit2を使ってPICkit 2 Programmer で書き込んだ。
書き込むプログラムはPIC24FJ64GB002でADKを試す(その2)でダウンロード出来るHexファイル

2.残りの回路を組む。

3.Android端末にアプリをインストールする。
Play ストアでMicrochipを検索してBasic Accessory Demoをインストールする。
Basic Accessory Demo

4.PICとAndroidを繋げて動作確認

参考
pickit2 で書き込めるリストが下記のアドレスで紹介されている。
http://www.microchip.com/stellent/idcplg?IdcService=SS_GET_PAGE&nodeId=1406&dDocName=en027813
PPIC24FJ64GB002もICkit 2 Programmer Application v2.61で書き込めるようです。

2012年8月20日月曜日

データベースの作成:sqlite

SQLiteOpenHelper クラスを使わないでデータベースの作成

import android.os.Bundle;
import android.app.Activity;
import android.content.ContentValues;
import android.database.sqlite.SQLiteDatabase;

public class MainActivity extends Activity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        //データベース作成
        SQLiteDatabase db;
        
        db = openOrCreateDatabase( "TestData.db" , SQLiteDatabase.CREATE_IF_NECESSARY , null );
        
        //テーブル作成
        String sql = "CREATE TABLE IF NOT EXISTS table_name (" 
          + "id INTEGER PRIMARY KEY AUTOINCREMENT,"
          + "name TEXT);";
  
        db.execSQL(sql);

        //レコードの追加
        ContentValues val = new ContentValues();
        val.put( "name", "sakkura" );
        db.insert( "table_name", null, val );
        
        db.close();

    }

}


2012年8月16日木曜日

TextView・一行表示・横スクロール

TextView で一行で表示される文字数を超えると自動的に改行される。


一行に制限する。
android:singleLine="true"


横スクロールさせる

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    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:ellipsize="marquee"
        android:focusable="true"
        android:focusableInTouchMode="true"
        android:marqueeRepeatLimit="marquee_forever"
        android:scrollHorizontally="true"
        android:singleLine="true"
        android:text="ABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWXYZ" />

        <TextView
        android:id="@+id/textView2"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:ellipsize="marquee"
        android:focusable="true"
        android:focusableInTouchMode="true"
        android:marqueeRepeatLimit="marquee_forever"
        android:scrollHorizontally="true"
        android:singleLine="true"
        android:text="ABCDEFGHIJKLMNOPQRSTUVWXYZ" />

</LinearLayout>

はみ出さない場合は、スクロールしない






2012年8月15日水曜日

ARROWS X F-10D 関連情報

Android携帯をやっと手に入れました。
最先端の高速クアッドコアCPU搭載 All in oneハイスペック防水スマートフォンという事でいろいろ調べて、F-10Dを選びました。

Android携帯端末ソフトウェア開発者向けサポート情報
docomo NEXT series ARROWS X F-10D

ディスプレイ・サイズ約 4.6インチ
赤外線通信対応IrSimpleTM/IrSSTMには非対応です。
Bluetooth®通信Bluetooth® 4.0
GPSオートGPS/海外GPS対応
モーションセンサー取扱説明書 34p
電子コンパス取扱説明書 129p
温度センサー取扱説明書 24p
湿度センサー取扱説明書 24p
RGBセンサー取扱説明書 24p
近接センサー取扱説明書 24p
FMトランスミッタ取扱説明書 24p
指紋センサー取扱説明書 24p
この表は、私が勝手に作りました。正確さは無保証

F-10D ADB用USBドライバ

サービス・機能とスペックを見ても書いてないセンサーや機能があるので、
お客様サポートの説明書をダウンロードして調べないと分からない。

センサーの名前も
モーションセンサーと加速度センサーは同じなのか?
電子コンパス(デジタルコンパス)とジャイロセンサー、地磁気センサーは同じなのか?
近接センサーとRGBセンサーの違いなど未確認です。

2012年8月9日木曜日

ソースコードの整形の設定・改行位置:Eclipse

ウインドウ->設定
Java->コードスタイル->フォーマッター

編集をクリックする。

折り返し タブを選択して
プレビュー・ウインドウの行の幅の設定 の部分を変更する。

2012年8月8日水曜日

Threadの利用方法:Java標準

Threadを利用しない場合



時間の掛かる処理終了後に画面が表示される。

import android.os.Bundle;
import android.app.Activity;
import android.widget.Toast;

public class MainActivity extends Activity {

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

  waitTime();// 時間のかかる処理

  Toast.makeText(this, "時間の掛かる処理終了後に実行されるはず", Toast.LENGTH_LONG).show();

 }

 private void waitTime() {
  // TODO 自動生成されたメソッド・スタブ
  try {
   Thread.sleep(10 * 1000);
  } catch (InterruptedException e) {
   // TODO 自動生成された catch ブロック
   e.printStackTrace();
  }
 }

}

Java標準のThreadを利用した並列処理を使った場合
時間の掛かる処理終了前に画面が表示される

import android.os.Bundle;
import android.app.Activity;
import android.widget.Toast;

public class MainActivity extends Activity {

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

  new Thread(new Runnable() {
   public void run() {
    // TODO Auto-generated method stub
    waitTime();// 時間のかかる処理
   }
  }).start();

  Toast.makeText(this, "時間の掛かる処理終了前に表示されるはず", Toast.LENGTH_LONG).show();

 }

 private void waitTime() {
  // TODO 自動生成されたメソッド・スタブ
  try {
   Thread.sleep(10 * 1000);
  } catch (InterruptedException e) {
   // TODO 自動生成された catch ブロック
   e.printStackTrace();
  }
 }

}

UI操作を行う場合 Handler を使う



時間の掛かる処理からデータを受け取ってUIを操作している。

import android.os.Bundle;
import android.os.Handler;
import android.app.Activity;
import android.widget.TextView;
import android.widget.Toast;

public class MainActivity extends Activity {

 private TextView textView;

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

  textView = (TextView) findViewById(R.id.textView1);
  
  final Handler handler = new Handler();
  
  new Thread(new Runnable() {
   public void run() {
    // TODO Auto-generated method stub
    
    final int num = waitTime();//時間のかかる処理
    
    handler.post(new Runnable() {
     public void run() {
      //int num = waitTime();// ここに置いたら別スレッドにならない
      textView.setText("UIを変更::" + num);
     }
    });
   }

  }).start();

  Toast.makeText(this, "これが先に実行されるはず", Toast.LENGTH_LONG).show();

 }

 private int waitTime() {
  // TODO 自動生成されたメソッド・スタブ
  try {
   Thread.sleep(10 * 1000);//10秒待つ
   
  } catch (InterruptedException e) {
   // TODO 自動生成された catch ブロック
   e.printStackTrace();
  }
  return 2012;
 }

}


NanoHTTPD を使ってみる2:indexファイルの利用

SDカードにindex.htmlがある場合

import java.io.File;
import java.io.IOException;
import android.app.Activity;
import android.os.Bundle;

public class MainActivity extends Activity {
 private MyHTTPD server;

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

 @Override
 protected void onResume() {
  super.onResume();

  //File am = new File("/data/data/com.example.servertest/lib/", "");//
 File am = new File("/mnt/sdcard/", "");//

  try {
   server = new MyHTTPD(8080, am);
  } catch (IOException e) {
   e.printStackTrace();
  }
 }

 @Override
 protected void onPause() {
  super.onPause();
  if (server != null)
   server.stop();
 }

 private class MyHTTPD extends NanoHTTPD {

  public MyHTTPD(int port, File wwwroot) throws IOException {
   super(port, wwwroot);
   // TODO 自動生成されたコンストラクター・スタブ
  }
 }
}




2012年8月3日金曜日

ボタンス・タイルのカスタマイズ


<?xml version="1.0" encoding="utf-8"?>
<selector xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="wrap_content">

    <!-- フォーカスされていない -->
    <!-- ボタンが押されていない -->
    <item android:state_focused="false"
        android:state_pressed="false">
    <layer-list>
            <item><shape xmlns:android="http://schemas.android.com/apk/res/android"
                android:shape="rectangle">
                    <gradient android:angle="270"
                        android:startColor="#000000"
                        android:centerColor="#ffffff"
                        android:endColor="#000000"
                        android:type="linear" />

                    <corners android:radius="4dip" />

                    <stroke android:width="1dip"
                        android:color="#ffffff" />
                    <padding android:top="5dip"
                        android:bottom="5dip"/>
                </shape></item>
        </layer-list></item>

    <!-- フォーカスされていない -->
    <!-- ボタンが押されている -->
    <item android:state_focused="false"
        android:state_pressed="true"><layer-list>
            <item><shape xmlns:android="http://schemas.android.com/apk/res/android"
                android:shape="rectangle">
                    <gradient android:angle="270"
                        android:startColor="#ffffff"
                        android:centerColor="#000000"
                        android:endColor="#ffffff"
                        android:type="linear" />

                    <corners android:radius="4dip" />

                    <stroke android:width="1dip"
                        android:color="#ffffff" />
                </shape></item>
        </layer-list></item>

    <!-- フォーカスされた -->
    <!-- ボタンが押されていない -->
    <item android:state_focused="true"
        android:state_pressed="false"><layer-list>
            <item><shape xmlns:android="http://schemas.android.com/apk/res/android"
                android:shape="rectangle">
                    <gradient android:angle="270"
                        android:startColor="#000000"
                        android:centerColor="#ffffff"
                        android:endColor="#000000"
                        android:type="linear" />

                    <corners android:radius="4dip" />

                    <stroke android:width="1dip"
                        android:color="#FF6600" />
                </shape></item>
        </layer-list></item>

    <!-- フォーカスされた -->
    <!-- ボタンが押されている -->
    <item android:state_focused="true"
        android:state_pressed="true"><layer-list>
            <item><shape xmlns:android="http://schemas.android.com/apk/res/android"
                android:shape="rectangle">
                    <gradient android:angle="270"
                        android:startColor="#ffffff"
                        android:centerColor="#000000"
                        android:endColor="#ffffff"
                        android:type="linear" />

                    <corners android:radius="4dip" />

                    <stroke android:width="1dip"
                        android:color="#CC6600" />
                </shape></item>
        </layer-list></item>

</selector>

<?xml version="1.0" encoding="utf-8"?>
<resources xmlns:android="http://schemas.android.com/apk/res/android">

    <style name="button" >
        <item name="android:layout_marginTop">1dip</item>
  <item name="android:layout_marginBottom">1dip</item>
    </style>

</resources>

<?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"
        style="@style/button"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="@layout/mybutton"
        android:text="Button" />

    <Button
        android:id="@+id/button2"
        style="@style/button"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:background="@layout/mybutton"
        android:text="Button" />

</LinearLayout>

Android で NanoHTTPD を使ってみる

android に webサーバ機能を追加

NanoHTTPD の情報
http://elonen.iki.fi/code/nanohttpd/

とりあえずやってみる

package com.example.servertest;

import java.io.File;
import java.io.IOException;
import java.util.Properties;

import android.app.Activity;
import android.os.Bundle;

public class MainActivity extends Activity {
 private MyHTTPD server;
 @Override
 public void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
 }

 @Override
 protected void onResume() {
  super.onResume();

  try {
   server = new MyHTTPD(8080, null);
  } catch (IOException e) {
   e.printStackTrace();
  }
 }

 @Override
 protected void onPause() {
  super.onPause();
  if (server != null)
   server.stop();
 }

 private class MyHTTPD extends NanoHTTPD {


  public MyHTTPD(int port, File wwwroot) throws IOException {
   super(port, wwwroot);
   // TODO 自動生成されたコンストラクター・スタブ
  }

  @Override
  public Response serve(String uri, String method, Properties header,
    Properties parms, Properties files) {
   System.out.println(method + " '" + uri + "' ");
   String msg = "<html><body><h1>Hello server</h1>\n";
   msg += "</body></html>\n";
   return new NanoHTTPD.Response(HTTP_OK, MIME_HTML, msg);
  }
 }
}

マニフェストファイルの設定
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE"/>
<uses-permission android:name="android.permission.INTERNET"/>

ダウンロードしてきた NanoHTTPD.java を同じパッケージ内に置く
実機にインストールしてアプリを実行する。

ブラウザでandroidのアドレスにアクセスする。


カメラ映像をPCに転送するアプリ

Androidでカメラ画像を転送するアプリでソースコードが公開されているもの
sipdroid
http://code.google.com/p/sipdroid/

ipcamera-for-android
http://code.google.com/p/ipcamera-for-android/

spydroid-ipcamera
http://code.google.com/p/spydroid-ipcamera/

SmartLab HTML5 Camera and Android


HTML5 Camera and Android とipcamera-for-android がシンプルで分かりやすい。


2012年8月2日木曜日

インターネット上の動画ファイルを再生:VideoView


import android.net.Uri;
import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
import android.widget.MediaController;
import android.widget.VideoView;

public class MainActivity extends Activity {

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        VideoView videoView = (VideoView)findViewById(R.id.videoView1);
        videoView.setMediaController(new MediaController(this));
        
        String LINK = "http://www.xxx.jp/xxx/xxx.mp4";
        
        Uri uri=Uri.parse(LINK);

        VideoView video=(VideoView)findViewById(R.id.videoView1);
        video.setVideoURI(uri);
        video.start();

    }

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        getMenuInflater().inflate(R.menu.activity_main, menu);
        return true;
    }
}

AVDでは再生は出来るが画質が悪い、
実機ではうまく再生できている様です。

android SDK Tools 20.0.1 で不具合

android SDK Tools を 20.0.1 にアップグレードしたらいろいろ不具合が出た。

LogCatがおかしい
ビューを作成できませんでした: プラグイン "com.android.ide.eclipse.ddms" は、クラス "com.android.ide.eclipse.ddms.views.LogCatView" をインスタンス化できませんでした。

プロジェクトが作成できない。
必要なパッケージがインストール出来ない。

eclipse フォルダの eclipse.exe -clean.cmd をダブルクリックで起動すればなおりました。



2012年8月1日水曜日

カウントダウンタイマーでシングルタップとダブルタップの判定



import android.app.Activity;
import android.os.Bundle;
import android.os.CountDownTimer;
import android.util.Log;
import android.view.MotionEvent;
import android.view.View;
import android.view.ViewConfiguration;
import android.widget.LinearLayout;

public class CountdownTimerActivity extends Activity {
    /** Called when the activity is first created. */
 private boolean mTimerRunning = false;
 private boolean doubleTapFlag = false;
 
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        LinearLayout listView = (LinearLayout) findViewById(R.id.LinearLayout1);
        
        //ダブルタップの間隔取得
        final int doubleTime = ViewConfiguration.getDoubleTapTimeout();

  listView.setOnTouchListener(new View.OnTouchListener() {
   
   public boolean onTouch(View v, MotionEvent event) {
    Log.i("TAG", "タップされた");

    // 二重起動を防止する
    if (!mTimerRunning) {
     mTimerRunning = true;
     // カウントダウンする
     new CountDownTimer(doubleTime, 1) {
      // カウントダウン処理
      public void onTick(long millisUntilFinished) {
     }
      // カウントが0になった時の処理
      public void onFinish() {
       if( doubleTapFlag ){
        Log.v("LOG", "ダブルタップ");
        doubleTapFlag = false;
       }else{
        Log.i("LOG", "シングルタップ");
       }
       
       mTimerRunning = false;
      }
     }.start();
    }else{
     //ダブルタップ
     doubleTapFlag = true;
    }
    return false;
   }
  });
 }
}

ダブルタップを単に取得したいなら、GestureDetectorクラスで簡単に取得できます。

中華タブレットでwifiデバック

eee pc を使ってwifiデバッグ でもいろいろ苦労したけど、
中華タブレットはもっと苦労した。

結局、USBケーブルを接続した状態で、
adb connect 192.168.24.xx を打てば接続は出来る。


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");

2012年6月30日土曜日

Android端末の音楽ファイルをパソコンで聞く:Bluetooth

使用Bluetoothアダプタ:BUFFALO BSHSBD04BK

ペアリング済のディバイスを接続して、
Media にチェックを入れる。

オーディオの音楽再生の部分が自動的に接続状態になる。

アンドロイド端末で普通再生すればパソコンでも再生される。


プログラムで再生
SDカードの中のファイルをメディアプレイヤーで再生させて、
パソコンにBluetoothで接続して、パソコンでも再生させる。

import android.content.*;
import android.app.Activity;
import android.os.Bundle;
import android.media.*;
import java.io.*;
import android.util.Log;

public class AnTemptestActivity extends Activity
{
    private static final String TAG = "BTAudioActivity";

    private MediaPlayer mPlayer = null;
    private AudioManager amanager = null;

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

        amanager = (AudioManager) getSystemService(Context.AUDIO_SERVICE);
        amanager.setBluetoothScoOn(true);
        amanager.setMode(AudioManager.MODE_IN_CALL);

        mPlayer = new MediaPlayer();

        try {
            mPlayer.setDataSource(new FileInputStream("/sdcard/Voice/rec.3gpp").getFD());

            mPlayer.setAudioStreamType(AudioManager.STREAM_VOICE_CALL);

            mPlayer.prepare();

            mPlayer.start();
        } catch(Exception e) {
            Log.e(TAG, e.toString());
        }
    }

    @Override
    public void onDestroy()
    {
        mPlayer.stop();
        amanager.setMode(AudioManager.MODE_NORMAL);
        amanager.setBluetoothScoOn(false);
        super.onDestroy();
    }
}