Intent
는 메시징 객체로 컴포넌트간의 통신 역할을 해준다.
Intent는 실행하고자 하는 4대 구성 요소와 관련된 정보를 가지고 있다.
안드로이드 4대 구성 요소(Activity, Service, Broadcast Receiver, Content Provier)를 실행하기 위해서 Intent객체가 필요하고, 구성 요소끼리 정보전달을 가능하게 해준다.
개발자는 실행하고자 하는 4대 구성 요소의 정보를 Intent에 담고 이를 안드로이드 OS에게 전달하면 안드로이드 OS에 의해 해당 구성 요소가 실행된다.
명시적 인텐트는 Intent에 클래스 객체나 컴포넌트 이름을 지정하여 호출할 대상을 확실히 알 수 있는 경우에 사용하며 실행되는 컴포넌트가 무엇인지 명시되어 있다.
val intent = Intent(this, MyActivity::class.java);
startActivity(intent);
현재 엑티비티에서 다른 엑티비티를 호출할 때 사용하며, 실행을 원하는 액티비티는 manifest에서 exported속성이 true로 설정되어야 한다.
암시적 인텐트는 작업을 지정하여 기기에서 해당 작업을 수행할 수 있는 모든 앱을 호출할 수 있도록 합니다.
호출하려는 컴포넌트의 패키지명과 클래스명을 모르는 경우(다른 앱을 선택해야 하는 경우) 사용하는 방법으로 호출하려는 작업의 내용을 지정하고, 시스템이 이 작업을 처리할 수 있는 컴포넌트를 찾아서 호출한다.
암시적 인텐트에는 웹브라우저 호출, 이메일 전송, 전화 앱 등이 해당한다.
주의: 사용자에게 개발자가 startActivity()로 전송한 암시적 인텐트
를 처리할 앱이 전혀 없을 수도 있고, 관리자가 프로필 제한이나 설정
을 적용한 탓에 앱에 엑세스할 수 없을 수도 있는데 이런 경우 호출은
실패하고 앱이 꺼지게 된다. 어느 Activity든 Intent를 수신하도록
하려면 Intent 객체의 resolveActivity()를 호출해야 한다. 결과가
null이 아닌 경우, 인텐트를 처리할 수 잇는 앱이 최소한 하나는 있다
는 듯이며 startActivity()를 호출해도 안전하다. 하지만 null일 경
우라면 해당 Intent는 사용하면 안되고 해당 Intent를 발생시키는 기
능을 비활성화 시켜놔야한다.
앱이 수신할 수 있는 암시적 인텐트가 어느 것인지 알리려면, <intent-filter>
요소를 사용하여 각 앱 구성 요소에 대해 하나 이상의 인텐트 필터를 매니페스트 파일에 선언합니다. 각 인텐트 필터는 인텐트의 작업, 데이터 및 카테고리를 기반으로 어느 유형의 인텐트를 수락하는지 지정합니다. 시스템은 인텐트가 인텐트 필터 중 하나를 통과한 경우에만 암시적 인텐트를 앱 구성 요소에 전달합니다.
앱 구성 요소는 자신이 수행할 수 있는 각각의 고유한 작업에 대하여 별도의 필터를 선언해야 합니다. 예를 들어 이미지 갤러리 앱에 있는 액티비티에 두 개의 필터가 있을 수 있습니다. 한 필터는 이미지를 보고, 다른 필터는 이미지를 편집하기 위한 것입니다. 액티비티가 시작되면, Intent를 검사한 다음 Intent에 있는 정보를 근거로 어떻게 동작할 것인지 결정합니다(편집기 제어 항목을 표시할 것인지 말 것인지 등).
<intent-filter>
내부에서는 다음과 같은 세 가지 요소 중 하나 이상을 사용하여 허용할 인텐트 유형을 지정할 수 있습니다.
<action>
name
특성에서 허용된 인텐트 작업을 선한한다. 값은 클래스 상수가 아닌 Action의 리터럴 문자열 값이어야 합니다.<data>
scheme
, host
, port
, path
)와 MIME 유형의 여러 가지 측면을 나타내는 하나 이상의 특성을 사용한다.<category>
name
특성에서 허용된 인텐트 카테고리를 선언한다. 값은 클래스 상수가 아닌 Action의 리터럴 문자열 값이어야 합니다.참고: 암시적 인텐트를 수신하려면 CATEGORY_DEFAULT 카테고리를 인
텐트 필터에 포함해야 한다. startActivity() 및
startActivityForResult() 메서드는 마치 CATEGORY_DEFAULT 범주
를 선언한 것처럼 모든 인텐트를 취급한다. 이 카테고리를 인텐트 필터
에서 선언하지 않으면 액티비티에 어떤 암시적 인텐트도 확인되지 않습
니다.
ex) 데이터 유형이 텍스트인 경우 ACTION_SEND 인텐트를 수신할 인텐트 필터가 있는 액티비티 선언
<activity android:name="ShareActivity">
<intent-filter>
<action android:name="android.intent.action.SEND"/>
<category android:name="android.intent.category.DEFAULT"/>
<data android:mimeType="text/plain"/>
</intent-filter>
</activity>
com.example.ExampleActivity
setComponent()
, setClass()
, setClassName()
을 사용 하거나 Intent
생성자를 통해 설정한다.참고 : Service를 시작하는 경우, Intent에 어느 서비스가 응답할지
확신할 수 없고, 사용자도 어느 서비스가 시작되었는지 볼 수 없게 되
기 때문에 항상 component name을 지정해야 한다.
수행할 작업을 나타내는 문자열이다.(EX. ACTION_VIEW
, ACTION_SEND
)
Action은 대체로 나머지 Intent의 구조(특히, 데이터와 엑스트라에 포함되어 있는 정보)를 결정한다.
보통은 Intent
클래스나 다른 프레임워크 클래스가 정의한 작업을 지정해야 한다.
Action을 수행할 데이터 및 해당 데이터의 MINE 유형을 참조하는 URI
객체이다.
데이터의 유형은 일반적으로 Intent의 Action에서 결정된다. 예를 들어 ACTION_EDIT 인 경우, 데이터에 편집할 문서의 URI가 들어 있어야 한다.
setData()
를 통해setType()
을 통해주의 : URI와 MIME 유형을 둘 다 설정하고자 하는 경우 꼭 ```setDataAndType()``` 하도록 해야한다.
```setData()```, ```setType()```은 서로의 값을 무효화 하기 때문
Intent를 처리해야 하는 component의 종류에 대한 추가 정보를 담은 문자열이다.
Intent 안에는 카테고리 설명이 포함할 수 있지만 대부분의 Intent는 카테고리가 필요하지 않다
카테고리는 addCategory()
로 지정할 수 있다.
보편적인 카테고리 2가지
CATEGORY_BROWSABLE
요청된 작업을 수행하는 데 필요한 추가 정보가 담긴 키-값 쌍으로 되어 있다.
putExtra
로 데이터를 추가할 수 있으며 모든 엑스트라 데이터를 포함한 Bundle
객체르 만든 다음 Bundle
을 Intent
에 putExtras()
로 삽입할 수도 있다.
주의 : Intent를 통해 다른 앱으로 데이터를 전송할 때 Pacelable
또는 Serializable 데이터를 사용하지 말도록 한다. 앱이 Bundle 객
체에 있는 데이터에 엑서스하려고 시도한 경우 Pacelable,
Serializable 클래스에 엑세스 하지 못하면 시스템이
RuntimeException을 발생시킨다.
플래그느 Intent 클래스에서 정의되고 인텐트에 대한 메타데이터와 같은 기능을 한다. 시작할 방법에 대한 지침(Activity가 어느 Action에 소속되어야 하는지 등)이나 Activity를 시작한 다음에 어떻게 처리해야 하는지(해당 Activity가 최근 Activity 목록에 소속되는지 여부)도 알려줄 수 있다.
setFlag()
를 통해 설정할 수 있고 Intent에 의해 실행되는 구성 요소의 유형에 따라 달라진다. FLAGACTIVE 플래그는 모두 Context.startActivity()와 함께 사용되고 FLAGRECIVER 플래그는 모두 Context.sendBroadcast()와 함께 사용된다.
앱에 Intent가 전달되었을 때 Intent Filter에 기술된 조건에 맞는 Intent만 통과시킴 (말 그대로 필터링)
Manifest에서 activity 내부에 선언하며, 해당 activity가 어떤 동작인지, 어떤 카테고리인지 명시하여 일치하는 암시적 인텐트를 받을 수 있도록 함
ex) A 앱에서 Intent에 ACTION_VIEW, CATEGORY_BROWSABLE를 설정한 후 전송하면?
암시적 인텐트이므로 CATEGORY_DEFAULT가 자동 추가된 후 전송
설치된 앱들 중 Intent Filter에 ACTION_VIEW, CATEGORY_BROWSABLE,
CATEGORY_DEFAULT가 정의된 앱들의 목록이 등장함
(해당 앱들의 Filter가 Intent를 통과시킴)
목록에서 앱을 선택하면 해당 앱으로 Intent 전달하며 컴포넌트를 실행함
(코드 예시)
[A 앱] // 웹 사이트를 열기 위한 인텐트를 전송
val intent = Intent().apply {
action = Intent.ACTION_VIEW
data = Uri.parse("https://www.google.com")
}
context.startActivity(intent)
[B 앱 Manifest] // A앱이 보낸 intent의 액션 및 카테고리가 일치하여 통과
<activity
android:name="com.text.BTestActivity">
<intent-filter>
<action android:name="action.ACTION_VIEW/>
<category android:name="android.intent.category.DEFAULT"/>
<category android:name="android.intent.category.BROWSABLE"/>
</intent-filter>
</activity>
B 앱에서 A 앱의 암시적 인텐트를 통과시켜 받았으므로, B 앱에서는 해당 인텐트의 data (google 주소)를 받아 웹을 띄워 줄 수 있다
이런 유용한 정보를 나눠주셔서 감사합니다.