hander_replace_fragment

ActivActivity から 動的にFragmentを呼び出した状態 から、Fragmentを入れ替える。

https://github.com/pppiroto/KaigiUtil/tree/c043c2b7ba37827ac39cc8420781b8a17b234802

入れ替えるのに、指定時間ウェイトする。

指定時間ウェイトするには、Handler.postDelayed() を使って、指定時間後にRunnableを実行するように設定。Runnableの処理は、Handlerがアタッチされたスレッド(UIスレッド)で実行されるため、Runnableの中で、呼び出し元のメソッドを呼び出すことができる。

Fragmentは、FragmentTransaction.replace() を用いてコンテナに追加されているFragmentを置き換える。

package info.typea.kaigiutil.contents.dummycall;

import android.os.Bundle;
import android.os.Handler;
import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.support.v4.app.FragmentManager;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.Toast;

import info.typea.kaigiutil.R;

public class DummyCallingFragment extends Fragment implements View.OnClickListener {
    private static final String ARG_MSG = "message";

    public DummyCallingFragment() {
    }

    public static DummyCallingFragment newInstance(String message) {
        DummyCallingFragment fragment = new DummyCallingFragment();
        Bundle args = new Bundle();
        args.putString(ARG_MSG, message);
        fragment.setArguments(args);
        return fragment;
    }

    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        super.onCreateView(inflater, container, savedInstanceState);
        View view = inflater.inflate(R.layout.fragment_dummy_calling, container, false);

        view.findViewById(R.id.btn_dummy_calling).setOnClickListener(this);

        return view;
    }

    private void dummyCallback() {
        Toast.makeText(this.getContext(),
                "Call back", Toast.LENGTH_LONG).show();

        FragmentManager fragmentManager =  this.getActivity().getSupportFragmentManager();
        fragmentManager.beginTransaction()
                .replace(R.id.content_fragment_container,
                        CallingScreenFragment.newInstance("test message"))
                .addToBackStack(null)
                .commit();
    }

    @Override
    public void onClick(View v) {
        switch(v.getId()) {
            case R.id.btn_dummy_calling:

                Handler hander = new Handler();
                hander.postDelayed(new Runnable() {
                    @Override
                    public void run() {
                        dummyCallback();
                    }
                }, 10000);

                break;
            default:
                break;
        }
    }
}

Android Notification(通知) サンプル

notification

開始したサービスから、通知を行い、通知をクリックしたら Activity を表示する。

https://github.com/pppiroto/KaigiUtil/tree/73bfa6c6a624207552066129b157353b159f2ca5

1.通知アイコン

通知アイコンを準備。

AndroidStudio から、res/drawable のコンテキストメニューから、New-Image Asset を選択、Icon Type に、Notification icons を指定

notification_icon

2.Notificationの作成

https://developer.android.com/guide/topics/ui/notifiers/notifications.html?hl=ja

package info.typea.kaigiutil;

import android.app.NotificationManager;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.os.IBinder;
import android.os.Vibrator;
import android.support.v4.app.NotificationCompat;
import android.support.v4.app.TaskStackBuilder;
import android.widget.Toast;

public class SleepDefenderService extends Service {
    private Vibrator vibrator;

    public SleepDefenderService() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        throw null;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        vibrator = (Vibrator)this.getSystemService(Context.VIBRATOR_SERVICE);
        if (!vibrator.hasVibrator()) {
            Toast.makeText(this, "no vibrator.", Toast.LENGTH_LONG).show();
        } else {
            Toast.makeText(this, "start vibe as service", Toast.LENGTH_SHORT).show();
            long pattern[] = {1000, 100};
            int repeatIndex = 0; // 繰り返し開始位置 -1の場合繰り返しなし
            vibrator.vibrate(pattern, repeatIndex);
        }

        doNnotify();

        return START_STICKY;
    }

    /**
     * notifications
     */
    private void doNnotify(){
        // 通知を作成
        NotificationCompat.Builder builder = new NotificationCompat.Builder(this)
                .setSmallIcon(R.drawable.ic_notification)
                .setContentTitle("Kaigi Util")
                .setContentText("Stop sleep defender for click here.")
                .setTicker("Sleep defender started.");

        // Activity を起動
        Intent resultIntent = new Intent(this, ContentActivity.class);
        TaskStackBuilder stackBuilder = TaskStackBuilder.create(this);
        stackBuilder.addParentStack(ContentActivity.class);
        stackBuilder.addNextIntent(resultIntent);
        PendingIntent pendingIntent =
                stackBuilder.getPendingIntent(
                        0,
                        PendingIntent.FLAG_UPDATE_CURRENT
                );
        builder.setContentIntent(pendingIntent);

        int notificationId = 1;

        NotificationManager manager = (NotificationManager)getSystemService(NOTIFICATION_SERVICE);

        manager.notify(notificationId, builder.build());
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        vibrator.cancel();
        Toast.makeText(this, "stop vibe as service", Toast.LENGTH_SHORT).show();
    }
}

Android Service サンプル(Vibrator)

android_vibrate


FragmentからVibratorを実行したが、Activityを終了すると、バイブレーションも終了してしまうため、サービスに処理を移行し動き続けるようにする。

https://github.com/pppiroto/KaigiUtil/tree/8d52606b8d69f2dd689888a186dda50ae530bdff

1.サービス

Fragmentでやっていた処理を、サービスに持ってくる

package info.typea.kaigiutil;

import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.os.IBinder;
import android.os.Vibrator;
import android.widget.Toast;

public class SleepDefenderService extends Service {
    private Vibrator vibrator;

    public SleepDefenderService() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        throw null;
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        vibrator = (Vibrator)this.getSystemService(Context.VIBRATOR_SERVICE);
        if (!vibrator.hasVibrator()) {
            Toast.makeText(this, "no vibrator.", Toast.LENGTH_LONG).show();
        } else {
            Toast.makeText(this, "start vibe as service", Toast.LENGTH_SHORT).show();
            long pattern[] = {1000, 100};
            int repeatIndex = 0; // 繰り返し開始位置 -1の場合繰り返しなし
            vibrator.vibrate(pattern, repeatIndex);
        }
        return START_STICKY;
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        vibrator.cancel();
        Toast.makeText(this, "stop vibe as service", Toast.LENGTH_SHORT).show();
    }
}

1.1 AndroidManifest.xml の application エントリーの子要素に、以下を追加

        <service
            android:name=".SleepDefenderService"
            android:enabled="true"
            android:exported="true"></service>

2.Fragment

Vibratorを操作していた、Fragmentを、サービスの呼び出しと停止に変更

package info.typea.kaigiutil;

import android.content.Intent;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.support.v4.app.Fragment;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;


public class SleepDefenderFragment extends Fragment implements View.OnClickListener {
    private static final String ARG_MSG = "message";

    public SleepDefenderFragment() {
    }

    public static SleepDefenderFragment newInstance(String message) {
        SleepDefenderFragment fragment = new SleepDefenderFragment();
        Bundle args = new Bundle();
        args.putString(ARG_MSG, message);
        fragment.setArguments(args);
        return fragment;
    }

    @Override
    public View onCreateView(LayoutInflater inflater, @Nullable ViewGroup container, @Nullable Bundle savedInstanceState) {
        super.onCreateView(inflater, container, savedInstanceState);
        return inflater.inflate(R.layout.fragment_sleep_defender, container, false);
    }

    @Override
    public void onViewCreated(View view, @Nullable Bundle savedInstanceState) {
        super.onViewCreated(view, savedInstanceState);

        view.findViewById(R.id.btn_vib_start).setOnClickListener(this);
        view.findViewById(R.id.btn_vib_stop).setOnClickListener(this);
    }

    @Override
    public void onClick(View v) {
        Intent intent = new Intent(this.getContext(), SleepDefenderService.class);

        switch(v.getId()) {
            case R.id.btn_vib_start:
                this.getContext().startService(intent);
                break;

            case R.id.btn_vib_stop:
                this.getContext().stopService(intent);
                break;
        }
    }
}