Android – RecyclerView

Gradleに追加する

compile 'com.android.support:recyclerview-v7:25.1.0'

RecyclerViewを設置する

<android.support.v7.widget.RecyclerView
    android:id="@+id/rview"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_alignParentTop="true"
    android:layout_alignParentStart="true" />

1行用xmlつくる

row.xml

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

    <TextView
        android:id="@+id/item_name"
        android:layout_width="match_parent"
        android:layout_height="match_parent"/>
</LinearLayout>

Adapterをつくる

RecyclerViewのアダプタは継承するやつが決まっているっぽい。RecyclerView.Adapterというのがある。下記3つの関数を実装する必要がある。

import android.support.v7.widget.RecyclerView;
import android.view.ViewGroup;

public class RecyclerAdatper extends RecyclerView.Adapter{
    @Override
    public RecyclerView.ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        return null;
    }

    @Override
    public void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {

    }

    @Override
    public int getItemCount() {
        return 0;
    }
}

onCreateViewHolderは、ViewHolderのインスタンスをつくって返す。RecyclerView.ViewHolderを継承したViewHolderクラスである必要があって、インスタンス作成時に一行用viewを渡す。
onBindViewHolderは、holderとポジションがわたってくるので、それを使って該当ポジションデータをholderにセットする。
getItemCountは表示する数。表示可能なデータよりも多い数を指定するとエラーになる。

public class RecyclerAdapter extends RecyclerView.Adapter<RecyclerAdapter.RViewHolder>{
    private List<String> itemList;

    public RecyclerAdapter(List<String> list) {
        itemList = list;
    }

    public static class RViewHolder extends RecyclerView.ViewHolder {
        public TextView nameView;
        public RViewHolder(View v) {
            super(v);
            nameView = (TextView) itemView.findViewById(R.id.item_name);
        }
    }

    @Override
    public RViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        ViewGroup view = (ViewGroup) LayoutInflater.from(parent.getContext()).inflate(R.layout.row, parent, false);
        return new RViewHolder(view);
    }

    @Override
    public void onBindViewHolder(RViewHolder holder, int position) {
        holder.nameView.setText(itemList.get(position));
    }

    @Override
    public int getItemCount() {
        return itemList.size();
    }
}

MainActivityをつくる

MainActivity.java

public class MainActivity extends AppCompatActivity {
    private ArrayList<String> dataList;

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

        dataList = new ArrayList();
        final RecyclerView recyclerView = (RecyclerView) findViewById(R.id.rview);
        recyclerView.setHasFixedSize(true);
        recyclerView.setLayoutManager(new LinearLayoutManager(this));
        recyclerView.setAdapter(new RecyclerAdapter(dataList));
        createData();
    }

    private void createData() {
        for (int i = 0; i < 30 ; i++ ) {
            dataList.add("item" + i);
        }
    }
}

recyclerView.setHasFixedSize(true);は、リストのレイアウトサイズが固定の場合、setHasFixedSize()をtrueにするとパフォーマンスが向上するらしい。

参考:
RecyclerView
RecyclerView の基本的な使い方
[Android] RecyclerView:リストを表示する

Android – ViewHolder

ListViewの効率化をViewHolderでします。
adapter使うときに、Viewが行ごとに変わりますが、かといって毎回findViewByIdやると効率が悪いので、viewHolderを使うといいらしいです。viewHolderは、各行用のviewの要素を覚えておいてくれるやつでっす。

ViewHolder使わない場合のコード

public class MyAdapter extends ArrayAdapter<Food> {
    private final LayoutInflater _inflater;

    public MyAdapter(Context context, List<Food> objects) {
        super(context, 0, objects);
        _inflater = LayoutInflater.from(context);
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        if(convertView == null) {
            convertView = _inflater.inflate(R.layout.row, parent, false);
        }
        Food food = super.getItem(position);
        ((TextView)convertView.findViewById(R.id.text_name)).setText(food.name);
        ((TextView)convertView.findViewById(R.id.text_price)).setText(String.valueOf(food.price));
        return convertView;
    }
}

ViewHolder使った場合のコード

public class MyAdapter extends ArrayAdapter<Food> {
    private final LayoutInflater _inflater;

    public MyAdapter(Context context, List<Food> objects) {
        super(context, 0, objects);
        _inflater = LayoutInflater.from(context);
    }

    private static class ViewHolder {
        TextView name;
        TextView price;
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        View cv = convertView;
        ViewHolder vh = null;
        if(cv == null) {
            cv = _inflater.inflate(R.layout.row, parent, false);
            vh = new ViewHolder();
            vh.name = (TextView)cv.findViewById(R.id.text_name);
            vh.price = (TextView)cv.findViewById(R.id.text_price);
            cv.setTag(vh);
        } else {
            vh = (ViewHolder)cv.getTag();
        }
        Food food = super.getItem(position);
        vh.name.setText(food.name);
        vh.price.setText(String.valueOf(food.price));
        return cv;
    }
}

Android – ListView

すごくシンプルなListViewの実装

adapterにセットする、個別行用のレイアウトファイルが必要ですが、ANDROIDがデフォルトで用意している、android.R.layout.simple_list_item_1を使っている。ただTextViewが1つ入っているだけのもの。

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/activity_main"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.logicky.listview.MainActivity">

    <ListView
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_alignParentTop="true"
        android:layout_centerHorizontal="true"
        android:id="@+id/listview" />
</RelativeLayout>

MainActivity.java

public class MainActivity extends AppCompatActivity {

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

        ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1);
        adapter.add("Apple");
        adapter.add("Orange");
        adapter.add("Coffee");
        ListView view = (ListView)findViewById(R.id.listview);
        view.setAdapter(adapter);
    }
}

android.R.layout.simple_list_item_1

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@android:id/text1"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:textAppearance="?android:attr/textAppearanceListItemSmall"
    android:gravity="center_vertical"
    android:paddingStart="?android:attr/listPreferredItemPaddingStart"
    android:paddingEnd="?android:attr/listPreferredItemPaddingEnd"
    android:minHeight="?android:attr/listPreferredItemHeightSmall" />

Adapterのインスタンスをつくる。つくるときに、個別行用のレイアウトファイルも指定する。
Adapterにデータを追加する。
ListViewを取得して、アダプタをセットする。

行データを配列で与える場合

public class MainActivity extends AppCompatActivity {

    private ListView view;
    private static final String[] foods = {"Apple", "Orange", "Coffee"};

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ArrayAdapter<String> adapter = new ArrayAdapter<String>(this, android.R.layout.simple_list_item_1, foods);
        view = (ListView)findViewById(R.id.listview);
        view.setAdapter(adapter);
    }
}

自作アダプタ

Foodクラスのデータをリスト表示するListViewを作る。表示項目は、Food名と値段。

ArrayAdapterを継承したMyAdapterクラスを作成し、getViewでFood名と値段を取得・セットする。
個別行データ用のレイアウトは、row.xmlに作成する。row.xmlは単純なテキストビュー2個だけのもの。

MyAdapter.java

public class MyAdapter extends ArrayAdapter<Food> {
    private final LayoutInflater _inflater;

    public MyAdapter(Context context, List<Food> objects) {
        super(context, 0, objects);
        _inflater = LayoutInflater.from(context);
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        if(convertView == null) {
            convertView = _inflater.inflate(R.layout.row, parent, false);
        }
        Food food = super.getItem(position);
        ((TextView)convertView.findViewById(R.id.text_name)).setText(food.name);
        ((TextView)convertView.findViewById(R.id.text_price)).setText(String.valueOf(food.price));
        return convertView;
    }
}

Food.java

public class Food {
    public String name;
    public int price;
}

row.xml

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

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

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:id="@+id/text_price" />
</LinearLayout>

MainActivity.java

public class MainActivity extends AppCompatActivity {
    private ListView view;
    private List<Food> foods;
    private MyAdapter adapter;

    private void setFoods() {
        Food food = new Food();
        food.name = "Apple";
        food.price = 120;
        foods.add(food);

        food = new Food();
        food.name = "Orange";
        food.price = 150;
        foods.add(food);
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        foods = new ArrayList<Food>();
        adapter = new MyAdapter(this, foods);
        view = (ListView)findViewById(R.id.listview);
        view.setAdapter(adapter);
        setFoods();
    }
}

参考:
【Android】ListViewを使うための基礎知識(1)
リストビューをカスタマイズする

Android – ProgressBar

回転するやつと横棒があるっぽい。

下記4つがある。

  • progressBar(Large)
  • ProgressBar
  • ProgressBar(Small)
  • ProgressBar(Horizontal)

styleを変えると色々なデザインにできる。Android5以上だと、デフォルトで、Widget.Material.Lightになるっぽい。かっこいい。
表示非表示切替は、visibilityで設定できる。visibilityがないと表示になる。visible、invisible、goneがある。goneは非表示で表示領域を確保しない。

Horizontalは表示割合の数値を設定できる。

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

    ProgressBar progressBar = (ProgressBar) findViewById(R.id.progress_hori);
    progressBar.setMax(100);
    progressBar.setProgress(30);
    progressBar.setSecondaryProgress(70);
}

Android – Firebase Auth ログアウト

FirebaseAuth auth = FirebaseAuth.getInstance();
auth.signOut();

public class MainActivity extends AppCompatActivity{
    private static final String TAG = "MainActivity";
    private FirebaseAuth auth;
    private FirebaseUser user;
    private String name;
    private String photo;

    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        auth = FirebaseAuth.getInstance();
        user = auth.getCurrentUser();

        if (user == null) {
            startActivity(new Intent(this, SignInActivity.class));
            finish();
            return;
        } else {
            name = user.getDisplayName();
            photo = user.getPhotoUrl().toString();
        }
    }

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

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        switch(item.getItemId()){
            case R.id.sign_out_menu:
                Log.d(TAG, "Sign Out");
                auth.signOut();
                startActivity(new Intent(this, SignInActivity.class));
                finish();
            default:
                return super.onOptionsItemSelected(item);
        }
    }
}

Android – オプションメニュー

オプションメニューを配置

res/menu/main_menu.xml

<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto">
    <item
        android:id="@+id/sign_out_menu"
        android:title="@string/sign_out"
        app:showAsAction="never"/>
</menu>

MainActivity.java

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

メニューをクリックしたときの処理

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    switch(item.getItemId()){
        case R.id.sign_out_menu:
            Log.d(TAG, "Settings Selected.");
            break;
    }
    return true;
}

Android – Firebase Auth ログインチェック

FirebaseAuthのインスタンスを取得して、getCurrentUser()でログイン中ユーザを取得できる。nullだったら未ログイン状態。

public class MainActivity extends AppCompatActivity{
    private FirebaseAuth auth;
    private FirebaseUser user;
    private String name;
    private String photo;

    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        auth = FirebaseAuth.getInstance();
        user = auth.getCurrentUser();

        if (user == null) {
            Log.d("HOGE", "未ログイン");
        } else {
            Log.d("HOGE", "ログイン中");
            name = user.getDisplayName();
            photo = user.getPhotoUrl().toString();
            Log.d("HOGE", name);
            Log.d("HOGE", photo);
        }
    }

}

Android – SharedPreference

参考:
基本的なSharedPreferencesの使い方
SharedPreferencesの使い方(基礎編)

サンプルコード

public class MainActivity extends AppCompatActivity{
    private SharedPreferences data;
    private SharedPreferences.Editor editor;
    private Button btn_up, btn_show;
    private TextView txt;
    private Integer cnt;

    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        data = getSharedPreferences("DataSave", Context.MODE_PRIVATE);
        editor = data.edit();
        cnt = getCount();
        txt = (TextView)findViewById(R.id.txt_count);
        showCount();

        btn_up = (Button)findViewById(R.id.btn_up);
        btn_up.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                cnt += 1;
                saveCount();
            }
        });

        btn_show = (Button)findViewById(R.id.btn_show);
        btn_show.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                showCount();
            }
        });
    }

    private void saveCount() {
        editor.putInt("count", cnt);
        editor.apply();
    }

    private Integer getCount() {
        return data.getInt("count", 0);
    }

    private void showCount() {
        txt.setText(String.valueOf(cnt));
    }

}

Android – Firebaseでログイン機能実装

Firebaseを使ってログイン機能を作ってみます。
参考:Firebase Authentication

サンプルアプリがあります。サンプルアプリは、パッケージ名を変更するのが結構大変だったので、アプリ登録でサンプルアプリと同名のパッケージ名で登録すると簡単に利用できました。
firebase/quickstart-android

Gradleの設定

参考:Android プロジェクトに Firebase を追加する

buildscriptの dependenciesに、

classpath 'com.google.gms:google-services:3.0.0'

appのdependenciesに、

compile 'com.google.firebase:firebase-core:10.0.1'

appの一番下に、

apply plugin: 'com.google.gms.google-services'

エラーがでた。

Error:Execution failed for task ':app:processDebugGoogleServices'.
> Missing api_key/current_key object

参考:Missing api_key/current key with Google Services 3.0.0
google-service.jsonにapi_keyがないぞってことかな?たしかに空になっている。もう一度作成したらapi_keyが入っていた。最初のいらないじゃん。。エラーでなくなった。

コンソールでログイン方法有効化

とりあえず今回はgoogleログインをしてみるので、googleを有効にします。Facebookとかは、Facebookアプリのシークレットとか入れる必要がありますが、googleは何もなく有効になりました。

Googleログイン

参考:Android で Google ログインを使用して認証する

下記もgradleに追加する必要がある。

compile 'com.google.firebase:firebase-auth:10.0.1'
compile 'com.google.android.gms:play-services-auth:10.0.1'

下記コードは、完全にGoogleのサンプルと同じですが、とりあえずこれを実行したらログイン機能が実装されていた。これだと、Firebaseのユーザテーブルに登録されちゃうので、自分でデータベース作ってる場合は、使えないけど、お手軽ではあります。自分のデータベースと連携させることもできるらしいので、後で調べる。

activity_main.xml

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:id="@+id/main_layout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@color/grey_100"
    android:orientation="vertical"
    android:weightSum="4">

    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="3"
        android:gravity="center_horizontal"
        android:orientation="vertical">

        <TextView
            android:id="@+id/title_text"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginBottom="@dimen/title_bottom_margin"
            android:text="@string/google_title_text"
            android:theme="@style/ThemeOverlay.MyTitleText" />

        <TextView
            android:id="@+id/status"
            style="@style/ThemeOverlay.MyTextDetail"
            android:text="@string/signed_out" />

        <TextView
            android:id="@+id/detail"
            style="@style/ThemeOverlay.MyTextDetail"
            tools:text="Firebase User ID: 123456789abc" />

    </LinearLayout>


    <RelativeLayout
        android:layout_width="fill_parent"
        android:layout_height="0dp"
        android:layout_weight="1"
        android:background="@color/grey_300">

        <com.google.android.gms.common.SignInButton
            android:id="@+id/sign_in_button"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:visibility="visible"
            tools:visibility="gone" />

        <LinearLayout
            android:id="@+id/sign_out_and_disconnect"
            android:layout_width="fill_parent"
            android:layout_height="wrap_content"
            android:layout_centerInParent="true"
            android:orientation="horizontal"
            android:paddingLeft="16dp"
            android:paddingRight="16dp"
            android:visibility="gone"
            tools:visibility="visible">

            <Button
                android:id="@+id/sign_out_button"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_weight="1"
                android:text="@string/sign_out"
                android:theme="@style/ThemeOverlay.MyDarkButton" />

            <Button
                android:id="@+id/disconnect_button"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_weight="1"
                android:text="@string/disconnect"
                android:theme="@style/ThemeOverlay.MyDarkButton" />
        </LinearLayout>

    </RelativeLayout>

</LinearLayout>

MainActivity.java

import android.content.Intent;
import android.os.Bundle;
import android.support.annotation.NonNull;
import android.util.Log;
import android.view.View;
import android.widget.TextView;
import android.widget.Toast;

import com.google.android.gms.auth.api.Auth;
import com.google.android.gms.auth.api.signin.GoogleSignInAccount;
import com.google.android.gms.auth.api.signin.GoogleSignInOptions;
import com.google.android.gms.auth.api.signin.GoogleSignInResult;
import com.google.android.gms.common.ConnectionResult;
import com.google.android.gms.common.api.GoogleApiClient;
import com.google.android.gms.common.api.ResultCallback;
import com.google.android.gms.common.api.Status;
import com.google.android.gms.tasks.OnCompleteListener;
import com.google.android.gms.tasks.Task;
import com.google.firebase.auth.AuthCredential;
import com.google.firebase.auth.AuthResult;
import com.google.firebase.auth.FirebaseAuth;
import com.google.firebase.auth.FirebaseUser;
import com.google.firebase.auth.GoogleAuthProvider;

public class MainActivity extends BaseActivity implements
        GoogleApiClient.OnConnectionFailedListener,
        View.OnClickListener {

    private static final String TAG = "GoogleActivity";
    private static final int RC_SIGN_IN = 9001;

    // [START declare_auth]
    private FirebaseAuth mAuth;
    // [END declare_auth]

    // [START declare_auth_listener]
    private FirebaseAuth.AuthStateListener mAuthListener;
    // [END declare_auth_listener]

    private GoogleApiClient mGoogleApiClient;
    private TextView mStatusTextView;
    private TextView mDetailTextView;

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

        // Views
        mStatusTextView = (TextView) findViewById(R.id.status);
        mDetailTextView = (TextView) findViewById(R.id.detail);

        // Button listeners
        findViewById(R.id.sign_in_button).setOnClickListener(this);
        findViewById(R.id.sign_out_button).setOnClickListener(this);
        findViewById(R.id.disconnect_button).setOnClickListener(this);

        // [START config_signin]
        // Configure Google Sign In
        GoogleSignInOptions gso = new GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
                .requestIdToken(getString(R.string.default_web_client_id))
                .requestEmail()
                .build();
        // [END config_signin]

        mGoogleApiClient = new GoogleApiClient.Builder(this)
                .enableAutoManage(this /* FragmentActivity */, this /* OnConnectionFailedListener */)
                .addApi(Auth.GOOGLE_SIGN_IN_API, gso)
                .build();

        // [START initialize_auth]
        mAuth = FirebaseAuth.getInstance();
        // [END initialize_auth]

        // [START auth_state_listener]
        mAuthListener = new FirebaseAuth.AuthStateListener() {
            @Override
            public void onAuthStateChanged(@NonNull FirebaseAuth firebaseAuth) {
                FirebaseUser user = firebaseAuth.getCurrentUser();
                if (user != null) {
                    // User is signed in
                    Log.d(TAG, "onAuthStateChanged:signed_in:" + user.getUid());
                } else {
                    // User is signed out
                    Log.d(TAG, "onAuthStateChanged:signed_out");
                }
                // [START_EXCLUDE]
                updateUI(user);
                // [END_EXCLUDE]
            }
        };
        // [END auth_state_listener]
    }

    // [START on_start_add_listener]
    @Override
    public void onStart() {
        super.onStart();
        mAuth.addAuthStateListener(mAuthListener);
    }
    // [END on_start_add_listener]

    // [START on_stop_remove_listener]
    @Override
    public void onStop() {
        super.onStop();
        if (mAuthListener != null) {
            mAuth.removeAuthStateListener(mAuthListener);
        }
    }
    // [END on_stop_remove_listener]

    // [START onactivityresult]
    @Override
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
        super.onActivityResult(requestCode, resultCode, data);

        // Result returned from launching the Intent from GoogleSignInApi.getSignInIntent(...);
        if (requestCode == RC_SIGN_IN) {
            GoogleSignInResult result = Auth.GoogleSignInApi.getSignInResultFromIntent(data);
            if (result.isSuccess()) {
                // Google Sign In was successful, authenticate with Firebase
                GoogleSignInAccount account = result.getSignInAccount();
                firebaseAuthWithGoogle(account);
            } else {
                // Google Sign In failed, update UI appropriately
                // [START_EXCLUDE]
                updateUI(null);
                // [END_EXCLUDE]
            }
        }
    }
    // [END onactivityresult]

    // [START auth_with_google]
    private void firebaseAuthWithGoogle(GoogleSignInAccount acct) {
        Log.d(TAG, "firebaseAuthWithGoogle:" + acct.getId());
        // [START_EXCLUDE silent]
        showProgressDialog();
        // [END_EXCLUDE]

        AuthCredential credential = GoogleAuthProvider.getCredential(acct.getIdToken(), null);
        mAuth.signInWithCredential(credential)
                .addOnCompleteListener(this, new OnCompleteListener<AuthResult>() {
                    @Override
                    public void onComplete(@NonNull Task<AuthResult> task) {
                        Log.d(TAG, "signInWithCredential:onComplete:" + task.isSuccessful());

                        // If sign in fails, display a message to the user. If sign in succeeds
                        // the auth state listener will be notified and logic to handle the
                        // signed in user can be handled in the listener.
                        if (!task.isSuccessful()) {
                            Log.w(TAG, "signInWithCredential", task.getException());
                            Toast.makeText(MainActivity.this, "Authentication failed.",
                                    Toast.LENGTH_SHORT).show();
                        }
                        // [START_EXCLUDE]
                        hideProgressDialog();
                        // [END_EXCLUDE]
                    }
                });
    }
    // [END auth_with_google]

    // [START signin]
    private void signIn() {
        Intent signInIntent = Auth.GoogleSignInApi.getSignInIntent(mGoogleApiClient);
        startActivityForResult(signInIntent, RC_SIGN_IN);
    }
    // [END signin]

    private void signOut() {
        // Firebase sign out
        mAuth.signOut();

        // Google sign out
        Auth.GoogleSignInApi.signOut(mGoogleApiClient).setResultCallback(
                new ResultCallback<Status>() {
                    @Override
                    public void onResult(@NonNull Status status) {
                        updateUI(null);
                    }
                });
    }

    private void revokeAccess() {
        // Firebase sign out
        mAuth.signOut();

        // Google revoke access
        Auth.GoogleSignInApi.revokeAccess(mGoogleApiClient).setResultCallback(
                new ResultCallback<Status>() {
                    @Override
                    public void onResult(@NonNull Status status) {
                        updateUI(null);
                    }
                });
    }

    private void updateUI(FirebaseUser user) {
        hideProgressDialog();
        if (user != null) {
            mStatusTextView.setText(getString(R.string.google_status_fmt, user.getEmail()));
            mDetailTextView.setText(getString(R.string.firebase_status_fmt, user.getUid()));

            findViewById(R.id.sign_in_button).setVisibility(View.GONE);
            findViewById(R.id.sign_out_and_disconnect).setVisibility(View.VISIBLE);
        } else {
            mStatusTextView.setText(R.string.signed_out);
            mDetailTextView.setText(null);

            findViewById(R.id.sign_in_button).setVisibility(View.VISIBLE);
            findViewById(R.id.sign_out_and_disconnect).setVisibility(View.GONE);
        }
    }

    @Override
    public void onConnectionFailed(@NonNull ConnectionResult connectionResult) {
        // An unresolvable error has occurred and Google APIs (including Sign-In) will not
        // be available.
        Log.d(TAG, "onConnectionFailed:" + connectionResult);
        Toast.makeText(this, "Google Play Services error.", Toast.LENGTH_SHORT).show();
    }

    @Override
    public void onClick(View v) {
        int i = v.getId();
        if (i == R.id.sign_in_button) {
            signIn();
        } else if (i == R.id.sign_out_button) {
            signOut();
        } else if (i == R.id.disconnect_button) {
            revokeAccess();
        }
    }
}

Android – Retrofit

Retrofitも使ってみる。なんか一番人気がありそう。REST APIに対して使いやすい感じっぽい。

設定

build.gradleに下記を設定する。

compile 'com.squareup.retrofit2:retrofit:2.1.0'
compile 'com.squareup.retrofit2:converter-gson:2.1.0'

使い方

参考:
Android:Retrofit2.0ではてなAPIとおしゃべりしてみた
Retrofit2 &OkHttpでAndroidのHTTP通信が快適だにゃん

なんか思ってたのと違う。近代的だ。すっきりするからいい感じな気がする。でも色々ブラックボックス化してるから、エラーのときとかイライラする可能性はあるかもなあと思った。

コードサンプル

APIにアクセスすると、itemの配列がjsonデータでもらえるようになっている。itemは、id、name、price、descriptionというフィールドがある。
URLを、https://hoge.com/items/index.jsonとすると、返ってくるのは、[{id:1, name:たまご…}, {}]みたいなJSONデータ。

まずは、モデルクラスをつくる

モデルクラス

item.java

package com.logicky.json.data;

public class Item {
    public String name;
    public Integer price;
    public String description;
}

Interface

次に、APIのURL毎に対応する関数が羅列されているインターフェースをつくる

MyApiInterface.java

package com.logicky.json;

import com.logicky.json.data.Item;
import java.util.List;
import retrofit2.Call;
import retrofit2.http.GET;

public interface MyApiInterface {
    String END_POINT = "https://hoge.com";

    @GET("items/index.json")
    Call<List<Item>> getItems();
}

Activity

package com.logicky.json;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import com.logicky.json.data.Item;
import java.util.List;
import retrofit2.Call;
import retrofit2.Callback;
import retrofit2.Response;
import retrofit2.Retrofit;
import retrofit2.converter.gson.GsonConverterFactory;

public class MainActivity extends AppCompatActivity {
    private MyApiInterface myapi;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Retrofit retrofit = new Retrofit.Builder()
                .baseUrl(MyApiInterface.END_POINT)
                .addConverterFactory(GsonConverterFactory.create())
                .build();
        myapi = retrofit.create(MyApiInterface.class);
        getItems();
    }

    private void getItems() {
        Log.d("TEST", "getItems");
        Call<List<Item>> call = myapi.getItems();
        call.enqueue(new Callback<List<Item>>() {
            @Override
            public void onResponse(Call<List<Item>> call, Response<List<Item>> response) {
                Log.d("TEST", "getItems Response");
                List<Item> items = response.body();
                Log.d("TEST", items.get(0).name);
            }

            @Override
            public void onFailure(Call<List<Item>> call, Throwable t) {
                Log.d("TEST", "getItems Failure");
                Log.d("TEST", t.getMessage());
            }
        });

    }

}

Android – ライブラリの使い方

socketでインターネット接続すると、毎回AsyncTaskとか使わなくてはいけなくてめんどくさいので、ライブラリを使うのが一般的らしい。

参考:色んなAndroidアプリが使っているオープンソースライブラリを調べた

おすすめライブラリぽいやつ
Retrofit
Android Asynchronous Http Client
OkHttp

とりあえず、Android Asynchronous Http Clientというのを使ってみる。

ライブラリの読み込み方

参考:Android Studioでライブラリを取り込む3つの方法
外部のリポジトリからローカルに取り込む方法が一番いい気がする。

Android Asynchronous Http ClientのInstallation & Basic Usageというところに、設定方法が書いてあった。

build.gradleのdependeciesに下記を追加する。

dependencies {
  compile 'com.loopj.android:android-async-http:1.4.9'
}

Activityでインポートする。

import com.loopj.android.http.*;

使ってみる

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import com.loopj.android.http.*;
import cz.msebera.android.httpclient.Header;

public class MainActivity extends AppCompatActivity {
    private static final String url = "http://logicky.com";

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

    private void loadAsync() {
        AsyncHttpClient client = new AsyncHttpClient();
        client.get(url, new DataAsyncHttpResponseHandler() {
            @Override
            public void onSuccess(int statusCode, Header[] headers, byte[] responseBody) {
                String body = new String(responseBody);
                Log.d("TEST", "body=" + body);
            }

            @Override
            public void onFailure(int statusCode, Header[] headers, byte[] responseBody, Throwable error) {
                Log.d("TEST", "エラー");
            }
        });
    }
}

Android – socket

パーミッション設定

インターネット接続する場合、AndroidManifest.xmlにパーミッション設定を書く必要がある。

<uses-permission android:name="android.permission.INTERNET"/>

インターネット接続をActivityのonCreateとかで使うとエラーになる。UIスレッドで使ってはいけない。別のスレッドで使う。AsyncTaskのdoInBackgroundとかでやる。

http通信してみる

MainActivity.java

public class MainActivity extends AppCompatActivity {
    private MyTask task;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        task = new MyTask();
        task.execute();
    }
}

MyTask.java

public class MyTask extends AsyncTask<Integer, Integer, Integer> {
    static final String host = "google.com";
    static final String path = "/";

    @Override
    protected Integer doInBackground(Integer...x) {
        try {
            Socket socket = new Socket();
            socket.connect(new InetSocketAddress(host, 80));

            String request = "GET " + path + " HTTP/1.1\n" +
                    "Host: " + host + "\n" +
                    "\n\n";
            OutputStream out = socket.getOutputStream();
            out.write(request.getBytes());
            out.flush();

            InputStream in = socket.getInputStream();
            byte[] buffer = new byte[1024];
            int length;
            while((length = in.read(buffer)) != -1) {
                Log.d("TEST", new String(buffer, 0, length));
            }
            out.close();
            in.close();
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
        return 1;
    }
}

Android – AsyncTask

参考:AsyncTask
参考:AsyncTaskを使った非同期処理のきほん

AsyncTask<Params, Progress, Result>

と書いてある。doInBackgroundメソッドの引数の型, onProgressUpdateメソッドの引数の型, onPostExecuteメソッドの戻り値の型らしい。

AsyncTaskクラスを継承したクラスを作って、非同期処理を書く。
Activityで、上記タスクのインスタンスを作って、executeで非同期処理を実行させる。
execute時に、doInBackgroundの引数を渡せる。doInBackgroundの引数は何個でも渡せるようにする必要があるっぽい。

MyTask.java

public class MyTask extends AsyncTask<Integer, Integer, String> {
    private TextView txt;

    public MyTask(TextView txt) {
        super();
        this.txt = txt;
    }

    @Override
    protected String doInBackground(Integer...x) {
        try {
            Log.d("TEST", "Start Sleep");
            Thread.sleep(5000);
        } catch(Exception e) {
            Log.d("TEST", e.getMessage());
        }
        return "終わりました";
    }

    @Override
    protected void onPostExecute(String result) {
        super.onPostExecute(result);
        txt.setText(result);
    }
}

MainActivity.java

public class MainActivity extends AppCompatActivity {
    private MyTask task;

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

        task = new MyTask((TextView)findViewById(R.id.result));
        task.execute();

        TextView hello = (TextView)findViewById(R.id.hello);
        hello.setText("World");
    }
}

非同期処理中のプログレスバー表示

AsyncTaskは、4ステップある。あと、1回しか使えない。なので一回だけ非同期処理したいときに使うやつらしい。

  1. onPreExecute 実行前
  2. doInBackground 実行開始
  3. onProgressUpdate 実行中
  4. onPostExecute 実行後

onPreExecuteでプログレスバーを初期設定・表示させて、doInBackgroundで進捗状況を計算して、onProgressUpdateを都度実行させて、onProgressUpdateでプログレスバーを更新して、onPostExecuteでプログレスバー消す。

MyTask.java

public class MyTask extends AsyncTask<Integer, Integer, String> {
    private TextView txt;
    private ProgressDialog progress;
    private Activity ac;
    private final Integer wait = 5000;

    public MyTask(Activity a, TextView t) {
        super();
        ac = a;
        txt = t;
    }

    @Override
    protected void onPreExecute() {
        super.onPreExecute();
        progress = new ProgressDialog(ac);
        progress.setProgressStyle(ProgressDialog.STYLE_HORIZONTAL);
        progress.setIndeterminate(false);
        progress.setMax(wait);
        progress.setProgress(0);
        progress.show();
    }

    @Override
    protected String doInBackground(Integer...x) {
        try {
            for(int i = 0; i < wait; i += 200){
                Thread.sleep(200);
                publishProgress(i);
            }
        } catch(Exception e) {
            Log.d("TEST", e.getMessage());
        }
        return "終わりました";
    }

    @Override
    protected void onProgressUpdate(Integer... values) {
        super.onProgressUpdate(values);
        progress.setProgress(values[0]);
    }

    @Override
    protected void onPostExecute(String result) {
        super.onPostExecute(result);
        progress.dismiss();
        txt.setText(result);
    }
}

Windows10でJavaのkeytoolを使う

FirebaseでAndroidのアプリを登録するときに、デバッグ用の署名証明書を登録するように言われます。Javaのkeytoolというツールを使って作成することができます。使い方は、Javaのkeytool.exeが入ってるフォルダに移動して、コマンドプロンプトとかでkeytoolと打つと使えます。

(コマンド例)

$ cd '/c/Program Files/Java/jdk1.8.0_102/bin'
$ ./keytool
キーおよび証明書管理ツール

コマンド:

 -certreq            証明書リクエストを生成します
 -changealias        エントリの別名を変更します
 -delete             エントリを削除します
 -exportcert         証明書をエクスポートします
 -genkeypair         鍵ペアを生成します
 -genseckey          秘密鍵を生成します
 -gencert            証明書リクエストから証明書を生成します
 -importcert         証明書または証明書チェーンをインポートします
 -importpass         パスワードをインポートします
 -importkeystore     別のキーストアから1つまたはすべてのエントリをインポートします
 -keypasswd          エントリの鍵パスワードを変更します
 -list               キーストア内のエントリをリストします
 -printcert          証明書の内容を出力します
 -printcertreq       証明書リクエストの内容を出力します
 -printcrl           CRLファイルの内容を出力します
 -storepasswd        キーストアのストア・パスワードを変更します

command_nameの使用方法については"keytool -command_name -help"を使用してください

Androidのデバッグ証明書のフィンガープリントを取得する

$ ./keytool -exportcert -list -v -alias androiddebugkey -keystore /c/Users/hoge/.android/debug.keystore