Devahoy Logo
PublishedAt

Android

Day 18 - ShowcaseView

Day 18 - ShowcaseView

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

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

สำหรับวันนี้ขอนำเสนอเรื่อง ShowcaseView เป็น Library ที่เอาไว้ทำเป็น Guide หรือ Showcase ให้แอพของเรา โดยโฟกัสไปยังตำแหน่งที่เราต้องการครับ

ShowcaseView

Installation

ขั้นตอนการติดตั้ง เพิ่ม dependencies ลงไปที่ไฟล์ build.gradle (ไฟล์ภายใน Module)

1
dependencies {
2
compile ('com.fortysevendeg.swipelistview:swipelistview:1.0-SNAPSHOT@aar') {
3
transitive = true
4
}
5
}

ต่อจากนั้นก็เพิ่ม Maven Central ที่ไฟล์ build.gradle ใน Root Project ดังนี้

1
repositories {
2
mavenCentral()
3
}

ต่อมาทำการดาวน์โหลดตัว Library มาไว้ในโปรเจ็คของเรา จากนั้นก็ เปิดไฟล์ settings.gradle แล้วทำการ include Library ที่โหลดมาเมื่อกี้ให้ถูกต้อง

1
include ':libs:showcaseview:library'

ด้านบน ผมเซฟตัว Library ไว้ในโฟลเดอร์ libs ชื่อโปรเจ็คว่า showcaseview และตัวโมดูลชื่อว่า library

ShowcaseView

ทำกด Sync Project with Gradle File เป็นอันเรียบร้อย

Getting Started

การใช้งาน ShowcaseView จะมีโค๊ดดังนี้

1
new ShowcaseView.Builder(this)
2
.setTarget(new ActionViewTarget(this, ActionViewTarget.Type.HOME))
3
.setContentTitle("ShowcaseView")
4
.setContentText("This is highlighting the Home button")
5
.setOnClickListener(this)
6
.build();

ลักษณะการสร้าง ShowcaseView จะคล้ายๆกับการทำ AlertDialog เพราะใช้ Builder Pattern เหมือนกัน เรามาดูว่าจากโค๊ดด้านบน คืออะไรบ้าง

  • setTarget() : ตัวนี้เป็นการระบุว่า จะให้มันโฟกัสไปยังตำแหน่งที่เราต้องการ
  • setContentTitle() : ใส่ Title ให้กับ ShowcaseView
  • setContentText() : ใส่ข้อความรายละเอียดให้กับ showcaseView
  • setOnClickListener() : เอาไว้รับ Listener เมื่อมีการกดปุ่ม Button

Create Project

มาลองสร้างโปรเจ็คกันเลยดีกว่าครับ จะได้เห็นภาพว่ามันใช้งานยังไง? เริ่่มแรกทำการสร้างไฟล์เลเอาท์ขึ้นมา 1 ไฟล์ ชื่อว่า activity_showcaseview.xml

1
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
2
android:orientation="vertical" android:layout_width="match_parent"
3
android:layout_height="match_parent">
4
5
<ImageView
6
android:layout_width="match_parent"
7
android:layout_height="match_parent"
8
android:layout_centerInParent="true"
9
android:scaleType="centerCrop"
10
android:src="@drawable/guitar" />
11
12
<RelativeLayout
13
android:layout_width="match_parent"
14
android:layout_height="match_parent"
15
android:background="#8831415a"
16
android:id="@+id/relative_layout"
17
android:padding="16dp">
18
<TextView
19
android:layout_width="wrap_content"
20
android:layout_height="wrap_content"
21
android:textSize="24sp"
22
android:textColor="@color/white"
23
android:text="@string/step1"
24
android:id="@+id/step1"
25
android:layout_marginTop="32dp"
26
android:layout_alignParentTop="true"
27
android:layout_alignParentLeft="true" />
28
29
<TextView
30
android:layout_width="wrap_content"
31
android:layout_height="wrap_content"
32
android:textSize="24sp"
33
android:textColor="@color/white"
34
android:text="@string/step2"
35
android:id="@+id/step2"
36
android:layout_alignParentBottom="false"
37
android:layout_alignParentRight="false"
38
android:layout_centerInParent="true" />
39
40
<TextView
41
android:layout_width="wrap_content"
42
android:layout_height="wrap_content"
43
android:textSize="20sp"
44
android:textColor="@color/white"
45
android:text="@string/step3"
46
android:id="@+id/step3"
47
android:layout_alignLeft="@+id/step1"
48
android:layout_marginBottom="36dp"
49
android:singleLine="false"
50
android:layout_alignParentBottom="true" />
51
</RelativeLayout>
52
53
</RelativeLayout>

ไฟล์ res/values/strings.xml ที่ใช้กับเลเอาท์ด้านบน

1
<resources>
2
<string name="step1">Step 1 : Learn Android</string>
3
<string name="step2">Step 2 : Make First App</string>
4
<string name="step3">Step 3 : Practice, Practice and Practice</string>
5
<string name="help">Help</string>
6
</resources>

ส่วนรูปภาพ ใครอยากเปลี่ยนรูปหรือลบทิ้งก็ได้ครับ แต่ถ้าจะเอารูปภาพตามตัวอย่าง ก็โหลดจากนี้ guitar

เมื่อออกแบบเลเอาท์เสร็จแล้ว จะได้ลักษณะแบบนี้

Layout

ต่อมาสร้างไฟล์ ShowcaseViewActivity.java ขึ้นมา จากนั้นก็ทำการเชื่อม View จาก Layout แบบปกติครับ ดังนี้

1
public class ShowcaseViewActivity extends ActionBarActivity {
2
3
private TextView mStep1, mStep2, mStep3;
4
private ShowcaseView mShowcaseView;
5
private int mState = 0;
6
7
@Override
8
protected void onCreate(Bundle savedInstanceState) {
9
super.onCreate(savedInstanceState);
10
setContentView(R.layout.day18_activity_showcaseview);
11
12
mStep1 = (TextView) findViewById(R.id.step1);
13
mStep2 = (TextView) findViewById(R.id.step2);
14
mStep3 = (TextView) findViewById(R.id.step3);
15
}
16
}

ต่อมาทำการสร้างออปเจ็ค ShowcaseView ขึ้นมาแบบนี้

1
mShowcaseView = new ShowcaseView.Builder(this)
2
.setTarget(new ViewTarget(mStep1))
3
.build();
4
mShowcaseView.setButtonText("Next Step");

จะเห็นว่าด้านบน ผมทำการเซตค่า Button ให้ชื่อว่า Next Step และทำการ setTarget() ไปที่ step1 นั่นก็คือ TextView ตัวบนสุดครับ

ส่วนวิธีการสร้าง Target ทำได้ 2 แบบครับ คือ

  • new ViewTarget(view); : โดย view ต้องเป็น view ที่เราทำการ findViewById() แล้ว
  • new ViewTarget(id, Activity) : parameter ตัวแรก คือ id ของ View เช่น R.id.step1 และตัวสองคือ Activity

ลองทดสอบรันดูครับ ผลลัพธ์จะได้ดังภาพ

Step1

จะเห็นว่ามันมีวงกลมไปโฟกัส ที่ตำแหน่ง TextView ที่ชื่อ mStep1 ที่เรา setTarget ไว้ และก็มี Button ปุ่มขวาล่างที่เราตั้งชื่อว่า Next Step ทีนี้ลองเปลี่ยนจาก mStep1 เป็น mStep2 ดูครับ

1
.setTarget(new ViewTarget(mStep2))

ลองรันใหม่อีกครั้ง

Step2

แต่ว่าเมื่อเรากดปุ่ม Button ตัว ShowcaseView มันก็จะหายไปทันที ทีนี้เราจะทำยังไง เมื่อกด ก็ให้มันไล่ไปทีละสเตปๆ เริ่มจาก mStep1 ไปจนถึง mStep3 ครับ

ตัว ShowcaseView มี Listener ให้เหมือนกันครับ เอาไว้รับ listener เหมือน Button ปกติเลยครับ กลับไปแก้ไขตรง ShowcaseView.Builder() ให้ครับ โดยเพิ่ม setOnClickListener(ShowcaseViewOnClick) ไปก่อนที่จะทำการ build() เป็นดังนี้

1
mShowcaseView = new ShowcaseView.Builder(this)
2
.setTarget(new ViewTarget(mStep1))
3
.setOnClickListener(ShowcaseViewOnClick)
4
.build();
5
mShowcaseView.setButtonText("Next Step");

ทีนี้ก็สร้าง interface ShowcaseViewOnClick ขึ้นมา (จริงๆ ใครถนัดทำ anonymous inner class ก็ได้นะครับ หรือจะ implements interface แล้ว override เมธอด onClick เอาก็ได้เหมือนกัน แต่ผมชอบวิธีนี้มากกว่า)

เรื่องที่เกี่ยวข้อง : Android Button และการรับ Event ด้วย OnClick

1
View.OnClickListener ShowcaseViewOnClick = new View.OnClickListener() {
2
@Override
3
public void onClick(View v) {
4
switch (mState) {
5
case 0:
6
mShowcaseView.setShowcase(new ViewTarget(mStep2), true);
7
// mShowcaseView.setTarget(new ViewTarget(mStep2));
8
break;
9
case 1:
10
mShowcaseView.setShowcase(new ViewTarget(mStep3), true);
11
// mShowcaseView.setTarget(new ViewTarget(mStep3));
12
break;
13
case 2:
14
ActionViewTarget target = new ActionViewTarget(ShowcaseViewActivity.this,
15
ActionViewTarget.Type.OVERFLOW);
16
mShowcaseView.setTarget(target);
17
break;
18
default:
19
mShowcaseView.hide();
20
mLayout.setBackgroundColor(Color.TRANSPARENT);
21
setVisible(true);
22
break;
23
}
24
mState += 1;
25
}
26
};

ข้อควรระวัง import OnClickListener ให้ถูก package ด้วยนะครับ ต้องเป็น

1
import android.view.View.OnClickListener;

จะเห็นว่าใน onClick() ผมทำการเพิ่มค่า mState ทุกครั้งที่มีการคลิก Button จากนั้นก็ทำ switch case เช็คว่า ถ้ากดครั้งแรก ให้ไปที่ step2 กดครั้งที่สองให้ไป Step3 กดครั้งที่สามไปบน ActionBar และกดครั้งสุดท้าย จะเป็นการซ่อน ShowcaseView

  • setShowcase(Target, animation); : จะโฟกัสและทำการเล่น Animation ด้วย หากค่าเป็น true
  • setTarget(Target); : จะแค่โฟกัสอย่างเดียว ไม่มีการเล่น animation

สุดท้ายผมทำการแก้โค๊ดใหม่ แล้วลองทดสอบอีกครั้งครับ โดยเพิ่มในส่วน menu เข้าไปด้วย res/menu/menu.xml

1
<menu xmlns:android="http://schemas.android.com/apk/res/android"
2
xmlns:app="http://schemas.android.com/apk/res-auto">
3
<item android:id="@+id/menu_item"
4
android:title="@string/help"
5
app:showAsAction="never"/>
6
</menu>

สุดท้ายเพิ่ม onCreateOptionsMenu ลงไป และโค๊ดทั้งหมดเป็นแบบนี้

1
package com.devahoy.learn30androidlibraries.day18;
2
3
import android.graphics.Color;
4
import android.os.Bundle;
5
import android.support.v7.app.ActionBarActivity;
6
import android.view.Menu;
7
import android.view.MenuItem;
8
import android.view.View;
9
import android.widget.RelativeLayout;
10
import android.widget.TextView;
11
12
import com.devahoy.learn30androidlibraries.R;
13
import com.github.amlcurran.showcaseview.ShowcaseView;
14
import com.github.amlcurran.showcaseview.targets.ActionViewTarget;
15
import com.github.amlcurran.showcaseview.targets.Target;
16
import com.github.amlcurran.showcaseview.targets.ViewTarget;
17
18
public class ShowcaseViewActivity extends ActionBarActivity {
19
20
private TextView mStep1, mStep2, mStep3;
21
private ShowcaseView mShowcaseView;
22
private RelativeLayout mLayout;
23
private int mState = 0;
24
25
@Override
26
protected void onCreate(Bundle savedInstanceState) {
27
super.onCreate(savedInstanceState);
28
setContentView(R.layout.day18_activity_showcaseview);
29
30
mStep1 = (TextView) findViewById(R.id.step1);
31
mStep2 = (TextView) findViewById(R.id.step2);
32
mStep3 = (TextView) findViewById(R.id.step3);
33
mLayout = (RelativeLayout) findViewById(R.id.relative_layout);
34
}
35
36
View.OnClickListener ShowcaseViewOnClick = new View.OnClickListener() {
37
@Override
38
public void onClick(View v) {
39
switch (mState) {
40
case 0:
41
mShowcaseView.setShowcase(new ViewTarget(mStep2), true);
42
// mShowcaseView.setTarget(new ViewTarget(mStep2));
43
44
break;
45
case 1:
46
mShowcaseView.setShowcase(new ViewTarget(mStep3), true);
47
// mShowcaseView.setTarget(new ViewTarget(mStep3));
48
break;
49
case 2:
50
ActionViewTarget target = new ActionViewTarget(ShowcaseViewActivity.this,
51
ActionViewTarget.Type.OVERFLOW);
52
// mShowcaseView.setTarget(target);
53
mShowcaseView.setTarget(target);
54
mLayout.setBackgroundColor(Color.parseColor("#FF31415A"));
55
mShowcaseView.setContentTitle("This is Help Menu");
56
setVisible(false);
57
mShowcaseView.setButtonText("OK");
58
break;
59
default:
60
mShowcaseView.hide();
61
mLayout.setBackgroundColor(Color.TRANSPARENT);
62
setVisible(true);
63
break;
64
}
65
mState += 1;
66
}
67
};
68
69
public void setVisible(boolean isVisible) {
70
if(isVisible) {
71
mStep1.setVisibility(View.VISIBLE);
72
mStep2.setVisibility(View.VISIBLE);
73
mStep3.setVisibility(View.VISIBLE);
74
} else {
75
mStep1.setVisibility(View.GONE);
76
mStep2.setVisibility(View.GONE);
77
mStep3.setVisibility(View.GONE);
78
}
79
}
80
81
@Override
82
public boolean onCreateOptionsMenu(Menu menu) {
83
getMenuInflater().inflate(R.menu.menu, menu);
84
85
mShowcaseView = new ShowcaseView.Builder(this)
86
.setTarget(new ViewTarget(mStep1))
87
.setOnClickListener(ShowcaseViewOnClick)
88
.build();
89
90
mShowcaseView.setButtonText("Next Step");
91
92
return super.onCreateOptionsMenu(menu);
93
}
94
95
@Override
96
public boolean onOptionsItemSelected(MenuItem item) {
97
return true;
98
}
99
}

ได้ผลลัพธ์เป็นที่น่าพอใจครับ กด Button ก็จะเลื่อนสเตปไปเรื่อยๆ

สรุป

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

Source Code

References

Authors
avatar

Chai Phonbopit

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

Related Posts