NUmeroのソース置き場

見習いプログラマー

タグ:開発

ViewPager+TabLayoutについてメモ。

実装

※コンパイラにcom.android.support:designのパスを通しておいてください。
1から作るよりNew ProjectのTabbed Activityをいじるほうが早いです。

MainActivity.java
import android.support.design.widget.TabLayout;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;

import android.support.v4.view.ViewPager;
import android.os.Bundle;

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        SectionsPagerAdapter sectionsPagerAdapter = new SectionsPagerAdapter(getSupportFragmentManager());

        ViewPager viewPager = (ViewPager) findViewById(R.id.container);
        viewPager.setAdapter(sectionsPagerAdapter);
        TabLayout tabLayout = (TabLayout) findViewById(R.id.tabs);
        tabLayout.setupWithViewPager(viewPager);
    }
}


activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<android.support.design.widget.CoordinatorLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/main_content"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:fitsSystemWindows="true">

    <android.support.design.widget.AppBarLayout
        android:id="@+id/appbar"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:paddingTop="@dimen/appbar_padding_top"
        android:theme="@style/AppTheme.AppBarOverlay">

        <android.support.v7.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_width="match_parent"
            android:layout_height="?attr/actionBarSize"
            android:background="?attr/colorPrimary"
            app:layout_scrollFlags="scroll|enterAlways"
            app:popupTheme="@style/AppTheme.PopupOverlay"/>

        <android.support.design.widget.TabLayout
            android:id="@+id/tabs"
            android:layout_width="match_parent"
            android:layout_height="wrap_content" />

    </android.support.design.widget.AppBarLayout>

    <android.support.v4.view.ViewPager
        android:id="@+id/container"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_behavior="@string/appbar_scrolling_view_behavior" />

</android.support.design.widget.CoordinatorLayout>



SectionsPagerAdapter.java
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentManager;
import android.support.v4.app.FragmentPagerAdapter;

public class SectionsPagerAdapter  extends FragmentPagerAdapter {

    public SectionsPagerAdapter(FragmentManager fm) {
        super(fm);
    }

    @Override
    public Fragment getItem(int position) {
        return new Section();
    }

    @Override
    public int getCount() {
        return 10;
    }

    @Override
    public CharSequence getPageTitle(int position) {
        return "SECTION";
    }
}

ここでは10個のFragmentのページを生成しています。

Fragmentの部分です。
Section.java
import android.os.Bundle;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;

public class Section extends Fragment {

    @Override
    public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) {
        View view = inflater.inflate(R.layout.fragment_section1, container, false);
        return view;
    }

}


fragment_section.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    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">

    <TextView
        android:id="@+id/section_label"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="Section"/>

</RelativeLayout>





おまけ


xml内のandroid.support.design.widget.TabLayoutの箇所に
・app:tabMode
・app:tabGravity
を指定することでTabの並び方を変えることができます。

先日公開したメモアプリにはAdmob広告を表示しています。
そこで起きた問題と解決法をメモ。

発生した問題


公開用のapkしようとgradle内のminifyEnabledの値をtrueにしたところエラーが発生。
その値をfalseにした場合はapkを生成することができました。

原因


Proguard使用時にエラーが出たため、Proguardによる難読化、圧縮によるものだと考えられる。

解決法


proguard-rules.proに以下を追加
-keep class com.google.android.gms.** {
    *;
}
-dontwarn com.google.android.gms.**

公式の方法だと解決しなかったためこの方法で一応解決(しているはず)。
この状態でapkを生成するとサイズが小さくなっています。(個人的にもう少し小さくなって欲しいところ…)

余談


com.google.android.gms:play-servicesのライブラリにパスを通すとき、7以降のバージョンだと連絡先等のパーミッションが自動で追加されるのをなんとかしたい…
現状の解決法としては自分の環境ではバージョンを6で使っています。

先日マンガ管理のアップデートを公開しました。
アプリの中では画像検索の際に毎回画像をダウンロードすると、通信量が多くなるのと、時間がかかるため、本体のキャッシュ領域に一時保存しています。
そのキャッシュの部分のメモです。

キャッシュの部分


DiscCache.java
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;

public class DiscCache {
    private File dir;
    private Bitmap bitmap;

    DiscCache(Context context){
        dir = new File(context.getCacheDir().getPath());
        if(!dir.exists()){
            dir.mkdir();
        }
    }

    public void save(String url, Bitmap bitmap){
        try {
            FileOutputStream os = new FileOutputStream(new File(createFilePath(dir, getKey(url))));
            bitmap.compress(Bitmap.CompressFormat.PNG, 100, os);
        }catch (FileNotFoundException e){
        }
    }

    public Bitmap getCache(String url){
        if(new File(createFilePath(dir,getKey(url))).exists()){
            bitmap = BitmapFactory.decodeFile(createFilePath(dir, getKey(url)));
            return bitmap;
        }else{
            return null;
        }
    }

    public void clear(){
        clearFiles(dir.listFiles());
    }

    private void clearFiles(File[] files){
        for (File file : files) {
            if (file.isFile()) {
                file.delete();
            } else if (file.isDirectory()) {
                clearFiles(file.listFiles());
            }
        }
    }

    private String getKey(String str){
        return String.valueOf(Math.abs(str.hashCode()));
    }

    private String createFilePath(File dir, String key){
        return dir.getAbsolutePath() + File.separator + key;
    }
}


URLそのままでファイル名を付けずに変換し、ここではPNG形式で保存しています。

使用例


Downloader.java
import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.os.AsyncTask;
import android.view.View;
import android.widget.ImageView;

import java.io.IOException;
import java.io.InputStream;
import java.net.URL;

public class Downloader extends AsyncTask {
    private String imageURL;
    private Context context;
    private ImageView imageView;

    Downloader(String imageURL,Context context, ImageView imageView){
        this.imageURL = imageURL;
        this.context = context;
        this.imageView = imageView;
    }

    @Override
    protected Bitmap doInBackground(Void... params) {
        DiscCache cache = new DiscCache(context);
        Bitmap bitmap = cache.getCache(imageURL);
        if(bitmap == null) {
            try {
                bitmap = getImage();
                cache.save(imageURL, bitmap);
            } catch (IOException e) {
                bitmap = BitmapFactory.decodeResource(context.getResources(), R.mipmap.ic_launcher);
            }
        }
        return bitmap;
    }

    @Override
    protected void onPostExecute(Bitmap result) {
        imageView.setImageBitmap(result);
    }

    private Bitmap getImage() throws IOException{
        URL url = new URL(imageURL);
        InputStream is = url.openConnection().getInputStream();

        BitmapFactory.Options options = new BitmapFactory.Options();
        options.inPreferredConfig = Bitmap.Config.ARGB_4444;
        options.inPurgeable = true;

        Bitmap bitmap = BitmapFactory.decodeStream(is, null, options);
        is.close();
        return bitmap;
    }
}


コンストラクタにいろいろ詰め込んでいるのは気にしないでください。
以前に保存したキャッシュがあればそちらのBitmapを、なければダウンロードしキャッシュに保存しています。
あまりややこしくないと思います。

キャッシュの削除は.clear()を呼べばキャッシュ以下のファイルはすべて削除されます。

PreferenceFragment+ToolBar+AppcompatActivityというもしかしたらものすごく意味わからない実装をしたのでメモ。

・準備


レイアウトファイルでToolBarを実装するためにThemeのToolBarを非表示にする。
styles.xml
<resources>
    <!-- Base application theme. -->
    <style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">
        <!-- Customize your theme here. -->
    </style>
</resources>



・実装

Fragmentの部分
SettingFragment.java
package numero.com.jp.myapplication;

import android.os.Bundle;
import android.preference.PreferenceFragment;

public class SettingFragment extends PreferenceFragment {
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        addPreferencesFromResource(R.xml.settings);
    }
}


Activityのレイアウトファイル
setting_main.xml
<?xml version="1.0" encoding="UTF-8"?>
<RelativeLayout
    android:layout_height="match_parent"
    android:layout_width="match_parent"
    xmlns:android="http://schemas.android.com/apk/res/android">
    <android.support.v7.widget.Toolbar
        android:id="@+id/toolbar"
        android:layout_height="wrap_content"
        android:layout_width="match_parent"
        android:elevation="4dp"
        android:minHeight="?attr/actionBarSize"
        android:background="?attr/colorPrimary" />
    <FrameLayout
        android:id="@+id/frame"
        android:layout_height="wrap_content"
        android:layout_width="match_parent"
        android:layout_below="@id/toolbar"/>
</RelativeLayout>


Activityの部分
SettingActivity.java
package numero.com.jp.myapplication;

import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.Toolbar;

public class SettingActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.setting_main);
        setSupportActionBar((Toolbar) findViewById(R.id.toolbar));

        getFragmentManager().beginTransaction().replace(R.id.frame, new SettingFragment()).commit();
    }
}


設定画面のレイアウトファイル
xml/settings
<?xml version="1.0" encoding="utf-8"?>
<PreferenceScreen xmlns:android="http://schemas.android.com/apk/res/android">
    <PreferenceCategory
        android:title="全般">
        <CheckBoxPreference
            android:key="value1"
            android:title="チェックボックス"
            android:summary="(説明)"
            android:defaultValue="false"/>
    </PreferenceCategory>
</PreferenceScreen>


設定画面のレイアウト等に関しては公式を参考にしてもらえればと思います。
image

OnClickListenerの実装について、様々な書き方があるのでまとめてみます。

レイアウトは共通のこちらを使います。
activity_main.xml
<RelativeLayout
    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:gravity="center"
    tools:context=".MainActivity">
    <Button
        android:id="@+id/button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="BUTTON"/>
</RelativeLayout>


Activityにインターフェイスを実装


MainActivity.java
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

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

        Button button = (Button)findViewById(R.id.button);
        button.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()){
            case R.id.button:
                Toast.makeText(getApplicationContext(),"buttonが押されました。",Toast.LENGTH_SHORT).show();
                break;
        }
    }
}


無名クラスに実装


MainActivity.java
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {

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

        View.OnClickListener listener = new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(getApplicationContext(),"buttonが押されました。",Toast.LENGTH_SHORT).show();
            }
        };

        Button button = (Button)findViewById(R.id.button);
        button.setOnClickListener(listener);
    }
}


直接setOnClickListenerに実装する場合は…
MainActivity.java
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {

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

        Button button = (Button)findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Toast.makeText(getApplicationContext(),"buttonが押されました。",Toast.LENGTH_SHORT).show();
            }
        });
    }
}


別クラスに実装


MainActivity.java
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {

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

        Button button = (Button)findViewById(R.id.button);
        button.setOnClickListener(new CustomListener());
    }

    public class CustomListener implements View.OnClickListener{
        @Override
        public void onClick(View v) {
            Toast.makeText(getApplicationContext(),"buttonが押されました。",Toast.LENGTH_SHORT).show();
        }
    }
}


レイアウトファイルのonClick属性を用いる


レイアウトファイルのButtonを次のようにします。
<Button
        android:id="@+id/button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="BUTTON"
        android:onClick="click"/>

MainActivity
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity {

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

    public void button_click(View v){
        Toast.makeText(getApplicationContext(),"buttonが押されました。",Toast.LENGTH_SHORT).show();
    }
}


このほかにもAndroidAnnotationやButter Knifeを用いた書き方もあります。
結論を言うと可読性と好みの問題であるのでどれを利用すべきというのはありませんが、どの方法でも読めるようにしておいて損はないと思います。

Android 5.0(Lollipop)から追加されたActionBarの代替となるコンポーネントで、以前のスタイルからではなくレイアウトに書きます。

・準備


themeのToolBarを非表示にする。
styles.xml
<resources>

    <!-- Base application theme. -->
    <style name="AppTheme" parent="Theme.AppCompat.Light.NoActionBar">
        <!-- Customize your theme here. -->
    </style>

</resources>



・実装


activity_main.xml
<RelativeLayout
    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"
    tools:context=".MainActivity">
    <android.support.v7.widget.Toolbar
        android:id="@+id/toolbar"
        android:layout_height="wrap_content"
        android:layout_width="match_parent"
        android:minHeight="?attr/actionBarSize"
        android:background="?attr/colorPrimary"
        android:elevation="4dp" />
    <!-- contents -->
</RelativeLayout>


xml内でViewのパーツとして扱います。

MainActivity.java
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.support.v7.widget.Toolbar;
import android.view.Menu;
import android.view.MenuItem;

public class MainActivity extends AppCompatActivity {

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

        Toolbar toolbar = (Toolbar)findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);

        //アイコンの設定
        getSupportActionBar().setIcon(R.mipmap.ic_launcher);
    }

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

    @Override
    public boolean onOptionsItemSelected(MenuItem item) {
        int id = item.getItemId();
        if (id == R.id.action_settings) {
            //メニューのクリック処理
            return true;
        }

        return super.onOptionsItemSelected(item);
    }

}

image



※他のサイトではActionBarActivityを継承しているのですが、非推奨になっているのでAppCompatActivityのほうがいいのかな…?


AndroidでFloatingActionButtonを実装してみようと思います。

image



・準備


FABはDesign Libraryに含まれているのでライブラリをgradleに追記します。
※ここではv22.2.1のライブラリを使用しています。

build.gradle(Module: app)
dependencies {
    compile fileTree(include: ['*.jar'], dir: 'libs')
    compile 'com.android.support:appcompat-v7:22.2.1'
    compile 'com.android.support:design:22.2.1'
}


・実装


activity_main.xml
<RelativeLayout
    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"
    tools:context=".MainActivity">
    <android.support.design.widget.FloatingActionButton
        android:layout_height="wrap_content"
        android:layout_width="wrap_content"
        android:id="@+id/fab"
        android:layout_gravity="right"
        android:layout_marginBottom="16dp"
        android:layout_marginRight="16dp"
        android:layout_alignParentRight="true"
        android:layout_alignParentBottom="true" />
    <!-- contents -->
</RelativeLayout>

FAB内部の画像はdrawable以下に画像を入れるなどして
android:src=""
で指定します。

MainActivity.java
import android.support.design.widget.FloatingActionButton;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Toast;

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

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

        FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
        fab.setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        switch (v.getId()){
            case R.id.fab:
                Toast.makeText(getApplicationContext(),"FABが押されました",Toast.LENGTH_SHORT).show();
                break;
        }
    }
}


image




FABの実装は基本的にボタンと同じようにするだけなので難しくはありません。

このページのトップヘ