手順4、画像 http://なんちゃら.jpg を表示する、しかも非同期処理で
手順3では画面に文字列を出力することができたものの、http://なんちゃら.jpg の様な画像ファイルを表示する事ができてません。
HTML であれば <img src=”http://なんちゃら.jpg”> とすれば、ブラウザ上に表示してくれるのですが、世界はAndroidアプリですのでHTMLタグは使えません。
(WebViewというものを用いれば利用出来ますが・・)
Androidアプリでは ImageView と Bitmap を用いて画像を描画します。
また、画像データをインターネット経由で取得し秒がするという処理は少し時間がかかる処理なので、メインのUIスレッド(画面に文字を表示する処理)とは別の非同期のスレッド(画像を読み込んで読み込み終わったら画面に出力する処理)を走らせてあげる必要があります、とどっかのサイトに書いてありました。
・画像URLを画像として表示するためにレイアウトファイルの修正
hogehoge_sub.xml を修正します。
赤字が修正したところ。 TextViewからImageViewにしたよ。
<?xml version=“1.0” encoding=“utf-8”?>
<LinearLayout
xmlns:android=“http://schemas.android.com/apk/res/android”
android:orientation=“vertical”
android:layout_width=“wrap_content”
android:layout_height=“fill_parent”
>
<TextView
android:id=“@+id/itemId”
android:layout_width=“fill_parent”
android:layout_height=“wrap_content”
/>
<TextView
android:id=“@+id/itemTitle”
android:layout_width=“fill_parent”
android:layout_height=“wrap_content”
/>
<TextView
android:id=“@+id/itemDetail”
android:layout_width=“fill_parent”
android:layout_height=“wrap_content”
/>
<ImageView
android:id=“@+id/itemIcon”
android:layout_width=“wrap_content”
android:layout_height=“wrap_content”
/>
<ImageView
android:id=“@+id/itemImage”
android:layout_width=“wrap_content”
android:layout_height=“wrap_content”
/>
</LinearLayout>
・画像URLを画像として表示させるためのアクティビティを修正
HTMLなら画像タグ<img src=”なんちゃら.jpg”> で完了なんだけどもなぁ。
何行も追加することになりましたね。
まずは非同期処理ではなく単純に画像を表示させてみます。
package jp.example.hello;
import java.io.BufferedInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.net.URL;
import java.net.URLConnection;
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.util.EntityUtils;
import org.json.JSONArray;
import org.json.JSONObject;
import android.app.Activity;
import android.app.ProgressDialog;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.TextView;
public class HogehogeActivity extends Activity {
privatestaticfinalString LogTag = “HogehogeLog”;
private ProgressDialog dialog;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.hogehoge);
// jsonを取得して解析する
LoadTask();
}
private StringmakeApiUrl(){
return “http://akamako.com/blogger/sample.php”;
}
privatevoidLoadTask(){
// ProgressDialogを作成
dialog = new ProgressDialog(this);
dialog.setMessage(“Connect to Server.”);
dialog.setCancelable(true);
dialog.show();
try{
String apiUrl = makeApiUrl();
// コンソールログに出力
Log.d(LogTag+” apiUrl”,apiUrl);
HttpGet get = new HttpGet(apiUrl);
HttpClient DefaultHttpClient = new DefaultHttpClient();
HttpResponse response = DefaultHttpClient.execute(get);
// ステータスコード
int status = response.getStatusLine().getStatusCode();
Log.d(LogTag+” status”,String.valueOf(status));
if (status != HttpStatus.SC_OK) {
throw new Exception(“Error!”);
}
// 結果を取得
String source =EntityUtils.toString(response.getEntity());
Log.d(LogTag+” source”,source);
// JSONObject という型があるんだって
JSONObject json = new JSONObject(source);
if (json.get(“items”) == JSONObject.NULL){
throw new Exception(“Error!”);
}
if (json.get(“count”) == JSONObject.NULL){
throw new Exception(“Error!”);
}
// 記事データ
JSONArray items = json.getJSONArray(“items”);
// 記事の数
String count = json.getString(“count”);
if(count.equals(“0”) == true){
// 記事データが無かったお
}
// items の中身の数
int maxI = items.length();
LinearLayout layout = (LinearLayout) findViewById(R.id.hogehoge_layout);
URL aURL;
URLConnection conn;
InputStream is;
BufferedInputStream bis;
Bitmap bm;
for(int i = 0; i < maxI; i++) {
// 1つ取り出す
JSONObject item = items.getJSONObject(i);
// 表示項目を初期化
String itemId = “”;
String itemTitle=“”;
String itemDetail=“”;
String itemIconUrl=“”;
String itemImageUrl=“”;
if(item.has(“id”)) {
itemId = item.getString(“id”);
}
if(item.has(“title”)) {
itemTitle = item.getString(“title”);
}
if(item.has(“detail”)) {
itemDetail = item.getString(“detail”);
}
if(item.has(“icon”)) {
itemIconUrl = item.getString(“icon”);
}
if(item.has(“image”)) {
itemImageUrl = item.getString(“image”);
}
Log.d(LogTag + ” itemId”, itemId);
Log.d(LogTag + ” itemTitle”, itemTitle);
Log.d(LogTag + ” itemDetail”, itemDetail);
Log.d(LogTag + ” itemIconUrl”, itemIconUrl);
Log.d(LogTag + ” itemImageUrl”, itemImageUrl);
// layout/hogehoge_list.xml が1記事のテンプレートです
View view = getLayoutInflater().inflate(R.layout.hogehoge_sub, null);
layout.addView(view);
// 各表示項目のスキーマ(?)を取り出す
TextView itemId_v = (TextView) view.findViewById(R.id.itemId);
TextView itemTitle_v = (TextView) view.findViewById(R.id.itemTitle);
TextView itemDetail_v = (TextView) view.findViewById(R.id.itemDetail);
// TextView itemIcon_v = (TextView) view.findViewById(R.id.itemIcon);
// TextView itemImage_v = (TextView) view.findViewById(R.id.itemImage);
ImageView itemIcon_v = (ImageView) view.findViewById(R.id.itemIcon);
ImageView itemImage_v = (ImageView) view.findViewById(R.id.itemImage);
// 該当の表示個所に当て込む
itemId_v.setText(itemId);
itemTitle_v.setText(itemTitle);
itemDetail_v.setText(itemDetail);
// itemIcon_v.setText(itemIconUrl);
// itemImage_v.setText(itemImageUrl);
aURL = new URL(itemIconUrl);
conn = aURL.openConnection();
conn.connect();
is = conn.getInputStream();
bis = new BufferedInputStream(is);
bm = BitmapFactory.decodeStream(bis);
bis.close();
is.close();
itemIcon_v.setImageBitmap(bm);
aURL = new URL(itemImageUrl);
conn = aURL.openConnection();
conn.connect();
is = conn.getInputStream();
bis = new BufferedInputStream(is);
bm = BitmapFactory.decodeStream(bis);
bis.close();
is.close();
itemImage_v.setImageBitmap(bm);
}
// 接続を解除する
DefaultHttpClient.getConnectionManager().shutdown();
}
catch (ClientProtocolException e){
Log.d(LogTag + ” ClientProtocolException”, e.getMessage());
}
catch (IOException e){
Log.d(LogTag + ” IOException”, e.getMessage());
}
catch(Exception e){
Log.d(LogTag + ” Exception”, e.getMessage());
}
if (dialog != null && dialog.isShowing()) {
dialog.dismiss();
}
}
}
・動作確認
うまくいくかなぁ
やたー!
写真が表示されたったー!
あれ、でもスクロールされないから下の方の記事データが見れない・・・
・スクロールビュー
スクロールビューを使うと画面上にスクロールできる枠を作れるらしい。
なんとなく 子ビューのhogehoge_sub.xmlではなく、親ビューhogehoge.xmlで子ビューが当て込まれるLinearLayoutをScrollViewで囲ってあげれば良いのではないか?!
(直感)
<?xml version=“1.0” encoding=“utf-8”?>
<ScrollView xmlns:android=“http://schemas.android.com/apk/res/android”
android:id=“@+id/ScrollView01”
android:layout_width=“fill_parent”
android:layout_height=“fill_parent”>
<LinearLayout
android:id=“@+id/hogehoge_layout”
android:orientation=“vertical”
android:layout_height=“wrap_content“
android:layout_width=“fill_parent”>
</LinearLayout>
</ScrollView>
・動作確認
おぉ!うまくスクロールになった!マグレ!ラッキー!
でもやっぱ表示されるまで多少時間がかかるね。数秒。
・非同期処理をアクティビティに埋め込む
さてここからですな。
非同期処理を実装してみよう。
非同期処理で重い処理(画像の読み込み・表示部分)を任せると、テキストデータだけパッと表示されて、しばらくすると画像が現れ始めるオシャレな動きになるはずや!
jqueryでいうところのajaxの様に、容量の大きな画像を読み込んでいる間やサーバー側の更新処理の完了待ちの間、画面を動かす事ができる。と。
やっぱり画面が固まるってのは、近年では”ダサい”と思われちゃう。大変な世の中です。
Android開発の必殺技AsyncTasc超基本形(超コピペ用)
public class hogeHogeTask extends AsyncTask<Void, Void, Void>{
// コンストラクタ
public hogeHogeTask(){}
// 最初にUIスレッドで呼び出されます。UIに関わる処理をします。
protected void onPreExecute() {}
// バックグラウンドで行う時間のかかる処理をします。
protected Void doInBackground(Void… params) {return null;}
// doInBackgroundメソッドの処理終了後、UIスレッドに返します
protected void onPostExecute(Void result){}
}
このクラスを取り敢えずHogehogeアクティビティのプライベートクラスなどで定義してみます。
この書き方を最初に目にした時は
AsyncTask<Void, Void, Void> の部分と (Void… params) の部分が全く意味が分からず「うぅ!(嫌悪)」とドギマギしました。
引数にさ … って何やねん!しゃっきりせい! って。
大丈夫。これらの書き方は単なる 呪文(決まり事)らしいです。
ここでは説明を省きますが、詳しくはGoogle先生に聞いて下さいね(笑)
クラス名は、画像をロードするから LoadImageTask にしよっと。
HogehogeActivity.java に追記した所が赤文字、コメントアウトしたところが灰色です。
package jp.example.hello;
import java.io.IOException;
import java.io.InputStream;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
import org.apache.http.HttpResponse;
import org.apache.http.HttpStatus;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.util.EntityUtils;
import org.json.JSONArray;
import org.json.JSONObject;
import android.app.Activity;
import android.app.ProgressDialog;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.AsyncTask;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.ImageView;
import android.widget.LinearLayout;
import android.widget.TextView;
public class HogehogeActivity extends Activity {
private static final String LogTag = “HogehogeLog”;
private ProgressDialog dialog;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.hogehoge);
// jsonを取得して解析する
LoadTask();
}
private String makeApiUrl(){
return “http://akamako.com/blogger/sample.php”;
}
private void LoadTask(){
// ProgressDialogを作成
dialog = new ProgressDialog(this);
dialog.setMessage(“Connect to Server.”);
dialog.setCancelable(true);
dialog.show();
try{
String apiUrl = makeApiUrl();
// コンソールログに出力
Log.d(LogTag+” apiUrl”,apiUrl);
HttpGet get = new HttpGet(apiUrl);
HttpClient DefaultHttpClient = new DefaultHttpClient();
HttpResponse response = DefaultHttpClient.execute(get);
// ステータスコード
int status = response.getStatusLine().getStatusCode();
Log.d(LogTag+” status”,String.valueOf(status));
if (status != HttpStatus.SC_OK) {
throw new Exception(“Error!”);
}
// 結果を取得
String source = EntityUtils.toString(response.getEntity());
Log.d(LogTag+” source”,source);
// JSONObject という型があるんだって
JSONObject json = new JSONObject(source);
if (json.get(“items”) == JSONObject.NULL){
throw new Exception(“Error!”);
}
if (json.get(“count”) == JSONObject.NULL){
throw new Exception(“Error!”);
}
// 記事データ
JSONArray items = json.getJSONArray(“items”);
// 記事の数
String count = json.getString(“count”);
if(count.equals(“0”) == true){
// 記事データが無かったお
}
// items の中身の数
int maxI = items.length();
LinearLayout layout = (LinearLayout) findViewById(R.id.hogehoge_layout);
// URL aURL;
// URLConnection conn;
// InputStream is;
// BufferedInputStream bis;
// Bitmap bm;
for(int i = 0; i < maxI; i++) {
// 1つ取り出す
JSONObject item = items.getJSONObject(i);
// 表示項目を初期化
String itemId = “”;
String itemTitle=“”;
String itemDetail=“”;
String itemIconUrl=“”;
String itemImageUrl=“”;
if(item.has(“id”)) {
itemId = item.getString(“id”);
}
if(item.has(“title”)) {
itemTitle = item.getString(“title”);
}
if(item.has(“detail”)) {
itemDetail = item.getString(“detail”);
}
if(item.has(“icon”)) {
itemIconUrl = item.getString(“icon”);
}
if(item.has(“image”)) {
itemImageUrl = item.getString(“image”);
}
Log.d(LogTag + ” itemId”, itemId);
Log.d(LogTag + ” itemTitle”, itemTitle);
Log.d(LogTag + ” itemDetail”, itemDetail);
Log.d(LogTag + ” itemIconUrl”, itemIconUrl);
Log.d(LogTag + ” itemImageUrl”, itemImageUrl);
// layout/hogehoge_list.xml が1記事のテンプレートです
View view = getLayoutInflater().inflate(R.layout.hogehoge_sub, null);
layout.addView(view);
// 各表示項目のスキーマ(?)を取り出す
TextView itemId_v = (TextView) view.findViewById(R.id.itemId);
TextView itemTitle_v = (TextView) view.findViewById(R.id.itemTitle);
TextView itemDetail_v = (TextView) view.findViewById(R.id.itemDetail);
// TextView itemIcon_v = (TextView) view.findViewById(R.id.itemIcon);
// TextView itemImage_v = (TextView) view.findViewById(R.id.itemImage);
ImageView itemIcon_v = (ImageView) view.findViewById(R.id.itemIcon);
ImageView itemImage_v = (ImageView) view.findViewById(R.id.itemImage);
// 該当の表示個所に当て込む
itemId_v.setText(itemId);
itemTitle_v.setText(itemTitle);
itemDetail_v.setText(itemDetail);
// itemIcon_v.setText(itemIconUrl);
// itemImage_v.setText(itemImageUrl);
// aURL = new URL(itemIconUrl);
// conn = aURL.openConnection();
// conn.connect();
// is = conn.getInputStream();
// bis = new BufferedInputStream(is);
// bm = BitmapFactory.decodeStream(bis);
// bis.close();
// is.close();
// itemIcon_v.setImageBitmap(bm);
//
// aURL = new URL(itemImageUrl);
// conn = aURL.openConnection();
// conn.connect();
// is = conn.getInputStream();
// bis = new BufferedInputStream(is);
// bm = BitmapFactory.decodeStream(bis);
// bis.close();
// is.close();
// itemImage_v.setImageBitmap(bm);
new LoadImageTask(itemIcon_v).execute(itemIconUrl);
new LoadImageTask(itemImage_v).execute(itemImageUrl);
}
// 接続を解除する
DefaultHttpClient.getConnectionManager().shutdown();
}
catch (ClientProtocolException e){
Log.d(LogTag + ” ClientProtocolException”, e.getMessage());
}
catch (IOException e){
Log.d(LogTag + ” IOException”, e.getMessage());
}
catch(Exception e){
Log.d(LogTag + ” Exception”, e.getMessage());
}
if (dialog != null && dialog.isShowing()) {
dialog.dismiss();
}
}
public class LoadImageTask extends AsyncTask<String, Void, Bitmap> {
// アイコンを表示するビュー
private ImageView imageView;
// コンストラクタ
public LoadImageTask(ImageView imageView) {
this.imageView = imageView;
}
// バックグラウンドで行う時間のかかる処理をします。
@Override
protected Bitmap doInBackground(String… urls) {
Bitmap img = null;
try {
Log.d(LogTag+” loadImageTask url : “,urls[0]);
URL imageUrl = new URL(urls[0]);
HttpURLConnection itemIconCon = (HttpURLConnection)(imageUrl).openConnection();
InputStream is = itemIconCon.getInputStream();
img = BitmapFactory.decodeStream(is);
} catch (MalformedURLException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return img;
}
// doInBackgroundメソッドの処理終了後、UIスレッドに返します
@Override
protected void onPostExecute(Bitmap result) {
// キャンセルされていたらなにもしない
if (isCancelled()) {
result = null;
}
if (result != null) {
if (imageView != null) {
imageView.setImageBitmap(result);
}
}
}
}
}
クラスの中にクラスがある!!
PHPエンジニアからするとあまり見ない記述。
まぁeclipseでシンタックスエラーになってないからいいか。
・動作確認
ドキドキ。。
お!?
ぬ、お!?
おー!
でけたー!
徐々に画像が読み込まれてスクロールバーも徐々に長くなっていった。
うまくできましたねぇ。
コメントを残す