08 February 2014

BluetoothSPP - การสร้างหน้าเลือกอุปกรณ์สำหรับเชื่อมต่อผ่าน Bluetooth

Updated on
        บทความนี้เก่ามากแล้ว และหยุดพัฒนาต่อแล้ว

        สำหรับผู้ที่หลงเข้ามาอ่านที่ได้ยลบทความเบื้องต้นของคลาส BluetoothSPP มาแล้ว ก็จะเห็นว่าคลาสของเจ้าของบล็อกจะมีหน้าเลือกอุปกรณ์ที่จะเชื่อมต่อไว้ให้แล้ว ซึ่งผู้ที่หลงเข้ามาอ่านไม่จำเป็นต้องสร้างหน้านั้นขึ้นมาเอง จึงทำให้เรียกใช้งานได้ง่ายมาก

        ถ้าผู้ที่หลงเข้ามาอ่านยังไม่ได้อ่านบทความการใช้งานคลาส BluetoothSPP มาก่อน ให้ไปอ่านก่อนที่ [Android Code] การเชื่อมต่ออุปกรณ์ผ่านบลูทูธแบบง่ายๆโดยใช้ BluetoothSPP


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

        โดยผู้ที่หลงเข้ามาอ่านสามารถสร้าง Layout ขึ้นมาเองได้โดยกำหนด ID ของ List View กับ Button ให้ตรงตามที่กำหนดไว้ก็พอ หรือจะกำหนดขนาดของ Text View ที่แสดงใน List View ก็ยังได้ และสามารถกำหนดข้อความที่แสดงในหน้านี้ได้เองอีกด้วย

        ยังจำคำสั่ง Intent ไปยัง DeviceActivity ในตัวอย่างพื้นฐานได้หรือไม่?

Intent intent = new Intent(getApplicationContext(), DeviceList.class);
startActivityForResult(intent, BluetoothState.REQUEST_CONNECT_DEVICE); 

        โดยคำสั่งนี้ยังมีอย่างอื่นอีก ซึ่งในบทความที่ผ่านมาเจ้าของบล็อกยังไม่ได้พูดถึง ซึ่งจะมาพูดให้ฟังในบทความนี้ล่ะ แต่ก่อนอื่นมาดูข้อตกลงใจการสร้าง Layout ที่จะเอามาใช้งานก่อน


การกำหนดคุณสมบัติของตัวอักษรที่อยู่ใน List View

        หลักการง่ายๆก็คือสร้าง Layout ที่มีแต่ Text View โดยผู้ที่หลงเข้ามาอ่านสามารถกำหนดคุณสมบัติของ Text View ตัวนั้นได้ตามใจชอบ ไม่ว่าจะขนาด สี หรือ Padding กับ Margin ก็ยังได้

bluetooth_layout_text.xml
<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="https://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:textColor="#7A481B"
    android:textSize="20sp"
    android:padding="10dp"
    android:text="Text" /> 


        จากตัวอย่างข้างบนนี้ก็คือเจ้าของบล็อกปรับเปลี่ยนคุณสมบัติของ Text View เล็กน้อย เปลี่ยนตัวหนังสือให้เป็นสีน้ำตาล มีขนาดใหญ่ขึ้นกว่าปกติหน่อย ส่วนข้อความที่กำหนดในนั้นจะเป็นอะไรก็ได้ เพราะว่าตอนไปแสดงใน List View ก็จะถูกกำหนดค่าใหม่เป็นชื่อและ Address ของอุปกรณ์บลูทูธอยู่ดี


การกำหนดรูปแบบของหน้าเลือกอุปกรณ์ที่จะเชื่อมต่อ

        เดิมทีหน้า Layout ที่ใช้ในหน้า DeviceActivity เจ้าของบล็อกจะสร้างให้รองรับกับหน้าจอหลายๆขนาดอยู่แล้ว 

Phone - Nexus 4


Tablet 7 - Nexus 7


Tablet 10 - Nexus 10


        แต่ถ้าผู้ที่หลงเข้ามาอ่านอยากจะสร้างหน้านี้ซะใหม่ ก็สามารถสร้าง Layout ขึ้นมาใหม่ได้เลย โดยมีเงื่อนไขว่า List View ต้องกำหนดชื่อ ID เป็น list_devices และ Button ที่ใช้กดแล้วค้นหาอุปกรณ์ต้องกำหนดชื่อ ID เป็น button_scan เท่านั้นนะ

        ย้ำว่า List View ต้องเป็น list_devices และ Button ต้องเป็น button_scan เท่าน้านนนนน

        ทีนี้ถ้าผู้ที่หลงเข้ามาอ่านออกแบบเองบ้างล่ะ สมมติว่าเป็นดังนี้

bluetooth_layout_list.xml
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    android:background="#FDE182" >

    <ListView
        android:id="@+id/list_devices"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentTop="true"
        android:layout_marginLeft="20dp"
        android:layout_marginRight="20dp"
        android:layout_marginTop="20dp"
        android:smoothScrollbar="true" />
        
    <Button
        android:id="@+id/button_scan"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentRight="true"
        android:layout_alignParentTop="true"
        android:padding="20dp"
        android:background="#FFC600"
        android:text="SCAN"
        android:textSize="25sp"
        android:textColor="#7A481B"
        android:textStyle="bold" />
        
</RelativeLayout>




การกำหนด Layout ที่ออกแบบเองลงใน BluetoothSPP

        กลับมาต่อกับ Intent ที่เจ้าของบล็อกพูดไว้ในตอนแรก จำได้มั้ยว่า Intent สามารถส่งค่าตัวแปรบางอย่างไปได้ด้วย? นั่นแหละวิธีกำหนด Layout ที่ผู้ที่หลงเข้ามาอ่านได้สร้างขึ้นมาเอง โดยมี Keyword ให้กำหนดทั้งหมดดังนี้
     
        layout_list : กำหนด Layout ที่จะใช้เป็นหน้าเลือกอุปกรณ์ จากตัวอย่างคือ bluetooth_layout_list.xml

        layout_text : กำหนดตัวอักษรที่ใช้แสดงใน List View จากตัวอย่างคือ bluetooth_layout_text.xml

        วิธีกำหนดก็ค่ากำหนดเป็น Integer ส่งไปพร้อมกับ Intent ดังนี้

Intent intent = new Intent(getApplicationContext(), DeviceList.class);
intent.putExtra("layout_list", R.layout.bluetooth_layout_list);
intent.putExtra("layout_text", R.layout.bluetooth_layout_text);
startActivityForResult(intent, BluetoothState.REQUEST_CONNECT_DEVICE); 

        เท่านี้ก็จะเป็นการกำหนด Layout ที่ได้สร้างขึ้นมาเรียบร้อยแล้ว อ้อ เพิ่มเติมอีกอย่าง ไม่จำเป็นต้องกำหนดทั้ง layout_list หรือ layout_text ก็ได้ ผู้ที่หลงเข้ามาอ่านอยากสร้างอย่างใดอย่างหนึ่งก็ได้ แล้วกำหนดแค่อันนั้นก็พอ

Intent intent = new Intent(getApplicationContext(), DeviceList.class);
intent.putExtra("layout_text", R.layout.bluetooth_layout_text);
startActivityForResult(intent, BluetoothState.REQUEST_CONNECT_DEVICE);

        หรือ

Intent intent = new Intent(getApplicationContext(), DeviceList.class);
intent.putExtra("layout_list", R.layout.bluetooth_layout_list);
startActivityForResult(intent, BluetoothState.REQUEST_CONNECT_DEVICE); 

        และนอกจากกำหนด Layout แล้ว ยังกำหนดข้อความที่แสดงในหน้านั้นได้อีกด้วย ซึ่งได้พูดถึงไปในตอนแรกแล้ว

        bluetooth_devices : Title เมื่อเข้าสู่หน้า DeviceActivity

        no_devices_found : ข้อความแสดงเมื่อไม่มีอุปกรณ์ใดให้เชื่อมต่อ

        scanning : ข้อความแสดงเมื่อกำลังทำการค้นหา

        scan_for_devices : ข้อความแสดงให้ผู้ใช้กดเพื่อทำการค้นหา

        select_device : Title หลังจากค้นหาอุปกรณ์เสร็จแล้ว






        โดยที่ Keyword สำหรับข้อความที่แสดงจะต้องกำหนดเป็น String ซึ่งสามารถเลือกได้เช่นกันว่าจะกำหนดเฉพาะตัวไหน เพราะไม่จำเป็นต้องกำหนดทุกตัวก็ได้เช่นกัน

Intent intent = new Intent(getApplicationContext(), DeviceList.class);
intent.putExtra("bluetooth_devices", "Bluetooth devices");
intent.putExtra("no_devices_found", "No device");
intent.putExtra("scanning", "กำลังทำการค้นหา");
intent.putExtra("scan_for_devices", "Search");
intent.putExtra("select_device", "Select");
startActivityForResult(intent, BluetoothState.REQUEST_CONNECT_DEVICE);

        เท่านี้ก็จะเห็นว่าข้อความในหน้า DeviceActivity ได้เปลี่ยนไปตามที่กำหนดแล้ว



        ดังนั้นบทความนี้ก็จะเพิ่มจากคำสั่งพื้นฐานเพียงแค่ไฟล์ Layout 2 ไฟล์ที่สร้างขึ้นมาเป็นตัวอย่าง และคำสั่ง putExtra ใน Intent ที่ส่งไปกำหนดในหน้า DeviceActivity



Main.java
package app.akexorcist.bluetoothsppdevicelist;

import android.os.Bundle;
import android.app.Activity;
import android.content.Intent;
import android.util.Log;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.widget.Toast;
import app.akexorcist.bluetoothspp.BluetoothSPP;
import app.akexorcist.bluetoothspp.BluetoothSPP.OnDataReceivedListener;
import app.akexorcist.bluetoothspp.BluetoothState;
import app.akexorcist.bluetoothspp.DeviceList;

public class Main extends Activity {
    BluetoothSPP bt;
    
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        
        bt = new BluetoothSPP(this);

        if(!bt.isBluetoothAvailable()) {
            Toast.makeText(getApplicationContext(), "Bluetooth is not available"
                    , Toast.LENGTH_SHORT).show();
            finish();
        }
        
        bt.setOnDataReceivedListener(new OnDataReceivedListener() {
            public void onDataReceived(byte[] data, String message) {
                Log.i("Check", "Length : " + data.length);
                Log.i("Check", "Message : " + message);
            }
        });
        
        Button btnConnect = (Button)findViewById(R.id.btnConnect);
        btnConnect.setOnClickListener(new OnClickListener(){
            public void onClick(View v){
                if(bt.getServiceState() == BluetoothState.STATE_CONNECTED) {
                    bt.disconnect();
                } else {
                    Intent intent = new Intent(getApplicationContext(), DeviceList.class);
                    intent.putExtra("bluetooth_devices", "Bluetooth devices");
                    intent.putExtra("no_devices_found", "No device");
                    intent.putExtra("scanning", "กำลังทำการค้นหา");
                    intent.putExtra("scan_for_devices", "Search");
                    intent.putExtra("select_device", "Select");
                    intent.putExtra("layout_list", R.layout.bluetooth_layout_list);
                    intent.putExtra("layout_text", R.layout.bluetooth_layout_text);
                    startActivityForResult(intent, BluetoothState.REQUEST_CONNECT_DEVICE); 
                }
            }
        });
    }
    
    public void onDestroy() {
        super.onDestroy();
        bt.stopService();
    }
    
    public void onStart() {
        super.onStart();
        if(!bt.isBluetoothEnabled()) {
            bt.enable();
        } else {
            if(!bt.isServiceAvailable()) { 
                bt.setupService();
                bt.startService(BluetoothState.DEVICE_ANDROID);
                setup();
            }
        }
    }
    
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
        if(requestCode == BluetoothState.REQUEST_CONNECT_DEVICE) {
            if(resultCode == Activity.RESULT_OK)
                bt.connect(data);
        } else if(requestCode == BluetoothState.REQUEST_ENABLE_BT) {
            if(resultCode == Activity.RESULT_OK) {
                bt.setupService();
            } else {
                Toast.makeText(getApplicationContext(), "Bluetooth was not enabled."
                        , Toast.LENGTH_SHORT).show();
                finish();
            }
        }
    }
    
    public void setup() {
        Button btnSend = (Button)findViewById(R.id.btnSend);
        btnSend.setOnClickListener(new OnClickListener(){
            public void onClick(View v){
                bt.send("Text", true);
            }
        });
    }
}


main.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent" >

    <Button
        android:id="@+id/btnSend"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerHorizontal="true"
        android:layout_centerVertical="true"
        android:text="Send" />

    <Button
        android:id="@+id/btnConnect"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentRight="true"
        android:layout_alignParentTop="true"
        android:text="Connect" />

</RelativeLayout>



AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="app.akexorcist.bluetoothsppdevicelist"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="8"
        android:targetSdkVersion="19" />

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="app.akexorcist.bluetoothsppdevicelist.Main"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>        
    </application>

</manifest>

        จากนั้นให้ลองทดสอบการทำงานดูก็จะพบว่าหน้า DeviceActivity ได้เปลี่ยนไปเรียบร้อยแล้ว



        ทั้งนี้ก็เพื่อให้ผู้ที่หลงเข้ามาอ่านสามารถนำไปใช้และนำไปปรับธีมให้เข้ากับแอปพลิเคชันของผู้ที่หลงเข้ามาอ่านทำขึ้น สำหรับผู้ที่หลงเข้ามาอ่านคนใดต้องการไฟล์ตัวอย่างสามารถดาวน์โหลดได้ที่

                BluetoothSPP Device List [GitHub]

                BluetoothSPP Device List [Google Drive]

                BluetoothSPP Device List [Sleeping For Less]



บทความที่เกี่ยวข้อง

        • เปลี่ยนเรื่องบลูทูธให้เป็นเรื่องง่ายด้วย BluetoothSPP Library
        • การเชื่อมต่ออุปกรณ์ผ่านบลูทูธแบบง่ายๆโดยใช้ BluetoothSPP
        • การใช้งาน Listener สำหรับบลูทูธใน BluetoothSPP
        • การใช้งานการเชื่อมต่ออัตโนมัติในคลาส BluetoothSPP