Diferencia entre revisiones de «Usuario:ManuelRomero/Android/Intents»

De WikiEducator
Saltar a: navegación, buscar
(Devolver valores a una activity que me invocó)
 
(8 revisiones intermedias por el mismo usuario no mostrado)
Línea 28: Línea 28:
 
==Usando un Intent==
 
==Usando un Intent==
 
*Un intent necesita información para saber qué aplicación tiene que lanzar
 
*Un intent necesita información para saber qué aplicación tiene que lanzar
*Esta información la podemos pasar de diferente manera, ahora lo veremos
+
*Esta información la podemos pasar de diferente manera.
*Para es información, un Intent es una estructura de datos con la siguiente composición  
+
*Desde el punto de vista de la información, un Intent es una estructura de datos con la siguiente composición :
#'''''Component''''' Es el componente de aplicación que queremos que se ejecute  
+
#'''''component''''', es el componente de la aplicación que queremos que se ejecute.
#'''''action''''' Indica la acción asociada al Intent. con ella trataremos de encontrar los componetes que llevan a caba esa acción  
+
#'''''action''''', indica la acción asociada al Intent. con ella trataremos de encontrar los componetes que llevan a caba esa acción.
#'''''data''''' Son los datos necesarios para una determinada accion
+
#'''''data''''', son los datos necesarios para una determinada accion.
#'''''category''''' forma de agrupar actividades que detallan  conciones que  pueden hacer
+
#'''''category''''', forma de agrupar actividades que detallan  conciones que  pueden hacer
#'''''extra''''' Permite dar valores a un objeto Bundle en formato etiqueta-valor para poder pasar varles al componentes que invocquemos
+
#'''''extra''''', permite dar valores a un objeto Bundle en formato etiqueta-valor; Con ello podremos pasar valores al componentes que invoquemos.
#'''''type''''' Especificad e forma explicita el tipo de datos de intent.  
+
#'''''type''''', especifica de forma explícita el tipo de datos del intent.  
 
*Vamos a ver a continuación como se puede usar un Intent
 
*Vamos a ver a continuación como se puede usar un Intent
===Intent Explícito o implicito===
+
===Intent Explícito o Implícito===
 +
;Intent explícito
 
*Un intent es explícito cuando de forma explícita detallamos el componente que queremos lanzar.
 
*Un intent es explícito cuando de forma explícita detallamos el componente que queremos lanzar.
 
*Para ello detallaremos el componente.
 
*Para ello detallaremos el componente.
Línea 55: Línea 56:
 
  getComponent()
 
  getComponent()
 
*Ejemplo
 
*Ejemplo
<>
+
<source lang=java>
 +
...
 +
Intent i = new Intent();
 +
i.setClass(this,Actividad2.class);
 +
startActivity(i);
 +
...
 +
/*Otra manera de hacer lo mismo*/
 +
Intent i=new Intent();
 +
i.setComponent(new ComponentName(this,Actividad2.class));
 +
startActivity(i);
  
 
+
/*Lo mismo usando el nombre del java*/
*Uso extremadamente sencillo
+
...
*Realicemos un ejemplo sencillo
+
Intent i = new Intent();
*Idea: hacer un programa como se ve en la figura
+
i.setClassName(this, "com.example.interaccionactivitys.Actividad2");
*Cada pantalla será un activity.
+
startActivity(i);
*El click del activity 1 abrirá el activity 2
+
...
*El click del activity 2 cerrará el activity 2 y volverá al activity 1
+
[[Imagen:intent1.png|400px]]
+
===Empezamos===
+
*Creamos un proyecto nuevo llamado p.e. '''''usandoIntents'''''
+
*Clase '''''ActividadPrimera.java'''''
+
*Layout ''''''actividad_primera.xml'''''
+
**Por supuesto los nombres son ''opciones''
+
*Creamos una clase nueva llamada '''''ActividadSegunada.java'''''
+
[[Imagen:claseNueva.png]]
+
*Al tener dos activity debemos indicarlo en el fichero '''''Manifiest.xml'''''
+
*Añadimos la nueva clase activity
+
*Vemos a continuación la parte del manifest con las clases
+
<source lang=xml>
+
      <activity
+
            android:name=".ActividadPrimera"
+
            android:label="@string/title_activity_actividad_primera" >
+
            <intent-filter>
+
                <action android:name="android.intent.action.MAIN" />
+
                <category android:name="android.intent.category.LAUNCHER" />
+
            </intent-filter>
+
      </activity>
+
      <activity
+
            android:name=".ActividadSegunda"
+
            android:label="@string/title_activity_actividad_segunda" >
+
            <intent-filter>
+
                <action android:name="com.example.objetosintents.ActividadSegunda" />
+
                <category android:name="android.intent.category.DEFAULT" />
+
            </intent-filter>
+
      </activity>
+
 
</source>
 
</source>
*Nos debemos fijar en el elemento '''''intent-filter'''''
+
;Intent Implícito
*En este caso el elementos '''''action''''' tiene ''el nombre del filtro especificado al nombre  de la clase', a diferencia del ''activity principal' que contenía el valor '''''MAIN'''''
+
*En este caso creamos el intent pasando un nombre de accion.
**De este modo otras actividades que quieran llamar a ésta lo harán por este nombre
+
*El sistema buscará qué actividad puede realizar esa acción y la ejecutará.
*El elemento '''''category''''' para el filtro  contiene el valor '''''DEFAULT''''' el lugar del valor '''''LAUNCHER'''''
+
*En caso de haber más de una actividad  que pueda realizar esa acción nos mostrará la lista para que la realicemos.
*Esto indica que esta actividad será lanzada por defecto siendo invocada por otra mediante el método ''''''startActivity()'''''' no desde el lanzador de aplicaciones del dispositivos
+
*La forma de asociar a una Activity una acción es usando el elemento '''''<action-filter>''''' del fichero manifest
 +
*La accion la podemos pasar en el constructor del Intent.
 +
*Android tiene aplicaciones por defe
  
===Los layouts===
+
==Intents con parámetros==
*Creamos los dos layouts para cada activity
+
*Es muy  posible que queramos pasar/recuperar datos entre actividades
*Cada uno contendrá una etiqueta de texto o label y un botón (por las especificaciones primeras
+
*Por ejemplo que una activity me pase usuario y contraseña para validar
*Los layouts les hemos llamado en este caso '''''actividad_primera.xml''''' y '''''actividad_segunda.xml'''''
+
===Pasar valores a través de un Intent===
;actividad_primera.xml
+
 
 +
#Una vez creado el Intent, creamos un objeto Bundle
 +
#A este objeto le asignamos valores en la forma de parejas etiqueta-valor usando métodos del tipo '''''putxxx'''''
 +
#asociamos el objeto Bundle al Intent con el método '''''putExtras( Bundle )''''' y ya tenemos los valores preaparados para que sean pasados a la activity o componente que active el Intent.
 
<source lang=java>
 
<source lang=java>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
+
//Primero defino el intent y le asocia una actividad que quiero que invoque.
    xmlns:tools="http://schemas.android.com/tools"
+
Intent intencion = new Intent();
    android:layout_width="match_parent"
+
intencion.getClass(this,miclase.class);
    android:layout_height="match_parent" >
+
  
    <TextView
+
//Defino el bundle para asignar parejas variable-valor
        android:id="@+id/textView1"
+
Bundle datos = new Bundle();
        android:layout_width="match_parent"
+
datos.putInt("edad",25);
        android:layout_height="wrap_content"
+
datos.putString("nombre","pedro");
        android:text="@string/actividad1"
+
        tools:context=".ActividadPrimera" />
+
  
    <Button
+
//Asocio el bundle al intent
        android:id="@+id/button1"
+
intencion.putExtras(datos);
        android:layout_width="wrap_content"
+
        android:layout_height="wrap_content"
+
        android:layout_alignParentLeft="true"
+
        android:layout_below="@+id/textView1"
+
        android:layout_marginLeft="40dp"
+
        android:layout_marginTop="41dp"
+
        android:onClick="activity1"
+
        android:text="Activar activity 2" />
+
  
</RelativeLayout>
+
//Activo la activity
 +
startActivity(intencion);
 
</source>
 
</source>
;actividad_segunda.xml
+
 
 +
===Leer los datos que pasa el intent===
 +
*Para leer los datos en la activity invocada con el intent, se hace de manera sencilla
 +
#Creamos un objeto budle
 +
#Lo instanciamos con el bundle que nos retorna el método getExtras del intent que lo ha activado (con el método getIntent()
 +
#Lemos los valores con los métodos '''''getXXX(etiqueta)'''''
 +
 
 
<source lang=java>
 
<source lang=java>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
+
public class ClaseInvocadaConIntent  extends Activity {
    xmlns:tools="http://schemas.android.com/tools"
+
....
    android:layout_width="match_parent"
+
@Override
    android:layout_height="match_parent" >
+
protected void onCreate(Bundle estado){
    <TextView
+
        android:id="@+id/textView1"
+
Bundle valores = getIntent().getExtras();
        android:layout_width="wrap_content"
+
int edad = valores.getInt("edad");
        android:layout_height="wrap_content"
+
String nombre = valores.getString("nombre");
        android:text="@string/actividad2"
+
                ......
        tools:context=".ActividadSegunda" />
+
}
    <Button
+
....
        android:id="@+id/button1"
+
        android:layout_width="wrap_content"
+
        android:layout_height="wrap_content"
+
        android:layout_below="@+id/textView1"
+
        android:layout_centerHorizontal="true"
+
        android:layout_marginTop="58dp"
+
        android:onClick="activity2"
+
        android:text="Cerrar actividad 2 y volver a 1" />
+
</RelativeLayout>
+
</source>
+
*En el fichero '''''string.xml''''' tenemos las etiquetas usadas
+
<source lang=xml>
+
  <string name="title_activity_actividad_primera">ActividadPrimera</string>
+
  <string name="title_activity_actividad_segunda">ActividadSegunda</string>
+
  <string name="actividad1">Esta es la actividad primera y principal</string>
+
  <string name="actividad2">Esta es la actividad segunda</string>
+
 
</source>
 
</source>
===El fichero java===
+
 
*Ahora sólo queda lo que hacemos en el método del click
+
===Devolver valores a una activity que me invocó===
*En la actividad primera tenemos el método '''''activity1''''' (ver el fichero del layout de esta actividad)
+
*Ahora vamos a ver como es el proceso en el que la activity invocada retorna valores a la activity que la invocó
 +
[[Archivo:ActivityRetornaValores.png]]
 +
;Situación
 +
*La activity A invoca a la activity B
 +
*La activity B  retorna un valor a la activity A
 +
;Paso 1
 +
La activity A llama a la Activity B
 +
Para ello usamos el método
 
<source lang=java>
 
<source lang=java>
  public void activity1(View v){
+
  startActivityForResult(Intnet i, int codigo);
    startActivity (new Intent("com.example.objetosintents.ActividadSegunda"));
+
    }
+
 
</source>
 
</source>
*La forma de iniciar una actividad es con el método '''''startActivity()''''
+
*En este caso pasamos un segundo parámetro que es un código que indentifica la invocación de la acción
*como parámetro le pasamos una instancia de un objeto Intents con el nombre de la clase de la actividad como parámetro
+
*Este código  permitirá asegurarnos que retorno de esta activity y no de otra.
*Se puede plantear más seccionado pero es lo mismo
+
*Supongamos que una activity pueda invocar a varias.
<source lang=java>
+
**en el retorno de cualquiera de ellas si retorna valor, siempre iré al método '''''onActivityResult(int resquest, int result , Intendt)
  public void activity1(View v){
+
**Analizando el parámetro '''''request''''' podemos ver de qué intent venimos (que será el mismo valor que le hayamos pasado en la invocación.
    Intent claseAEjecutar = new Intent();
+
 
    claseAEjecutar.setAction("com.example.objetosintents.ActividadSegunda");
+
<source lang = java>
    startActivity (claseAEjecutar);
+
........
    }
+
public void clickActividad_B(View v){
</source>
+
Intent i = new Intent(this,ActividadB.class);
*Igualmente podemos especificar el nombre del la clase .class
+
String nombre=((EditText)findViewById(R.id.TNombre)).getText().toString();
<source lang=java>
+
String apellido=((EditText)findViewById(R.id.editText1)).getText().toString();
  public void activity1(View v){
+
Bundle datos = new Bundle();
    Intent claseAEjecutar = new Intent();
+
datos.putString("nombre",nombre);
    claseAEjecutar.setClass(this, ActividadSegunda.class);
+
datos.putString("apellido",apellido );
    startActivity (claseAEjecutar);
+
i.putExtras(datos);
    }
+
startActivityForResult(i,OK_ACTIVIDAD_2);
</source>
+
}
*Y lo mismo de manera más compacta
+
 
<source lang=java>
+
public void clickActividad_C(View v){
public void activity1(View v){
+
Intent i = new Intent(this,ActividadB.class);
    startActivity (new Intent(this,ActividadSegunda.class));
+
String nombre=((EditText)findViewById(R.id.TNombre)).getText().toString();
    }
+
String apellido=((EditText)findViewById(R.id.editText1)).getText().toString();
 +
Bundle datos = new Bundle();
 +
datos.putString("nombre",nombre);
 +
datos.putString("apellido",apellido );
 +
i.putExtras(datos);
 +
startActivityForResult(i,OK_ACTIVIDAD_3);
 +
}
 +
 
 +
@Override
 +
public void onActivityResult(int codAccion, int codDatos, Intent i){
 +
.........
 +
 +
if (codAccion==OK_ACTIVIDAD_2)
 +
Log.i("ActividadA","Vengo de actividad OK_2, Actividad B");
 +
if (codAccion==OK_ACTIVIDAD_3)
 +
Log.i("ActividadA","Vengo de actividad OK_3, Actividad C");
 +
 +
......
 +
 
 +
}
 +
}
 
</source>
 
</source>
  
=Intents con parámetros=
 
*Es muy  posible que queramos pasar/recuperar datos entre actividades
 
*Por ejemplo que una activity me pase usuario y contraseña para validar
 
 
===Devolviendo resultados===
 
===Devolviendo resultados===
 
*Ahora queremos hacer un diseño como el siguiente
 
*Ahora queremos hacer un diseño como el siguiente

Última revisión de 04:25 13 ene 2014

Road Works.svg Trabajo en proceso, espera cambios frecuentes. Tu ayuda y retroalimentación son bienvenidos.
Ver página de charlas.
Road Works.svg






Intents

ClaseIntent.png

Referencia
http://developer.android.com/reference/android/content/Intent.html
  • Definición de la página oficial


Icon define.gif

Definición

Un intent es una descripción abstracta de una operación que va a llevar a cabo


  • Los objetos de tipo android.content.Intent se utilizan para enviar mensajes asíncronos dentro de una aplicación o entre varias aplicaciones.
  • Estos mensajes permiten desde una aplicación lanzar o invocar la ejecución de un componente de aplicación
  1. Activity.
  2. Service.
  3. ContentProvider.
  4. Boradcast.
  • Así, los Intents permiten enviar o recibir información desde y hacia otros componetes de aplicación.
  • Esta información puede consistir simplemente en iniciar la ejecución de una aplicación en android, por ejemplo una actividad.


Intents para lanzar activity

  • Un Intent es un objeto que tiene una estructura de datos que contiene la información necesaria para lazar la ejecución de un componente.
  • Este componentes puede ser como ya hemos comentado una activity desarrollada en nuestra aplicación o bien también podemos ejecutar una aplicación disponible en android http://developer.android.com/guide/appendix/g-app-intents.html
  • Al Intent se le puede asociar la Activity que queremos que se lance, o bien a la Activity asociarle un nombre de accion para que el Intent tenga asociada dicha accion y al invocarlo se puda ejecutar la Activity que tenga asociada dicha acción
  • La forma de asociar a una Activity, una acción usado el elemento <intent-filter> en el fichero Manifest de nuestra activity

Usando un Intent

  • Un intent necesita información para saber qué aplicación tiene que lanzar
  • Esta información la podemos pasar de diferente manera.
  • Desde el punto de vista de la información, un Intent es una estructura de datos con la siguiente composición :
  1. component, es el componente de la aplicación que queremos que se ejecute.
  2. action, indica la acción asociada al Intent. con ella trataremos de encontrar los componetes que llevan a caba esa acción.
  3. data, son los datos necesarios para una determinada accion.
  4. category, forma de agrupar actividades que detallan conciones que pueden hacer
  5. extra, permite dar valores a un objeto Bundle en formato etiqueta-valor; Con ello podremos pasar valores al componentes que invoquemos.
  6. type, especifica de forma explícita el tipo de datos del intent.
  • Vamos a ver a continuación como se puede usar un Intent

Intent Explícito o Implícito

Intent explícito
  • Un intent es explícito cuando de forma explícita detallamos el componente que queremos lanzar.
  • Para ello detallaremos el componente.
  • Esto se puede hacer en el contructor o usando métodos explícitos de la clase
  • En el constructor
Intent(Context packageContext, Class<?> cls)
/*Actividad 2 es una actividad del paquete*/
...
    Intent i = new Intent(this, Actividad2.class);
    startActivity(i);
...
  • Usando métodos de la clase Intent para establecer el componente que queremos invocar
getComponent(...), getClass(...) getClassName(...).
  • Y para recuperar el nombre del componente
getComponent()
  • Ejemplo
...
	Intent i = new Intent();
	i.setClass(this,Actividad2.class);
	startActivity(i);
...
/*Otra manera de hacer lo mismo*/
	Intent i=new Intent();
	i.setComponent(new ComponentName(this,Actividad2.class));
	startActivity(i);
 
/*Lo mismo usando el nombre del java*/
...
	Intent i = new Intent();
	i.setClassName(this, "com.example.interaccionactivitys.Actividad2");
	startActivity(i);
...
Intent Implícito
  • En este caso creamos el intent pasando un nombre de accion.
  • El sistema buscará qué actividad puede realizar esa acción y la ejecutará.
  • En caso de haber más de una actividad que pueda realizar esa acción nos mostrará la lista para que la realicemos.
  • La forma de asociar a una Activity una acción es usando el elemento <action-filter> del fichero manifest
  • La accion la podemos pasar en el constructor del Intent.
  • Android tiene aplicaciones por defe

Intents con parámetros

  • Es muy posible que queramos pasar/recuperar datos entre actividades
  • Por ejemplo que una activity me pase usuario y contraseña para validar

Pasar valores a través de un Intent

  1. Una vez creado el Intent, creamos un objeto Bundle
  2. A este objeto le asignamos valores en la forma de parejas etiqueta-valor usando métodos del tipo putxxx
  3. asociamos el objeto Bundle al Intent con el método putExtras( Bundle ) y ya tenemos los valores preaparados para que sean pasados a la activity o componente que active el Intent.
//Primero defino el intent y le asocia una actividad que quiero que invoque.
 Intent intencion = new Intent();
 intencion.getClass(this,miclase.class);
 
//Defino el bundle para asignar parejas variable-valor
 Bundle datos = new Bundle();
 datos.putInt("edad",25);
 datos.putString("nombre","pedro");
 
//Asocio el bundle al intent
 intencion.putExtras(datos);
 
//Activo la activity
 startActivity(intencion);

Leer los datos que pasa el intent

  • Para leer los datos en la activity invocada con el intent, se hace de manera sencilla
  1. Creamos un objeto budle
  2. Lo instanciamos con el bundle que nos retorna el método getExtras del intent que lo ha activado (con el método getIntent()
  3. Lemos los valores con los métodos getXXX(etiqueta)
public class ClaseInvocadaConIntent  extends Activity {
....	
	@Override 
	protected void onCreate(Bundle estado){
 
		Bundle valores = getIntent().getExtras();
		int edad = valores.getInt("edad");
		String nombre = valores.getString("nombre");
                ......
	}
....

Devolver valores a una activity que me invocó

  • Ahora vamos a ver como es el proceso en el que la activity invocada retorna valores a la activity que la invocó

Archivo:ActivityRetornaValores.png

Situación
  • La activity A invoca a la activity B
  • La activity B retorna un valor a la activity A
Paso 1
La activity A llama a la Activity B

Para ello usamos el método

 startActivityForResult(Intnet i, int codigo);
  • En este caso pasamos un segundo parámetro que es un código que indentifica la invocación de la acción
  • Este código permitirá asegurarnos que retorno de esta activity y no de otra.
  • Supongamos que una activity pueda invocar a varias.
    • en el retorno de cualquiera de ellas si retorna valor, siempre iré al método onActivityResult(int resquest, int result , Intendt)
    • Analizando el parámetro request podemos ver de qué intent venimos (que será el mismo valor que le hayamos pasado en la invocación.
........
	public void clickActividad_B(View v){
		Intent i = new Intent(this,ActividadB.class);
		String nombre=((EditText)findViewById(R.id.TNombre)).getText().toString();
		String apellido=((EditText)findViewById(R.id.editText1)).getText().toString();
		Bundle datos = new Bundle();
		datos.putString("nombre",nombre);
		datos.putString("apellido",apellido );
		i.putExtras(datos);
		startActivityForResult(i,OK_ACTIVIDAD_2);
	}
 
	public void clickActividad_C(View v){
		Intent i = new Intent(this,ActividadB.class);
		String nombre=((EditText)findViewById(R.id.TNombre)).getText().toString();
		String apellido=((EditText)findViewById(R.id.editText1)).getText().toString();
		Bundle datos = new Bundle();
		datos.putString("nombre",nombre);
		datos.putString("apellido",apellido );
		i.putExtras(datos);
		startActivityForResult(i,OK_ACTIVIDAD_3);
	}
 
	@Override
	public void onActivityResult(int codAccion, int codDatos, Intent i){
.........
 
		if (codAccion==OK_ACTIVIDAD_2)
			Log.i("ActividadA","Vengo de actividad OK_2, Actividad B");
		if (codAccion==OK_ACTIVIDAD_3)
			Log.i("ActividadA","Vengo de actividad OK_3, Actividad C");
 
......
 
	}
}

Devolviendo resultados

  • Ahora queremos hacer un diseño como el siguiente

Intent2.png

  1. Realizamos los diseños
fichero manifest con dos activity
Dos ficheros java
dos ficheros de layout

Los ficheros java

    public void validarUsuario(View v){
    	Intent datosUsuario = new Intent();
    	EditText tNombre= (EditText) findViewById(R.id.Nombre);
    	EditText tApellido= (EditText) findViewById(R.id.Apellido);
 
    	//Obtenemos el valor de la caja de texto
 
    	//Utilizamos putExtra() para obtener los pares nombre/valor
    	datosUsuario.putExtra("nombre", tNombre.getText().toString());
    	datosUsuario.putExtra("apellido", tApellido.getText().toString());
 
    	//Utilizamos el método setData() para devover algunos valores
    	datosUsuario.setData(Uri.parse("Enviando valores"));
 
    	//Establecemos el resultado con OK
    	setResult(RESULT_OK,datosUsuario);
 
    	//Destruimos el objeto
    	finish();
 
 
    	//iniciamos actividad
    	datosUsuario.setClass(this, ValidacionDatos.class);
    	startActivityForResult(datosUsuario,1);
    }
 
public void onActivityResult (int codigoRespuesta, int codigoResultado, Intent datos){
	if (codigoRespuesta==1){
		if (codigoResultado == RESULT_OK){
			Toast.makeText(this, "no no se se", Toast.LENGTH_LONG);
		}
	}
 
}
public class ValidacionDatos extends Activity {
	public void onCreate(Bundle estadoActividad){
		super.onCreate(estadoActividad);
		setContentView(R.layout.activity_validacion_datos);
 
 
	//Obtener los datos pasados usand getStringExtra()
	Toast.makeText(this,getIntent().getStringExtra("nombre"),Toast.LENGTH_LONG).show();
 
	//Obtener los datos pasados usand getStringExtra()
	Toast.makeText(this,getIntent().getStringExtra("apellido"),Toast.LENGTH_LONG).show();
 
 
	//Obetener el objeto Bundle pasado
	//Bundle nose=getIntent().getExtras();
 
	//Obtener los datos pasados usand getStringExtra()
	//Toast.makeText(this,nose.getString("nombre"),Toast.LENGTH_LONG).show();
 
	//Obtener los datos pasados usand getStringExtra()
	//Toast.makeText(this,nose.getString("apellido"),Toast.LENGTH_LONG).show();
	}
 
	public void clickValidacion(View v) {
		Toast.makeText(this,"Valicación terminada",Toast.LENGTH_LONG).show();
		finish();
 
	}
}