2013年7月24日水曜日

Android NDK に挑戦



やっと、HelloJni が動いたのでメモ

SDK は adt-bundle-windows-x86_64-20130717.zip を解凍して利用

eclipse は adt-bundle-windows-x86_64-20130717 の中にあるやつを利用。

NDK は android-ndk-r8e-windows-x86_64.zip を解凍して利用

eclipseの設定
NDK Location を設定

サンプルをワークスペースに読み込む。
ファイルからエクスポートを選択する。


D:\soft\android-ndk-r8e\samples\hello-jni を選択して、
ワークスペースにコピーにチェックを入れる。

読み込まれたプロジェクトを右クリックして、
Android Tool から Add Native Support... を選択する。

そのまま フィニッシュ

このまま実行してもエラーが出る。

sdk のバージョンを調整する。

Android SDK Manager で Android 2.3.3 のパッケージをダウンロードする。
eclipse の再起動が必要かも

AVD を作成する。

ビルドターゲットを変更する。
パッケージエクスプローラからブロジェクトを選んで右クリックして、
プロパティーを選択して、
Android 2.3.3 をを選択する。


マニフェストの minSdkVersion を9に変更する
たぶんこの辺が重要なんだろうか。

C++ のパースペクティブから、
プロジェクトのクリーン、
プロジェクトのビルド、をして、
実行すれば完了。

感想。
いろいろなサイトで、環境変数の登録との説明とかがあったが、
今のところ必要ないようです。

JNI は 、Java から C++ のライブラリを使ったり、その逆のための仕組み。
Android 固有の仕組みではない。

2013年7月18日木曜日

ListView の最後の行の下線を表示させる


layout_height の指定を、wrap_content から、match_parent に変更する。

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

ListViewの中のボタンの処理

ListView の中のボタンを押したら2行目を非表示にする。


1行目のボタンを押した場合↓
package com.example.listviewtest;

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

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

public class MainActivity extends Activity {

 private HashMap<String, String> data;

 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, 
    R.layout.row, 
    new String[] { "title", "comment" }, 
    new int[] { android.R.id.text1,
    android.R.id.text2 });

  ListView listView = (ListView) findViewById(R.id.listView1);
  listView.setAdapter(adapter);//
 }
}
package com.example.listviewtest;

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

import android.content.Context;
import android.util.Log;
import android.view.LayoutInflater;
import android.view.View;
import android.view.View.OnClickListener;
import android.view.ViewGroup;
import android.widget.Button;
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(final int position, View convertView, ViewGroup parent) {
  final ViewHolder holder;

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

  if (view == null) {
   view = inflater.inflate(R.layout.row, 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);

  Button btn = (Button) view.findViewById(R.id.button1);
  btn.setTag(position);
  
  btn.setOnClickListener(new OnClickListener() {
   @Override
   public void onClick(View arg0) {
    // TODO 自動生成されたメソッド・スタブ
    Log.v("buttonクリック", "ポジション: " + position);
    if(holder.line2.getVisibility() == android.view.View.GONE){
     holder.line2.setVisibility(View.VISIBLE);
    }else
    holder.line2.setVisibility(View.GONE);
   }
  });
 
  return view;
 }
}
main.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" >

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

</LinearLayout>
row.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">

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content" >

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

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

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


2013年7月17日水曜日

GPSを利用する

AndroidManifest.xmlにパーミッションを設定する。

GPSによる位置情報利用
<uses-permission android:name=”android.permission.ACCESS_FINE_LOCATION”/>

ネットワークによる位置情報利用
<uses-permission android:name=”android.permission.ACCESS_COARSE_LOCATION”/>

ロケーションマネージャーのインスタンスを取得
LocationManager locationManager = (LocationManager) getSystemService(LOCATION_SERVICE);

GPSの状態を取得する
locationManager.addGpsStatusListener(this);
 /*
  * Listener
  * GPS の状態が通知されるメソッド
  */
 @Override
 public void onGpsStatusChanged(int event) {
  // TODO 自動生成されたメソッド・スタブ
  String status = "";
  if (event == GpsStatus.GPS_EVENT_FIRST_FIX) {
   status = "FIRST FIX:初めて位置情報を確定した:";
  } else if (event == GpsStatus.GPS_EVENT_SATELLITE_STATUS) {
   status = "SATELLITE STATUS:GPSが位置情報を取得中";
  } else if (event == GpsStatus.GPS_EVENT_STARTED) {
   status = "STARTED:GPSを使い位置情報の取得を開始した";
  } else if (event == GpsStatus.GPS_EVENT_STOPPED) {
   status = "STOPPED:GPSの位置情報取得が終了した";
  }
  Log.d("onGpsStatusChanged", status);
 }


利用できるプロバイダの一覧表示
List<String> providers = locationManager.getAllProviders();
  for (String provider : providers) {
   Log.d("providers", provider);
  }

実行結果:
07-17 12:55:05.809: D/providers(2053): network
07-17 12:55:05.809: D/providers(2053): passive
07-17 12:55:05.809: D/providers(2053): gps

gps(GPS, AGPS)
人工衛星より位置情報を取得する。
建物の中や、高い建物付近だとGPSが遮られて位置を取得できない。
バッテリー消費大。高精度。

network (AGPS, CellID, Wifi MACID)
携帯端末の基地局や接続しているWifiから位置情報を特定する。
バッテリー消費中。中精度。 

passive(CellID, Wifi MACID)
位置情報取得設定が全てオフの場合実行される。
network との違いは、AGPSを使用しない事。
バッテリー消費小。低精度

条件を指定して最適なプロバイダを取得する。
  Criteria criteria = new Criteria();
  criteria.setAccuracy(Criteria.ACCURACY_LOW);//細かい位置精度
/*  criteria.setBearingRequired(false); // 方位不要
  criteria.setSpeedRequired(false); // 速度不要
  criteria.setAltitudeRequired(false); // 高度不要
*/  
  String provider = locationManager.getBestProvider(criteria, true); 
  Log.d("getBestProvider", provider);//gps

プロバイダの状態を表示する
  List<String> providers = locationManager.getAllProviders();
    for (String provider : providers) {
    boolean status = locationManager.isProviderEnabled(provider);
    Log.d("providers", provider + "::" + status);
  }
表示結果:
07-17 13:25:37.829: D/providers(2297): network::false
07-17 13:25:37.830: D/providers(2297): passive::true
07-17 13:25:37.831: D/providers(2297): gps::true

使えるプロバイダを取得する場合
List<String> providers = locationManager.getProviders(true);


アプリのインストール場所を指定


android:installLocation="internalOnly"
内部メモリーに保存される。

android:installLocation="auto"
内部メモリーに保存するか、外部に保存するか自動的に決まります。
ユーザー自身で内部と外部への移動が可能。
android:installLocation="preferExternal"
外部メモリーに保存します。
ユーザー自身で内部と外部への移動が可能です。
外部メモリーがいっぱいだと、内部メモリーに保存される。