13 марта 2014 г.

Задачи и обратный стек (Tasks and back stack). Часть 2

Для понимания предыдущей моей статьи на практике я модифицировал свое мега приложение AP003 так, чтобы в Активностях отображалось количество активностей в текущей задаче, а так же то была ли активность запущена в первые или же она уже была запущена (возвращена из обратного стека).

В написании кода помогли следующие статьи (кои стоит прочитать):

http://startandroid.ru/ru/uroki/vse-uroki-spiskom/62-urok-25-task-chto-eto-takoe-i-kak-formiruetsja.html

http://startandroid.ru/ru/uroki/vse-uroki-spiskom/190-urok-116-povedenie-activity-v-task-intent-flagi-launchmode-affinity

http://stackoverflow.com/questions/9839592/getting-a-list-of-running-processes-and-killing-a-specific-process

http://stackoverflow.com/questions/7992563/how-to-find-back-stack-activities-in-an-android-application

и статейка о работе оператора if в java

http://developer.alexanderklimov.ru/android/java/if.php

Приложение все тоже. В нем есть три Активности – ActivityA, ActivityB и ActivityC. На каждой Активности есть кнопка вызывающая следующую активность. A->B->C->A и так далее. Сейчас добавил счетчик количества активностей в задаче а так же вывод информации о том была ли уже запущена Активность или нет.

Итак сперва запустим приложение, чтобы было понятно о чем речь, потом рассмотрим код.
Запустили и видим что Активность нам сообщает, (1) что этот экземпляр Активности запущен в первый раз. А так же видно количество Активностей в текущей задаче (2). У нас пока одна Активность, так как кнопку Start Activity B мы не нажимали.

T0001

Теперь посмотрим логи

T0002
Здесь все стандартно. Все как на классических диаграммах.

Далее жмем кнопку Home (3). Наше приложение уходит в фоновый режим. Смотрим логи

T0003

Приложение перешло в остановленное состояние.

Теперь заново откроем его тапнув по его иконке в списке приложений

T0004

Теперь Активность сообщает что она была уже запущена и просто восстановлена из памяти.

T0005

Из логов видно, что события onCreate для Активности не было, и она не создавалась заново, а была просто возобновлена ее работа методом onRestart.

Теперь нажмем кнопку Start Activity B

T0006

Мы видим что Активность В была запущена впервые и что в нашей задаче уже две активности А и В.

Посмотрим логи

T0007

Мы видим что после нажатия на кнопку Start Activity B, сперва Активность А перешла в приостановленное состояние onPause и только потом создалась и отобразилась Активность В. И уже после этого Активность А перешла в остановленное состояние onStop. Это один из важных моментов для понимания.

Теперь нажмем кнопку Обратно (Back), чтобы вернуться к Активности А и посмотрим на нее и логи

T0008

Сейчас Активность А сообщает, что она уже была запущена прежде, а Активностей в нашей задаче снова одна, так как Активность В была уничтожена при нажатии в ней кнопки обратно, что подтверждают и логи

T0009

Теперь снова запустим Activity B, а из нее Activity C

T0010

Комментировать особо не буду, тут и так все понятно по аналогии с предыдущим. Просто приведу еще логи выполнения.

T0011

Сейчас у нас Активности А и В находятся в остановленном состоянии, но они существуют. Количество Активностей в нашей задаче равно 3 – А, В и С.

Теперь запустим из Активности С Активность А. Это будет НОВЫЙ экземпляр Актиновости А, другая Активность А, с которой мы начинали сейчас находится на дне стека.

T0012

Собственно скрин это и показывает. Теперь в нашей задаче 4 активности А, В, С и снова А. Теперь если нажать кнопку HOME. И снова вернуться к приложению, то мы увидим, что новая Активность А уже была запущена.

T0013

И логи

T0014

Видно что Активность А была приостановлена, затем остановлена и затем возобновила свое состояние.

Теперь если мы начнем последовательно нажимать кнопку ОБРАТНО, то очистим стек
Активностей и выйдем из приложения.

T0015

По существу этот материал как раз продемонстрировал на практике, то что было изложено в этой моей статье.

Теперь перейдем к коду этого приложения.

T0016

В приложении три Активности и три файла разметки. Я приведу код ActivityA.java и layout_a.xml, а так же код AndroidManifest.xml. Файлы остальных Активностей и разметок аналогичны.

И так ActivityA.java

package com.example.ap0003;

import java.util.List;

import android.app.Activity;
import android.app.ActivityManager;
import android.app.ActivityManager.RunningTaskInfo;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log;
import android.view.View;
import android.widget.TextView;

public class ActivityA extends Activity {

 final String TAG = "States";
 TextView tvTextLife;
 List<ActivityManager.RunningTaskInfo> list;
 ActivityManager am;
 Integer TotalActCount;
 Boolean FirstStart;
 Boolean NextAct;

 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.layout_a);
  Log.d(TAG, "ActivityA: onCreate()");
  // флаг что активность запущена впервые
  FirstStart = true;
  // кнопка запуска следующей Активности не нажималась
  NextAct = false;
 }

 @Override
 protected void onStart() {
  super.onStart();
  Log.d(TAG, "ActivityA: onStart()");
 }

 @Override
 protected void onResume() {
  super.onResume();
  Log.d(TAG, "ActivityA: onResume()");
  // получаем список 10 последних задач
  am = (ActivityManager) getSystemService(ACTIVITY_SERVICE);
  list = am.getRunningTasks(10);
  // перебираем список задач и выбираем свою по имени пакета
  // com.example.ap0003
  for (RunningTaskInfo task : list) {
   if (task.baseActivity.flattenToShortString().startsWith(
     "com.example.ap0003")) {
    // находим поле для вывода информации о количестве запущенных
    // Активностей
    tvTextLife = (TextView) findViewById(R.id.textActCountA);
    TotalActCount = task.numActivities;
    // коррекция счетчика для кнопки ОБРАТНО
    if (NextAct == true & FirstStart == false)
     TotalActCount = TotalActCount - 1;
    // выводим количество Активностей в задаче
    tvTextLife.setText("Activites in task " + TotalActCount);
    // коррекция счетчика для кнопки ДОМОЙ
    NextAct = false;
   }
  }
 }

 @Override
 protected void onPause() {
  super.onPause();
  Log.d(TAG, "ActivityA: oPause()");
  // флаг что активность уже была запущена
  FirstStart = false;
  // находим текстовое поле по его идентификатору
  tvTextLife = (TextView) findViewById(R.id.textStateActA);
  // присваиваем значение атрибуту Text для выбранного TextView
  tvTextLife.setText("Этот экземпляр ActivityA уже был запущен!");
 }

 @Override
 protected void onStop() {
  super.onStop();
  Log.d(TAG, "ActivityA: onStop()");
 }

 @Override
 protected void onRestart() {
  super.onRestart();
  Log.d(TAG, "ActivityA: onRestart()");
 }

 @Override
 protected void onDestroy() {
  super.onDestroy();
  Log.d(TAG, "ActivityA: onDestroy()");
 }

 public void onClickStartB(View v) {
  Intent intent = new Intent(ActivityA.this, ActivityB.class);
  startActivity(intent);
  // кнопка запуска следующей Активности была нажата
  NextAct = true;

 }

}

Обратите внимание на выделенные строки. Это я задал флаги состояния и их проверку, чтобы правильно скорректировать значение счетчика Активностей в задаче. Если этого не сделать, то поскольку, например, при нажатии кнопки ОБРАТНО восстанавливается предыдущая Активность, а та с которой был переход еще не уничтожена, то значение счетчика будет показывать на одну активность больше чем есть на самом деле. Тоже самое и с кнопкой Домой.

Теперь код layout_a.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:background="#f20808"
    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=".ActivityA" >

    <TextView
        android:id="@+id/textView1"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:text="@string/strA" />

    <Button
        android:id="@+id/buttonStartB"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_below="@+id/textView1"
        android:onClick="onClickStartB"
        android:text="Start Activity B" />

    <TextView
        android:id="@+id/textStateActA"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignLeft="@+id/buttonStartB"
        android:layout_below="@+id/buttonStartB"
        android:background="#fffcfc"
        android:text="Этот экземпляр АсtivityA запущен в первый раз" />

    <TextView
        android:id="@+id/textActCountA"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignLeft="@+id/textStateActA"
        android:layout_below="@+id/textStateActA"
        android:layout_marginTop="15dp"
        android:background="#fffcfc"
        android:text="TextView" />

</RelativeLayout>

Выделенная строка 24 показывает вызов метода из ActivityA по нажатию кнопки.

И теперь код AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.example.ap0003"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="19" />
    <uses-permission android:name="android.permission.GET_TASKS"/>

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="com.example.ap0003.ActivityA"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
        <activity
            android:name="com.example.ap0003.ActivityB"
            android:label="@string/app_name" >
        </activity>
        <activity
            android:name="com.example.ap0003.ActivityC"
            android:label="@string/app_name" >
        </activity>
    </application>

</manifest>

Строка 10 добавляет разрешение для получения информации о задачах приложением. Если эту сроку не добавить, то приложение будет завершаться с ошибкой, так как у него не будет полномочий на это.

На этом пока закончим эту часть.

Комментариев нет:

Отправить комментарий