Devahoy Logo
PublishedAt

Android

การใช้ Intent เพื่อเปิดหน้า Activity และส่งข้อมูลระหว่าง Activity

การใช้ Intent เพื่อเปิดหน้า Activity และส่งข้อมูลระหว่าง Activity

บทความนี้ขอนำเสนอบทความพื้นฐานสำหรับมือใหม่เลยครับ เกี่ยบกับเรื่อง Intent และการเปิด Activity หนึ่งไปยังอีก Activity หนึ่ง รวมถึงการส่งข้อมูลระหว่าง Activity ด้วยครับ

จริงๆ ตั้งใจว่าจะไม่เขียนพวกพื้นฐานเท่าไหร่ เพราะหาอ่านได้ตามเว็บอื่นๆ หรือตามหนังสือก็มีอยู่เยอะแยะมากมาย แต่มาคิดอีกที เราอาจจะคิดว่ามันเยอะแยะ แต่บางคนละไม่คิดแบบเรา เค้าอาจจะหาไม่เจอ หรือยังไม่เข้าใจ ก็ต้องพยายามอ่านหลายๆเว็บ ผมก็เลยตัดสินใจ จะทำบทความครอบคลุมพื้นฐานด้วยละกัน อย่างน้อยก็เผื่อมีบางคนมาอ่าน แล้วได้ประโยชน์ ไม่บ้างก็น้อยแหละ :D

มาเข้าเรื่อง Intent และ Activity กันเลยดีกว่า

TLDR

  1. สร้าง Activity ขึ้นใหม่ สมมติชื่อ AboutActivity
  2. เพิ่ม activity name ใน AndroidManifest.xml ให้รู้จัก AboutActivity
  3. ใช้ startActivity(intent); เพื่อเปิดหน้าใหม่
  4. ใช้ finish() เพื่อกลับมา Activity เดิม

การเปิด Activity ไปอีก Activity หนึ่ง

สำหรับบางคน ที่เริ่มเขียนใหม่ อาจจะใช้แค่ Activity เดียวในแอพ แต่ว่าอยากจะให้กดปุ่ม แล้วเปิดอีก Activity หนึ่ง หรือเปลี่ยน Activity จะทำยังไง?

เริ่มต้นสร้างโปรเจ็คใหม่เลยครับ สมมติว่าทุกคน สามารถสร้างโปรเจ็คใหม่เป็นนะครับ ใครไม่เป็นอ่านบทความนี้ สร้างโปรเจ็คด้วย Android Studio

ที่นี้เราจะใช้ ไฟล์ 2 ไฟล์คือ

  • MainActivity.java
  • AboutActivity.java

และเลเอาท์ 2 ไฟล์ คือ

  • activity_main.xml
  • activity_about.xml

เลเอาท์สร้างไว้ที่ /res/layout/ ส่วน Activity สร้างไว้ที่เดียวกับคลาส MainActivity เลย

Layout

activity_main.xml

1
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
2
android:layout_width="match_parent"
3
android:layout_height="match_parent" >
4
5
<Button
6
android:id="@+id/button_about"
7
android:layout_width="wrap_content"
8
android:layout_height="wrap_content"
9
android:text="About"
10
android:paddingTop="24dp"
11
android:paddingBottom="24dp"
12
android:paddingLeft="48dp"
13
android:paddingRight="48dp"
14
android:layout_centerVertical="true"
15
android:layout_centerHorizontal="true"/>
16
17
<ImageView
18
android:layout_width="128dp"
19
android:layout_height="128dp"
20
android:id="@+id/imageView"
21
android:src="@drawable/ic_launcher"
22
android:layout_above="@+id/button_back"
23
android:layout_centerHorizontal="true"/>
24
25
</RelativeLayout>

activity_about.xml

1
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
2
android:layout_width="match_parent"
3
android:layout_height="match_parent">
4
5
<TextView
6
android:text="AboutActivity"
7
android:layout_width="wrap_content"
8
android:layout_height="wrap_content"
9
android:textSize="36sp"
10
android:layout_marginTop="32dp"
11
android:layout_alignParentTop="true"
12
android:layout_centerHorizontal="true"/>
13
14
<Button
15
android:id="@+id/button_back"
16
android:layout_width="wrap_content"
17
android:layout_height="wrap_content"
18
android:text="Back"
19
android:paddingTop="24dp"
20
android:paddingBottom="24dp"
21
android:paddingLeft="48dp"
22
android:paddingRight="48dp"
23
android:layout_centerVertical="true"
24
android:layout_centerHorizontal="true"/>
25
26
</RelativeLayout>

ทีนี้เป้าหมายเรา คือ กดปุ่ม Button จาก MainActivity แล้วให้ทำการเปิด AboutActivity ขึ้นมา จากนั้นหน้า AboutActivity เมื่อกด Back ก็จะกลับมาหน้า MainActivity อีกครั้ง

ง่ายมากครับ แค่เพิ่มโค๊ด สองบรรทัดนี้ลงไป ใน onClickListener คือเมื่อทำการกดปุ่มก็จะเปิด AboutActivity ขึ้นมา นั่นเอง

1
Intent intent = new Intent(this, AboutActivity.class);
2
startActivity(intent);

โค๊ดมันมีแค่นี้แหละ คือ เราจะสร้างออปเจ็ค intent ขึ้นมา โดยรับ parameter 2 ตัว ตัวแรกคือ Context ภายใน Activityแรก และ parameter ตัวที่สองคือ Activity ที่เราต้องการเปิด

โค๊ดเต็มๆของ MainActivity คือ

1
package com.devahoy.sample.intent;
2
3
import android.content.Intent;
4
import android.os.Bundle;
5
import android.support.v7.app.ActionBarActivity;
6
import android.view.View;
7
import android.widget.Button;
8
9
public class MainActivity extends ActionBarActivity {
10
11
@Override
12
protected void onCreate(Bundle savedInstanceState) {
13
super.onCreate(savedInstanceState);
14
setContentView(R.layout.activity_main);
15
16
Button buttonAbout = (Button) findViewById(R.id.button_about);
17
18
buttonAbout.setOnClickListener(new View.OnClickListener() {
19
@Override
20
public void onClick(View v) {
21
Intent intent = new Intent(MainActivity.this, AboutActivity.class);
22
startActivity(intent);
23
}
24
});
25
}
26
}

ส่วนฝั่ง AboutActivity เมื่อกดปุ่ม back ก็จะกลับมา Activity จะใช้ finish() ครับ หมายถึงการปิด Activity ครับ แต่ถ้าเราไปใช้แบบ ด้านบน คือเปลี่ยนเป็นแบบนี้

1
Intent intent = new Intent(this, MainActivity.class);
2
startActivity(intent);

มันก็จะเปิดหน้า MainActivity ได้ครับ แต่ว่ามันจะมีหน้าซ้อนกัน 2 หน้าครับ (โดยปกติใน Android Activity มันจะถูกเปิดซ้อนกันไปเรื่อยๆครับ) ไม่เชื่อคุณลองรันโปรแกรม แล้วสั่งเปิดหน้าสอง แล้วกด back หลายๆครั้งดูครับ Activity มันจะซ้อนกันตามจำนวนที่คุณกดเลย

คลาส AboutActivity.java

1
package com.devahoy.sample.intent;
2
3
import android.os.Bundle;
4
import android.support.v7.app.ActionBarActivity;
5
import android.view.View;
6
import android.widget.Button;
7
8
public class AboutActivity extends ActionBarActivity {
9
10
@Override
11
protected void onCreate(Bundle savedInstanceState) {
12
super.onCreate(savedInstanceState);
13
setContentView(R.layout.activity_about);
14
15
Button buttonBack = (Button) findViewById(R.id.button_back);
16
17
buttonBack.setOnClickListener(new View.OnClickListener() {
18
@Override
19
public void onClick(View v) {
20
finish();
21
}
22
});
23
}
24
}

แต่ถ้าหากอยากจะใช้แบบ ส่ง Intent ก็ทำได้ครับ โดย MainActivity ต้อง เรียก finish() ด้วยครับ เช่น

MainActivity.java

1
Intent intent = new Intent(this, AboutActivity.class);
2
startActivity(intent);
3
finish();

AboutActivity.java

1
Intent intent = new Intent(this, MainActivity.class);
2
startActivity(intent);
3
finish();

แบบนี้ก็จะไม่เกิด Activity ซ้ำกันแล้ว แต่ว่าในกรณีที่เรามี 2 Activity แค่กดปิด แล้วกลับไปหน้าหลัก ดีกว่า ที่จะต้องมาสร้างหน้าหลักใหม่ อีก จริงมั้ยครับ :D

สุดท้ายที่ไฟล์ AndroidManifest.xml เราต้องไปเพิ่มในส่วนของ Android Name ให้มันรู้จัก Activity ที่เราเพิ่มมา

1
<activity android:name=".AboutActivity">
2
</activity>

ปัญหาการใช้ this

จะเห็นว่า บางทีพวกนี้จะเป็นปัญหาสำหรับมือใหม่เลย คือใส่ this แต่ไม่รู้ว่ามันอยู่ใน block, scope ไหน ก็ยังไปใส่ this เช่น

1
btnSubmit.setOnClickListener(new OnClickListener() {
2
@Override
3
public void onClick(View v) {
4
Intent intent = new Intent(this, AboutActivity.class);
5
startActivity(intent);
6
}
7
});

ด้านบน เรียกเปิด Activity ใหม่ จากภายใน Anonymous Inner Class ของ OnClickListener ซึ่ง this มันก็จะไปมองเป็น OnClickListener ไม่ได้มองเป็น MainActivity วิธีแก้ก็ต้องเปลี่ยนเป็นแบบนี้

1
btnSubmit.setOnClickListener(new OnClickListener() {
2
@Override
3
public void onClick(View v) {
4
Intent intent = new Intent(MainActivity.this, AboutActivity.class);
5
//หรือแบบนี้
6
//Intent intent = new Intent(getApplicationContext(), AboutActivity.class);
7
startActivity(intent);
8
}
9
});

หรือทางแก้อีกแบบ ก็ประกาศตัวแปรหนึ่งเป็น global ไปเลย เช่น

1
public class YourClass extends Activity {
2
private Context mContext;
3
4
@Override
5
protected void onCreate(Bundle savedInstanceState) {
6
mContext = getApplicationContext();
7
}
8
}

ทีนี้ทุกอย่างที่ต้องการ ApplicationContext ก็แค่ส่ง mContext ไปเท่านั้น

อีกปัญหานึงที่เจอบ่อยๆ คือ จะเปิด Activity ใหม่ จากหน้า Fragment แต่ดันไปใช้ this หรือ MyFragment.this มันก็จะไปได้อย่างไร? เนื่องจาก context มันอยู่ใน Activity ฉะนั้นก็ต้องเรียกแบบนี้

1
Intent intent = new Intent(getActivity(), AboutActivity.class);
2
startActivity(intent);

การส่งข้อมูลข้าม Activity

วิธีการส่งข้อมูลข้าม Activity กรณีเรามีข้อมูลอยู่ MainActivity แต่อยากให้มันไปโชว์ที่หน้า DetailActivity เราจะทำอย่างไร?

วิธีการ ก็คือใช้ Intent เหมือนเดิมครับ โดยทาง Intent จะมีเมธอด putExtra(key, value) มาให้เราสำหรับส่งค่าไปครับ parameter แรกจะเป็นชื่อเรียก ส่วน parameter สอง จะเป็นชนิดของตัวแปร จะเป็น int, String, boolean, float ก็ได้ครับ ดังเช่นตัวอย่าง

ในคลาส MainActivity เรา ทำการเปิด AboutActivity โดยส่งค่า ไปด้วย

1
Intent intent = new Intent(this, AboutActivity.class);
2
intent.putExtra("name", "devahoy");
3
intent.putExtra("isSmart", true);
4
intent.putExtra("star", 5);
5
startActivity(intent);

จากนั้นที่ AboutActivity ก็รับค่าที่ส่งมา แบบนี้ เปิด onCreate()

1
Bundle bundle = getIntent().getExtras();
2
if (bundle != null) {
3
String name = bundle.getString("name");
4
int star = bundle.getInt("star");
5
boolean isSmart = bundle.getBoolean("smart");
6
7
String result = String.format("Name is %s, star : %s, smart : %s",
8
name, star, isSmart);
9
Toast.makeText(this, "Name : " + result, Toast.LENGTH_SHORT).show();
10
}

เราจะทำการเช็คก่อน ว่ามีการส่งค่ามาจริงไหม ถ้ามีจริง ก็ใช้ Bundle.getXXX เช่นจะรับค่า String ก็ใช้ getString(key) ข้างในก็เป็น key ตัวเดียวกับที่่ไว้ก่อนส่งมาจาก MainActivity

สุดท้าย ก็โชว์ Toast แสดงผลลัพธ์ที่ส่งมา มีแค่นี้แหละครับ ลองนำไปประยุกต์ใช้กันดูครับ

Authors
avatar

Chai Phonbopit

เป็น Web Dev ในบริษัทแห่งหนึ่ง ทำงานมา 10 ปีกว่าๆ ด้วยภาษาและเทคโนโลยี เช่น JavaScript, Node.js, React, Vue และปัจจุบันกำลังสนใจในเรื่องของ Blockchain และ Crypto กำลังหัดเรียนภาษา Rust

Related Posts