2013年12月19日木曜日

世界の終わり セカイカメラのサービス終了

セカイカメラサービス終了のお知らせ
http://sekaicamera.com/the_end_of_sekai
↑URLがナイス

数年前に拡張現実の先駆けであったセカイカメラが、
サービスを終了する。
利用者が少なかったからだろうか。
それとも、マネタイズがうまくいかなかったからなのだろうか。

セカイカメラは終了するが、拡張現実事態にはまだまだ需要はあるはずだ。
ゲームや広告をはじめ、
事業としては観光業界とか家具販売とか美術館とか。

拡張現実の実現方法にもいろいろあって、
個人が簡単に実現できるものから、技術が必要なものもある。
以下、拡張現実の種類を勝手に分類してみた。

  • 位置情報型
利用者のいる緯度経度から付加情報を表示させる。
セカイカメラは位置情報型だろう。
観光事業では、観光名所にタグをつけ、旅行者向けの情報を出現させるといったことができる。
位置情報を使った道案内もできるだろう。

  • マーカー型
QRコードなどのマーカーを読み取ることで付加情報を表示させる。
広告に載せてクーポンを提供することができたりする。
マーカーを床に置いて家具を配置してみると、購入後の雰囲気がわかるといったサービスが可能。

  • マーカーレス型
こちらは付加情報を出現させるために、マーカーではなく、現実の物を使う手法。
木があったらクリスマスツリーにしてみるとか、
顔認識でその人の名前を表示させるとか。

  • センサー型
端末がセンサーを認識すると付加情報を出現させる。
位置情報型よりも細かく場所を特定したい場合に利用する。
美術館でセンサーに反応して美術品の説明を表示させたりできると思う。
ゲームで、センサーに反応したらイベントが発生するとかしたら面白そう。

あと何年かしたら、拡張現実が生活の一部になっていることを予想している。
楽しそう。

2013年12月18日水曜日

Webサービスの企画書を見れる

Webサービスの企画書が公開されているのを見つた。


ネットで見れる!国内有名Webサービスの「ナマ企画書」まとめ

ネットで見れる!国内有名Webサービスの「ナマ企画書」まとめ【第2弾】


当然ですが、それぞれに特徴があっていい。
普段は、自分や同じグループの人間が作ったものしか見ないので、
視野が広がり、今後の製作物の参考になる。

「この企画の目的は」や「○○をターゲットに」といった
決まり文句を使うことになれてしまうと、
同じような企画書しか作れなくなってしまう。
また、内輪で作って、内輪で見せ合ってばかりだと、
別の人が作っても、しばらくするとパターン化されてしまう。
だいたいは上司やリーダーの好みというか、意見に従う形というか、
流れが同じようなものが出来上がってしまう。

色々な企画書を見ることで、おっと感心することもあれば、
ここはイケてない書き方だなーと思うところもある。

自分がそのサービスの企画書を作成するのならば、
どのようなものになるのか、考えてみるのも勉強になりそうだ。

2013年12月16日月曜日

米大統領が「プログラミングしてみよう!」

この前、オバマ大統領が、
「プログラミングしてみよう!」
的なことを発信していた。

・Youtube
http://www.youtube.com/watch?v=6XvmhE1J9PY

これは、Computer Science Education Weekに参加する若い人向けに話した内容で、
プログラミングやコンピュータサイエンスの重要性を伝えようとしている。

アメリカでは大統領がこのような発言をするが、
日本では、国のトップがこのようなことを話すようになるまでには
まだまだ時間が必要なのだろう。

2013年8月13日火曜日

android:複数のアクティビティでデータを共有する

アクティビティ間でのデータのやりとりはインテントを使用すれば可能であるが、
そのデータが共有データである場合、画面が遷移するたびにインテントにput、getを繰り返すのは効率がよくない。
そこで、すべてのアクティビティから参照できるよう、Singletonでデータを保持する方法を試してみる。

  • 共有するデータクラスをSingletonで作成する。
  • 共有データが必要なアクティビティにて、getInstanceでインスタンスを取得する。

共有するデータクラスは以下のように定義する。
package net.kuttya.singletonsample;

public class SampleData {
    // 共有するデータクラス

    // 自クラスをインスタンスに持つ
    private static SampleData data = null;

    // メンバ変数
    private int dataInt = 0;

    public static SampleData getInstance() {
        if(data == null) {
            // インスタンスがnullの場合のみ生成する
            data = new SampleData();
        }
        return data;
    }

    private SampleData() {
        // コンストラクタをprivateにすることで、
        // インスタンスを新しく生成できなくする
    }

    public void setDataInt(int i) {
        this.dataInt = i;
    }

    public int getDataInt() {
        return this.dataInt;
    }
}

呼び出し元のアクティビティでは、次の動作を実装する。
package net.kuttya.singletonsample;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;

public class MainActivity extends Activity {

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

        Button b = (Button)findViewById(R.id.button1);
        b.setOnClickListener(new OnClickListener() {

            public void onClick(View v) {
                // 共有するデータを設定
                SampleData data = SampleData.getInstance();
                data.setDataInt(12345);

                // インテント生成
                Intent intent = new Intent(Intent.ACTION_VIEW);
                intent.setClassName("net.kuttya.singletonsample", "net.kuttya.singletonsample.CallActivity");

                // アクティビティ呼び出し
                startActivity(intent);
            }
        });
    }
}

呼び出し先のアクティビティでは、次の動作を実装する。
package net.kuttya.singletonsample;

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

public class CallActivity extends Activity {

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

        // 共有データを取得
        SampleData data = SampleData.getInstance();

        // 表示
        TextView textView1;
        textView1 = (TextView) findViewById(R.id.textView1);
        textView1.setText(Integer.toString(data.getDataInt()));
    }
}

ただし、データの共有がうまくいかないケースもある。
例えば、アクティビティ単位でプロセスを分けた場合、
Singletonであったとしても、それぞれのプロセスでインスタンスが生成されてしまう可能性がある。
他にも、共有するデータにキャッシュ情報が含まれていたりすると、
キャッシュがなにかのきっかけでクリアされてしまった場合、
共有データないのキャッシュ情報には参照先が存在しなくなる。

保持するデータには注意が必要である。

にほんブログ村 IT技術ブログ Androidアプリ開発へ

2013年8月12日月曜日

android:Parcelableインタフェースを使用したアクティビティ間のデータ受け渡し

androidでは、プロセス間通信が頻繁に行われる。
特に、アクティビティ間でデータをやりとりする場合にはインテントを利用する。
インテントでインスタンスをやりとりする場合は、一工夫必要である。

インスタンスはintなどの型とはことなり、なんでもインテントにputすることはできない。
インテントにインスタンスを受け渡すためには、Parcelableインタフェースを実装する必要がある。

Parcelableインタフェースを利用するには、以下の手順が必要となる。
  • インテントに受け渡すクラスにParcelableを実装する。
  • インテントに受け渡すクラスでCREATORメンバ変数を定義する。
  • 受け渡し元のアクティビティにおいて、インテントにはParcelableを実装したクラスのインスタンスをputする。
  • 受け渡し先のアクティビティにおいて、同様にインテントにはParcelableを実装したクラスのインスタンスにgetする。
受け渡しするデータクラスは以下のように定義する。
package net.kuttya.parcelablesample;

import android.os.Parcel;
import android.os.Parcelable;

public class SampleData implements Parcelable {
    // インテントで受け渡しするデータクラス

    public String sampleString;
    public int sampleInt;
    public int[] sampleIntArray;

    public int describeContents() {
     // ファイルディスクリプタを返す(Parcelableインタフェースの実装メソッド)
        // ファイルディスクリプタを持たない場合は0を返す
        return 0;
    }

    public void writeToParcel(Parcel dest, int flags) {
        // シリアライズ処理(Parcelableインタフェースの実装メソッド)

        dest.writeString(sampleString);
        dest.writeInt(sampleInt);
        dest.writeIntArray(sampleIntArray);
    }

    public static final Parcelable.Creator CREATOR =
            new Parcelable.Creator() {

                public SampleData createFromParcel(Parcel source) {
                    // デシリアライズ処理
                    // writeToParcelメソッドでシリアライズした順番と同じ順番でメンバ変数をデシリアライズすること

                    // インスタンス生成
                    SampleData data = new SampleData();

                    // インスタンスに受け取るデータを格納する
                    data.sampleString = source.readString();
                    data.sampleInt = source.readInt();

                    // 配列の場合は、引数に格納先を指定する
                    data.sampleIntArray = new int[data.sampleInt];
                    source.readIntArray(data.sampleIntArray);

                    return data;
                }

                public SampleData[] newArray(int size) {
                    // 配列を受け渡す処理

                    return new SampleData[size];
                }
            };
}

受け渡し元のアクティビティでは、次の動作を実装する。
package net.kuttya.parcelablesample;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;

public class MainActivity extends Activity {

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

        Button b = (Button)findViewById(R.id.button1);
        b.setOnClickListener(new OnClickListener() {

            public void onClick(View v) {
                // シリアライズするインスタンスを生成
                SampleData data = new SampleData();
                data.sampleString = "abcde";
                data.sampleInt = 3;
                data.sampleIntArray = new int[data.sampleInt];
                data.sampleIntArray[0] = 123;
                data.sampleIntArray[1] = 456;
                data.sampleIntArray[2] = 789;

                // インテント生成
                Intent intent = new Intent(Intent.ACTION_VIEW);
                intent.setClassName("net.kuttya.parcelablesample", "net.kuttya.parcelablesample.CallActivity");

                // データをシリアライズ
                intent.putExtra("data", data);

                // アクティビティ呼び出し
                startActivity(intent);
            }
        });
    }
}

受け渡し先のアクティビティでは、次の動作を実装する。
package net.kuttya.parcelablesample;

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.widget.TextView;

public class CallActivity extends Activity {

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

        // インテント取得
        Intent intent = getIntent();

        //データをデシリアライズして受け取る
        SampleData data = intent.getParcelableExtra("data");

        // 表示
        TextView textView1,textView2,textView3;
        textView1 = (TextView) findViewById(R.id.textView1);
        textView2 = (TextView) findViewById(R.id.textView2);
        textView3 = (TextView) findViewById(R.id.textView3);
        textView1.setText(data.sampleString);
        textView2.setText(Integer.toString(data.sampleInt));
        textView3.setText(Integer.toString(data.sampleIntArray[0]));
    }
}


にほんブログ村 IT技術ブログ Androidアプリ開発へ

2012年12月31日月曜日

android:プリファレンスでデータを保存する

プリファレンスを使うことによって、android端末内にデータを保存できる。

プリミティブな型でデータを保存することは少ないかと思うが、
  • 保存できる型は、
     boolean
     float
     int
     long
     String
  • データの格納先は「/data/data/パッケージ名/shared_prefs/ファイル名.xml」
  • SharedPreferencesのインスタンスはgetSharedPreferencesで取得する。
  • getSharedPreferencesメソッドの第1引数は、ファイル名。
     下記のサンプルの場合、「sample.xml」がファイル名となる。
  • getSharedPreferencesメソッドの第2引数は、プリファレンスへのアクセスモード。
     MODE_PRIVATE:他のアプリケーションからはアクセスできない。
     MODE_WORLD_READABLE:他のアプリケーションからの読み込みを許可する。
     MODE_WORLD_WRITEABLE:他のアプリケーションからの書き込みを許可する。
  • プリファレンスへの読み書きはSharedPreferences.Editorを使う。

package net.kuttya.sharedpreferencessample;

import android.os.Bundle;
import android.app.Activity;
import android.content.Context;
import android.content.SharedPreferences;
import android.view.Menu;

public class MainActivity extends Activity {

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

        // 初期化
        SharedPreferences preferences = getSharedPreferences("sample", Context.MODE_PRIVATE);
        SharedPreferences.Editor editor = preferences.edit();

        // 書き込み
        // 第1引数はキー、第2引数は書き込む値
        editor.putBoolean("boolkey", true);
        editor.putFloat("floatkey", 3.5f);
        editor.putInt("intkey", 7);
        editor.putLong("longkey", 9);
        editor.putString("Stringkey", "文字列");
        editor.commit();

        // 読み込み
        // 第1引数はキー、第2引数はデフォルト値
        boolean b = preferences.getBoolean("boolkey", false);
        float f = preferences.getFloat("floatkey", 1.1f);
        int i = preferences.getInt("intkey", 1);
        long l = preferences.getLong("longkey", 5);
        String str = preferences.getString("Stringkey", "");
    }
}


にほんブログ村 IT技術ブログ Androidアプリ開発へ

2012年12月30日日曜日

android:JSONを使う(Twitter-APIの検索結果をパースする)

JSON形式のデータをandroidで扱う。

JSON形式のデータは、WebAPIから取得することにする。
WebAPIは何でもいいのだが、今回はTwitter-APIを使うことにする。

次のサイトにAPIの使い方が書かれているため、参考にしてほしい。
https://dev.twitter.com/docs/api/1/get/search

  • Uri.Builderで、WebAPIのリクエストURLを作成する。
  • HttpURLConnectionで、WebAPIに対してリクエストを投げる。
  • WebAPIから受け取ったレスポンス(JSON形式データ)を、BufferedInputStream、ByteArrayOutputStreamを使用して文字列データに変換する。
  • JSONObject、JSONArrayを使用して、JSONデータをパースする。

package net.kuttya.jsonsample;

import java.io.BufferedInputStream;
import java.io.ByteArrayOutputStream;
import java.net.HttpURLConnection;
import java.net.URL;

import org.json.JSONArray;
import org.json.JSONObject;

import android.net.Uri;
import android.os.Bundle;
import android.os.StrictMode;
import android.app.Activity;
import android.view.View;
import android.widget.EditText;
import android.widget.TextView;

public class JsonSampleActivity extends Activity {
    private EditText editSearchWord;
    private TextView viewSearchResult;

    @Override
    public void onCreate(Bundle savedInstanceState) {

        // おまじない
        StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder().permitAll().build());

        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        editSearchWord = (EditText)findViewById(R.id.edittext_searchWord);
        viewSearchResult = (TextView)findViewById(R.id.textview_serchResult);
    }

    // 検索ボタン押下処理
    public void onClickSearch(View view) {

        // 検索ワードのチェック
        String searchWord = editSearchWord.getText().toString();
        if(searchWord.length() < 1) {
            viewSearchResult.setText("検索する文字列を指定してください。");
            return;
        }

        // Twitter-APIのURLを生成
        Uri.Builder uriBuilder = new Uri.Builder();
        uriBuilder.scheme("http");
        uriBuilder.authority("search.twitter.com");
        uriBuilder.path("search.json");
        uriBuilder.appendQueryParameter("lang", "ja");
        uriBuilder.appendQueryParameter("rpp", "20");
        uriBuilder.appendQueryParameter("q", searchWord);
        String uriStr = uriBuilder.toString();

        try {
            // コネクション生成
            HttpURLConnection connection = null;
            URL url = new URL(uriStr);
            connection = (HttpURLConnection)url.openConnection();
            connection.setRequestMethod("GET");

            // リクエスト送信
            connection.connect();

            // レスポンスコードチェック
            if(connection.getResponseCode() != 200) {
                viewSearchResult.setText("検索失敗!!");
                return;
            }

            // レスポンス文字列取得
            BufferedInputStream inputStream = new BufferedInputStream(connection.getInputStream());
            ByteArrayOutputStream responseArray = new ByteArrayOutputStream();
            byte[] buff = new byte[1024];

            int length;
            while((length = inputStream.read(buff)) != -1) {
                if(length > 0) {
                    responseArray.write(buff, 0, length);
                }
            }

            // JSONをパース
            StringBuilder viewStrBuilder = new StringBuilder();
            JSONObject jsonObj = new JSONObject(new String(responseArray.toByteArray()));
            JSONArray result = jsonObj.getJSONArray("results");
            for(int i = 0; i < result.length(); i++) {
                JSONObject tweet = result.getJSONObject(i);
                viewStrBuilder.append(tweet.getString("created_at") + "\n");        // つぶやき日時
                viewStrBuilder.append(tweet.getString("from_user_name") + "\n");    // 投稿者
                viewStrBuilder.append(tweet.getString("text") + "\n");                // 投稿内容
                viewStrBuilder.append("----------\n");
            }

            // 表示
            viewSearchResult.setText(viewStrBuilder.toString());
        } catch(Exception e) {
            viewSearchResult.setText(e.getMessage());
        }
    }
}


マニフェストファイルにインターネットを使う旨を登録する。
    

検索結果を取得したときのキャプチャ。






















にほんブログ村 IT技術ブログ Androidアプリ開発へ