https://blog.csdn.net/qinjuning/article/details/7726093
一、 BluetoothAdapter类介绍
1、BluetoothAdapter STATE 状态值 , 即开关状态
int STATE_OFF 蓝牙已经关闭
int STATE_ON 蓝牙已经打开
int STATE_TURNING_OFF 蓝牙处于关闭过程中 ,关闭ing
int STATE_TURNING_ON 蓝牙处于打开过程中 ,打开ing
2、BluetoothAdapter SCAN_MOD状态值 ,即扫描状态
可以扫描其他设备的,当然它同时能被其他蓝牙设备扫码。
int SCAN_MODE_CONNECTABLE 表明该蓝牙可以扫描其他蓝牙设备
int SCAN_MODE_CONNECTABLE_DISCOVERABLE
表明该蓝牙设备同时可以扫码其他蓝牙设备,并且可以被其他蓝牙设备扫描到。
int SCAN_MODE_NONE : 该蓝牙不能扫描以及被扫描。
3、获得蓝牙适配器实例
public static synchronized BluetoothAdapter getDefaultAdapter ()
功能:获得本设备的蓝牙适配器实例。
返回值:如果设备具备蓝牙功能,返回BluetoothAdapter 实例;否则,返回null对象。
4、打开/关闭蓝牙的两种方法:
4.1、打开蓝牙:
①、直接调用函数enable()去打开蓝牙设备 ;
②、系统API去打开蓝牙设备,该方式会弹出一个对话框样式的Activity供用户选择是否打开蓝牙设备。
注意:如果蓝牙已经开启,不会弹出该Activity界面。
PS:在目前Android手机中,是不支持在飞行模式下开启蓝牙的。如果蓝牙已经开启,那么蓝牙的开关状态会随着飞行模式的状态而发生改变。
4.2、关闭蓝牙
直接调用API 函数即disable()即可。
public boolean disable ()
功能:关闭蓝牙设备。
返回值:该函数会立即返回。
true 表示关闭操作成功
false 表示蓝牙操作失败 , ①、当前蓝牙已经关闭 ; ②、其他一些异常情况
5、扫描蓝牙设备
public boolean startDiscovery ()
功能: 扫描蓝牙设备
注意: 如果蓝牙没有开启,该方法会返回false ,即不会开始扫描过程。
public boolean cancelDiscovery ()
功能: 取消扫描过程。
注意: 如果蓝牙没有开启,该方法会返回false。
public boolean isDiscovering ()
功能: 是否正在处于扫描过程中。
注意: 如果蓝牙没有开启,该方法会返回false。
6、 获取蓝牙相关信息
public String getName ()
功能:获取蓝牙设备Name
public String getAddress ()
功能:获取蓝牙设备的硬件地址(MAC地址),例如:00:11:22:AA:BB:CC
public boolean setName (String name)
功能:设置蓝牙设备的Name
public Set<BluetoothDevice> getBondedDevices ()
功能:获取与本机蓝牙所有绑定的远程蓝牙信息,以BluetoothDevice类实例返回。
注意:如果蓝牙未开启,该函数会返回一个空集合。
public static boolean checkBluetoothAddress (String address)
功能: 验证蓝牙设备MAC地址是否有效。所有设备地址的英文字母必须大写,48位,形如:00:43:A8:23:10:F1 。
返回值: true 设备地址有效
false 设备地址无效
public BluetoothDevice getRemoteDevice (String address)
功能:以给定的MAC地址去创建一个 BluetoothDevice 类实例(代表远程蓝牙实例)。即使该蓝牙地址不可见,也会产生
一个BluetoothDevice 类实例。
返回:BluetoothDevice 类实例 。注意,如果该蓝牙设备MAC地址不能被识别,其蓝牙Name为null。
异常:如果MAC address无效,抛出IllegalArgumentException。
7、蓝牙相关广播
Action值 说明
ACTION_STATE_CHANGED 蓝牙状态值发生改变
ACTION_SCAN_MODE_CHANGED 蓝牙扫描状态(SCAN_MODE)发生改变
ACTION_DISCOVERY_STARTED 蓝牙扫描过程开始
ACTION_DISCOVERY_FINISHED 蓝牙扫描过程结束
ACTION_LOCAL_NAME_CHANGED 蓝牙设备Name发生改变
ACTION_REQUEST_DISCOVERABLE 请求用户选择是否使该蓝牙能被扫描
PS:如果蓝牙没有开启,用户点击确定后,会首先开启蓝牙,继而设置蓝牙能被扫描。
ACTION_REQUEST_ENABLE 请求用户选择是否打开蓝牙
ACTION_FOUND (该常量字段位于BluetoothDevice类中,稍后讲到)
说明:蓝牙扫描时,扫描到任一远程蓝牙设备时,会发送此广播。
最后,在使用这两个类时,需要以下两个权限:
BLUETOOTH 值为 "android.permission.BLUETOOTH"
BLUETOOTH_ADMIN 值为 "android.permission.BLUETOOTH_ADMIN"
蓝牙连接打印机
myBluetoothAdapter = BluetoothAdapter.getDefaultAdapter(); // 获取本地蓝牙适配器
if (myBluetoothAdapter != null && myBluetoothAdapter.isEnabled()) {//判断蓝牙可用且已开启
address = preferences.getString("printAddress", ""); // 查看配置文件中是否保存有蓝牙地址
if (!isEmpty(address)) {
Set<BluetoothDevice> devices = myBluetoothAdapter.getBondedDevices(); // 获取所有已经配置的设置列表
if (devices.size() == 0) {
return;
}
for(BluetoothDevice device : devices) {
String address1 = device.getAddress();
//遍历已配对列表,上次连接的设备已配对时自动配对
if (address.equals(address1)) {
new Thread(new Runnable() {
@Override
public void run() {
connectionBluetooth(address); // 通过连接蓝牙地址连接蓝牙打印机
}
}).start();
}
}
}
}
/**
* 通过连接蓝牙地址连接蓝牙打印机
*/
private void connectionBluetooth(String addr) {
BluetoothDevice device = myBluetoothAdapter.getRemoteDevice(addr);
connectionBluetooth(device);
}
/**
* 通过蓝牙设置数据连接蓝牙打印机
*/
private void connectionBluetooth(BluetoothDevice device) {
connectionDevice = device;
connectedDeviceName = device.getName();
initKeZhi(address);
}
/**
* 柯芝打印机连接方式
*/
private boolean initKeZhi(String adress){
int arg = OperateInterface.STATE_NONE;
if (adress == "" || adress == null) {
showBlueStat(getResources().getString(R.string.notSelectPrinter));
return false;
}
BluetoothDevice myDevice;
if (myBluetoothAdapter == null) {
showBlueStat(getResources().getString(R.string.readingBluetoothError));
return false;
}
myDevice = myBluetoothAdapter.getRemoteDevice(adress);
if (myDevice == null) {
showBlueStat(getResources().getString(R.string.readingBluetoothError));
return false;
}
arg = OperateInterface.STATE_CONNECTING;
myHandler.obtainMessage(OperateInterface.MESSAGE_STATE_CHANGE, arg, 0).sendToTarget();
if(zpSDK.connect(adress))
{
blueConnectResult = true;
arg = OperateInterface.STATE_CONNECTED;
}else {
blueConnectResult = false;
showBlueStat(getResources().getString(R.string.connectError));
arg = OperateInterface.STATE_NONE;
myHandler.obtainMessage(OperateInterface.MESSAGE_STATE_CHANGE, arg, 0).sendToTarget();
return false;
}
myHandler.obtainMessage(OperateInterface.MESSAGE_STATE_CHANGE, arg, 0).sendToTarget();
return true;
}、
点击未配对时弹出选择框
选择配对时跳转到系统蓝牙配对界面
startActivity(new Intent(Settings.ACTION_BLUETOOTH_SETTINGS));
选择连接时
/**
* 弹出已配对连接列表,选择蓝牙进行连接
*/
private void selectBluetooth() {
Set<BluetoothDevice> devices = myBluetoothAdapter.getBondedDevices(); // 获取所有已经配置的设置列表
if (devices.size() == 0) { // 判断已经配对列表,如果为0则提示
Toast.makeText(this, R.string.noPairedDevices, Toast.LENGTH_LONG).show(); // 没有已经配对的设备
openOptionsMenu();
return;
}
final Dialog dialog = new Dialog(this); // 实例一个对话框
dialog.requestWindowFeature(Window.FEATURE_NO_TITLE); // 设置对话框无标题
View view = LayoutInflater.from(this).inflate(R.layout.bluetooth_device_list, null); // 创建蓝牙列表布局界面
ListView listView = (ListView) view.findViewById(R.id.bluetoothDeviceList); // 获取ListView
List<BluetoothDevice> data = new ArrayList<BluetoothDevice>(); // 将已经配对的set集合转换成List集合
data.addAll(devices);
BluetoothDeviceAdapter adapter = new BluetoothDeviceAdapter(this, data, R.layout.bluetooth_device_item);
listView.setAdapter(adapter); // 将适配器应用到ListView
listView.setOnItemClickListener(new OnItemClickListener() { // 注册列表条目点击事件
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
BluetoothDevice device = (BluetoothDevice) parent.getItemAtPosition(position);
address = device.getAddress();
connectedDeviceName = device.getName();
connectionBluetooth(device);
dialog.dismiss();
}
});
dialog.setContentView(view); // 将列表应用到弹出框中
Window dialogWindow = dialog.getWindow(); // 获取当前屏幕(以下为设置弹出框的属性)
WindowManager.LayoutParams lp = dialogWindow.getAttributes(); // 获取对话框的属性
WindowManager windowManager = getWindowManager(); // 获取窗口管理器
Display display = windowManager.getDefaultDisplay(); // 获取窗口默认显示的对象
lp.width = (int) (display.getWidth() * 0.95); // 设置对话框的属性,宽度为当前默认屏幕的宽度的0.9
dialogWindow.setGravity(Gravity.CENTER); // 设置对话框在屏幕的水平居中
dialog.show(); // 显示弹出框
}
最后在onDestroy()中断开连接zpSDK.disconnect();