Devahoy Logo
PublishedAt

Android

Day 13 - View Pager

Day 13 - View Pager

สวัสดีครับ บทความนี้เป็นบทความที่ 13 แล้วนะครับ ที่ผมจะมาเขียน ในซีรีย์ Learn 30 Android Libraries in 30 days

สำหรับบทความทั้งหมด อ่านได้จากด้านล่างครับ

สำหรับวันนี้ขอนำเสนอเรื่อง ViewPager ครับ จริงๆแล้ว วันนี้ผมกะเขียนเรื่อง ViewPagerIndicator แต่ว่าเห็นว่าเนื้อหาส่วนใหญ่ จำเป็นต้องรู้ เช่น ViewPager และ Fragment ซะก่อน ทำให้ผมตัดสินใจเลือกทำเป็นบทความสอน ViewPager ขึ้นมาซะเลย ทำให้บทความนี้เป็นการทบทวนการใช้ ViewPager (ซึ่งไม่ได้ใช้นาน) ของผมเองแล้วก็นำมาเรียบเรียงเป็นบทสอนด้วยครับ

Getting Started

การสร้าง ViewPager จำเป็นต้องมี 3 ส่วนหลักๆคือ

  • ViewPager : เป็นคลาส Layout Manager ที่เอาไว้ swipe เพื่อเปลี่ยน Fragment
  • Fragment : สำหรับแสดงหน้า Layout แต่ละหน้า
  • FragmentStatePageAdapter : เป็น Adapter ที่เอาไว้จัดการ Fragment แต่ละหน้า (นึกถึง ListView หรือ GridView ก็จะมี Adapter คอยจัดการ Layout และข้อมูล)

Step 1 : Create Framgent

ขั้นแรก เราจะสร้างหน้า Layout ขึ้นมาก่อน เริ่มต้นด้วยการสร้าง day13_fragment.xml ขึ้นมา ซึ่งมีแค่ TextView ในส่วนนี้จะเอาไว้แสดง ข้อความไม่ให้ซ้ำกันในแต่ละหน้า

1
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
2
android:id="@+id/linear_layout"
3
android:orientation="vertical"
4
android:gravity="center"
5
android:layout_width="match_parent"
6
android:layout_height="match_parent">
7
8
<TextView
9
android:layout_width="wrap_content"
10
android:layout_height="wrap_content"
11
android:id="@+id/greeting"
12
android:textColor="@android:color/white"
13
android:textSize="32sp" />
14
15
</LinearLayout>

ต่อมาสร้างคลาส Fragment ขึ้นมา ผมตั้งชื่อว่า SimpleFragment.java

1
package com.devahoy.learn30androidlibraries.day13;
2
3
import android.graphics.Color;
4
import android.os.Bundle;
5
import android.support.v4.app.Fragment;
6
import android.view.LayoutInflater;
7
import android.view.View;
8
import android.view.ViewGroup;
9
import android.widget.LinearLayout;
10
import android.widget.TextView;
11
12
import com.devahoy.learn30androidlibraries.R;
13
14
public class SimpleFragment extends Fragment {
15
16
@Override
17
public View onCreateView(LayoutInflater inflater, ViewGroup container,
18
Bundle savedInstanceState) {
19
20
ViewGroup rootView = (ViewGroup) inflater.inflate(
21
R.layout.day13_fragment, container, false);
22
23
return rootView;
24
}
25
}

จากด้านบน สร้าง Fragment จากนั้นเพิ่มเธอด onCreateView() เพื่อทำการสร้าง ViewGroup จาก เลเอาท์ด้านบนที่เราเพิ่งสร้างไป

ต่อมาผมทำการ binding View 2 ตัวคือ TextView และ LinearLayout เพื่อที่จะให้แต่ละหน้าของ Fragment หน้าตาแตกต่างกัน โดยเพิ่มข้อมูลจำลองไปเป็นแบบนี้

1
package com.devahoy.learn30androidlibraries.day13;
2
3
import android.graphics.Color;
4
import android.os.Bundle;
5
import android.support.v4.app.Fragment;
6
import android.view.LayoutInflater;
7
import android.view.View;
8
import android.view.ViewGroup;
9
import android.widget.LinearLayout;
10
import android.widget.TextView;
11
12
import com.devahoy.learn30androidlibraries.R;
13
14
public class SimpleFragment extends Fragment {
15
16
@Override
17
public View onCreateView(LayoutInflater inflater, ViewGroup container,
18
Bundle savedInstanceState) {
19
20
int position = 0;
21
22
int[] colors = {
23
Color.rgb(0xF6, 0x47, 0x47), // #F64747
24
Color.rgb(0x9A, 0x12, 0xB3), // #9A12B3
25
Color.rgb(0x22, 0xA7, 0xF0), // #22A7F0
26
Color.rgb(0x4B, 0x77, 0xBE), // #4B77BE
27
Color.rgb(0x1B, 0xBC, 0x9B), // #1BBC9B
28
Color.rgb(0xF2, 0x79, 0x35) // #F27935
29
};
30
31
final String[] names = {
32
"John Doe", "Jane Doe", "Chuck Norris", "Janie Roe", "James Roe"
33
};
34
35
Bundle bundle = getArguments();
36
37
position = bundle.getInt(SimplePagerAdapter.ARGS_POSITION);
38
39
ViewGroup rootView = (ViewGroup) inflater.inflate(
40
R.layout.day13_fragment, container, false);
41
42
TextView greeting = (TextView) rootView.findViewById(R.id.greeting);
43
LinearLayout linearLayout = (LinearLayout)
44
rootView.findViewById(R.id.linear_layout);
45
46
linearLayout.setBackgroundColor(colors[position % colors.length]);
47
greeting.setText("Hello " + names[position % names.length]);
48
49
return rootView;
50
}
51
}

จากโค๊ดด้านบน เรายังไม่มีในส่วนของ getArguments() นะครับ เนื่องจากส่วนนี้จะถูกส่งมาจาก Activity ซึ่งเรายังไม่ได้สร้าง โดยเราจะส่งเป็น หน้า Page มาว่าหน้าที่เท่าไหร่ จากนั้นก็แสดง colors และ names ตามตำแหน่งที่ส่งมา

โค๊ดข้างบนยังมี error นะครับ เนื่องจากยังไม่ได้สร้างทั้ง Activity และ PagerAdapter ที่กำลังจะสร้างในขั้นตอนถัดไป

Step 2 : FragmentStatePagerAdapter

ต่อมาหลังจากสร้าง Fragment แล้ว ต่อไปก็เป็นการสร้าง Adapter โดยทำการ extends เ้้จ้า FragmentStatePagerAdapter ให้สร้างคลาสขึ้นมาใหม่ 1 คลาส สมมติผมตั้งชื่อว่า SimplePagerAdapter

1
package com.devahoy.learn30androidlibraries.day13;
2
3
import android.os.Bundle;
4
import android.support.v4.app.Fragment;
5
import android.support.v4.app.FragmentManager;
6
import android.support.v4.app.FragmentStatePagerAdapter;
7
8
class SimplePagerAdapter extends FragmentStatePagerAdapter {
9
10
public static final String ARGS_POSITION = "name";
11
public static final int NUM_PAGES = 5;
12
13
public SimplePagerAdapter(FragmentManager fm) {
14
super(fm);
15
}
16
17
@Override
18
public Fragment getItem(int position) {
19
20
Fragment fragment = new SimpleFragment();
21
Bundle args = new Bundle();
22
args.putInt(ARGS_POSITION, position);
23
fragment.setArguments(args);
24
25
return fragment;
26
}
27
28
@Override
29
public int getCount() {
30
return NUM_PAGES;
31
}
32
}

จากโค๊ดด้านบน เมื่อเราทำการ extends FragmentStatePagerAdapter แล้ว เราก็ต้อง override เมธอด 2 ตัวครับ คือ getItem() สำหรับแสดงหน้า Fragment และ getCount() สำหรับแสดงจำนวน page ทั้งหมดของ ViewPager

ในส่วน getItem() จะเห็นว่าเราทำการ new SimpleFragment() ขึ้นมา แล้วส่งค่า argument ให้กับ Fragment หากลองขึ้นไปดูคลาส SimpleFragment จะเห็นว่าเรา เรียก getArgument() เพื่อรับค่าตัวนี้ไว้นั่นเอง

ต้องทำการสร้าง Constructor เพื่อสั่ง call ตัว Constructor ของ FragmentStatePagerAdapter ด้วยนะครับ!

Step 3 : ViewPager

มาถึงขั้นตอนสุดท้ายสำหรับการสร้าง ViewPager กันแล้ว การจะใช้ ViewPager เราต้องทำการสร้าง Layout ขึ้นมา่ก่อน ผมตั้งชื่อว่า activity_viewpager.xml ซึ่งภายใน Layout มี ViewPager อยู่อันเดียวครับ

1
<?xml version="1.0" encoding="utf-8"?>
2
<android.support.v4.view.ViewPager
3
android:id="@+id/pager"
4
xmlns:android="http://schemas.android.com/apk/res/android"
5
android:layout_width="match_parent"
6
android:layout_height="match_parent"/>

ต่อมา สร้างคลาสใหม่ขึ้นมา ตั้งชื่อว่า ViewPagerActivity.java

1
package com.devahoy.learn30androidlibraries.day13;
2
3
import android.os.Bundle;
4
import android.support.v4.view.PagerAdapter;
5
import android.support.v4.view.ViewPager;
6
import android.support.v7.app.ActionBarActivity;
7
8
import com.devahoy.learn30androidlibraries.R;
9
10
public class ViewPagerActivity extends ActionBarActivity {
11
12
@Override
13
protected void onCreate(Bundle savedInstanceState) {
14
super.onCreate(savedInstanceState);
15
setContentView(R.layout.day13_activity_viewpager);
16
17
ViewPager viewPager = (ViewPager) findViewById(R.id.pager);
18
PagerAdapter adapter = new SimplePagerAdapter(getSupportFragmentManager());
19
20
viewPager.setAdapter(adapter);
21
}
22
}

จากโค๊ดด้านบน เราทำการเซท Layout โดยใช้เลเอาท์ด้านบน ที่มีแค่ ViewPager จากนั้นก็ setAdapter ให้มันซะ

ทดสอบรันโปรแกรมครับ จะได้ผลลัพธ์แบบภาพข้างล่างนี้

Normal

Step 4 : Title Strip

หากเราต้องการเพิ่มในส่วน Title เข้าไปแต่ละหน้าด้วยละ ทำยังไง? ทำได้โดยการใช้ PagerTitleStrip ครับ โดยการเปิดไฟล์ activity_viewpager.xml จากนั้นเพิ่มโค๊ดในส่วน PagerTitleStrip เข้าไปดังนี้ (สังเกตว่าต้องเพิ่มลงไปอยู่ภายใน ViewPager นะครับ ไม่ใช่ต่อกัน)

1
<?xml version="1.0" encoding="utf-8"?>
2
<android.support.v4.view.ViewPager
3
android:id="@+id/pager"
4
xmlns:android="http://schemas.android.com/apk/res/android"
5
android:layout_width="match_parent"
6
android:layout_height="match_parent">
7
8
<android.support.v4.view.PagerTitleStrip
9
android:id="@+id/pager_title_strip"
10
android:layout_width="match_parent"
11
android:layout_height="wrap_content"
12
android:layout_gravity="top"
13
android:background="#3EAFF9"
14
android:paddingBottom="4dp"
15
android:paddingTop="8dp"
16
android:textColor="#fff"/>
17
18
</android.support.v4.view.ViewPager>

ทดสอบรับโปรแกรม

No Title

อ้าวเห้ย มีแต่ Strip แต่ไม่มี Title ขึ้นนี่นา? ถูกแล้วครับ เนื่องจากว่าเรายังไมไ่ด้เพิ่ม Title ให้มันครับ กลับไปแก้ไขโดยไปที่ไฟล์ SimplePagerAdapter จากนั้นเพิ่่มเมธอด getPageTitle() ลงไปดังนี้

1
package com.devahoy.learn30androidlibraries.day13;
2
3
import android.os.Bundle;
4
import android.support.v4.app.Fragment;
5
import android.support.v4.app.FragmentManager;
6
import android.support.v4.app.FragmentStatePagerAdapter;
7
8
class SimplePagerAdapter extends FragmentStatePagerAdapter {
9
...
10
11
@Override
12
public CharSequence getPageTitle(int position) {
13
return "PAGE#" + (position + 1);
14
}
15
}

แล้วรองรันโปรแกรมใหม่อีกครั้ง

Strip

OK! ได้ผลเป็นที่น่าพอใจแล้ว :D

Step 5 : Tab Title

สุดท้ายแล้ว หากเรายังเปลี่ยน Title จาก Strip ไปใช้ Tab แทน จะทำยังไง?

ต้องใช้ ActionBar เข้ามาช่วยครับ โดยการเซท ActionBar.NAVIGATION_MODE_TABS หากจะใช้ Tab แทน Strip ก็ให้ลบ PagerTitleStrip ออกจาก Layout ซะก่อนครับ เหลือไว้แค่ ViewPager เหมือนเดิม

จากนั้นเปิดไฟล์ ViewPagerActivity เพิ่มในส่วนของ ActionBar ดังนี้

1
package com.devahoy.learn30androidlibraries.day13;
2
3
import android.app.ActionBar;
4
import android.app.FragmentTransaction;
5
import android.os.Bundle;
6
import android.support.v4.view.PagerAdapter;
7
import android.support.v4.view.ViewPager;
8
import android.support.v7.app.ActionBarActivity;
9
10
import com.devahoy.learn30androidlibraries.R;
11
12
public class ViewPagerActivity extends ActionBarActivity {
13
14
private ViewPager mViewPager;
15
16
@Override
17
protected void onCreate(Bundle savedInstanceState) {
18
super.onCreate(savedInstanceState);
19
20
final ActionBar actionBar = getActionBar();
21
actionBar.setNavigationMode(ActionBar.NAVIGATION_MODE_TABS);
22
23
setContentView(R.layout.day13_activity_viewpager);
24
25
mViewPager = (ViewPager) findViewById(R.id.pager);
26
PagerAdapter adapter = new SimplePagerAdapter(getSupportFragmentManager());
27
mViewPager.setAdapter(adapter);
28
29
for (int i = 0; i < SimplePagerAdapter.NUM_PAGES; i++) {
30
actionBar.addTab(actionBar.newTab()
31
.setText("Tab #" + i)
32
.setTabListener(tabListener));
33
}
34
}
35
36
private ActionBar.TabListener tabListener = new ActionBar.TabListener() {
37
@Override
38
public void onTabSelected(ActionBar.Tab tab, FragmentTransaction ft) {
39
mViewPager.setCurrentItem(tab.getPosition());
40
}
41
42
@Override
43
public void onTabUnselected(ActionBar.Tab tab, FragmentTransaction ft) {
44
45
}
46
47
@Override
48
public void onTabReselected(ActionBar.Tab tab, FragmentTransaction ft) {
49
50
}
51
};
52
}

จากโค๊ดด้านบน เราทำการ getActionBar() จากนั้นก็ทำการ set ให้อยู่ในโหมด Tab จากนั้นก็วนลูปตามจำนวนของ Page แล้ว addTab() โดยตั้งชื่อTab ตามตำแหน่งของ Page และทำการ setTabListener() ให้แท็ป เมื่อมีการคลิกที่แท็ป ตรงส่วน onTabSelected เราจะเซตค่า ViewPager.setCurrentItem() เป็นหน้าเดียวกับที่ Tab ถูกเลือก

Tab

Cool เลือกแท้ป แล้ว Page เปลี่ยน แต่ว่าตอน swipe Page เจ้า Tab กลับไม่ยอมเปลี่ยนตาม ทำไม? เพราะว่าเรายังไม่ได้ setListener() ให้กับ ViewPager เมื่อตอนที่มีการเปลี่ยน Page ครับ ทำได้โดยเพิ่มนี้ลงไป

1
mViewPager.setOnPageChangeListener(new ViewPager.OnPageChangeListener() {
2
@Override
3
public void onPageScrolled(int position, float positionOffset, int positionOffsetPixels) {
4
5
}
6
7
@Override
8
public void onPageSelected(int position) {
9
getActionBar().setSelectedNavigationItem(position);
10
}
11
12
@Override
13
public void onPageScrollStateChanged(int state) {
14
15
}
16
});

สุดท้ายทดสอบรันโปรแกรมใหม่อีกครั้ง (ผมเพิ่ม PAGE_NUM เป็น 50) คราวนี้สมบูรณ์เรียบร้อยแล้ว จบครับ :D

Completed

สรุป

บทความนี้ขอใช้เป็นบทความสอน การใช้ ViewPager นะครับ เพื่อปูพื้นฐานนำไปสู่การใช้ Library ที่ชื่อ ViewPagerIndicator ด้วยครับ เนื่องจากว่ามันต้องพอรู้เรื่อง ViewPager และ Fragment บ้าง ถ้าเขียน ViewPagerIndicator ไปเลย เกรงว่าจะมีบางคนงงอีกครับ ซึ่งบทความที่ผ่านมา ก็มีคนมาถามอยู่เรื่องๆ (ผมอุตส่าห์แนบลิงค์ศึกษาเพิ่มเติมไปแล้วนะ หาก search หาซักนิด จากคีย์เวิร์ดที่ให้ไป) สำหรับบทความนี้ ก็ถือเป็น Library เนอะ เพราะเป็น Support Library Version 4 นะครับ :D

Source Code

References

Authors
avatar

Chai Phonbopit

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

Related Posts