小城天长网站建设,沈阳企业模板建站,信阳企业网站开发,做网站网站怎么赚钱蓝牙页面-手动点击搜索蓝牙 然后选中某个打印机 进行手动点击打印--权限配置什么的 在上面某个都已配置了import React, { useEffect, useState, useRef } from react;
import {Button,View,Alert,PermissionsAndroid,Platform,Text,ScrollView,TouchableOpacity,
} from reac…蓝牙页面-手动点击搜索蓝牙 然后选中某个打印机 进行手动点击打印--权限配置什么的 在上面某个都已配置了import React, { useEffect, useState, useRef } from react; import { Button, View, Alert, PermissionsAndroid, Platform, Text, ScrollView, TouchableOpacity, } from react-native; import { BleManager, Device } from react-native-ble-plx; import { Buffer } from buffer; import iconv from iconv-lite; const manager new BleManager(); const BluetoothPrinter ({ route }) { const { arrearsData } route.params || {}; console.log(打印数据:, arrearsData); const [isScanning, setIsScanning] useState(false); const [foundDevices, setFoundDevices] useState([]); const [isConnected, setIsConnected] useState(false); const [currentDevice, setCurrentDevice] useState(null); const [servicesInfo, setServicesInfo] useState(); const [connectionStatus, setConnectionStatus] useState(未连接); const [writeCharacteristics, setWriteCharacteristics] useState([]); const [scanTimeLeft, setScanTimeLeft] useState(0); const scanTimeoutRef useRef(null); const scanTimerRef useRef(null); useEffect(() { const stateSubscription manager.onStateChange(state { console.log(蓝牙状态:, state); if (state PoweredOn) { console.log(蓝牙已开启); } else if (state PoweredOff) { Alert.alert(提示, 请开启手机蓝牙); setConnectionStatus(蓝牙未开启); stopScan(); // 蓝牙关闭时停止扫描 } }, true); return () { stateSubscription.remove(); stopScan(); // 组件卸载时停止扫描 }; }, []); // 权限请求函数 const requestBluetoothPermission async () { if (Platform.OS ! android) { return true; } try { const apiLevel Platform.Version; let permissions []; if (apiLevel 31) { permissions [ android.permission.BLUETOOTH_SCAN, android.permission.BLUETOOTH_CONNECT, android.permission.ACCESS_FINE_LOCATION, ]; } else { permissions [ android.permission.BLUETOOTH, android.permission.BLUETOOTH_ADMIN, android.permission.ACCESS_FINE_LOCATION, ]; } const granted await PermissionsAndroid.requestMultiple(permissions); const allGranted permissions.every( permission granted[permission] PermissionsAndroid.RESULTS.GRANTED, ); return allGranted; } catch (err) { console.warn(权限请求错误:, err); return false; } }; const scanDevices async () { try { const hasPermission await requestBluetoothPermission(); if (!hasPermission) { Alert.alert(权限被拒绝, 需要蓝牙权限才能扫描设备); return; } const state await manager.state(); if (state ! PoweredOn) { Alert.alert(蓝牙未开启, 请先开启手机蓝牙); return; } setIsScanning(true); setFoundDevices([]); setServicesInfo(); setConnectionStatus(扫描中...); setScanTimeLeft(15); // 设置15秒倒计时 console.log(开始扫描蓝牙设备...); // 启动倒计时 scanTimerRef.current setInterval(() { setScanTimeLeft(prev { if (prev 1) { clearInterval(scanTimerRef.current); return 0; } return prev - 1; }); }, 1000); // 15秒后自动停止扫描 scanTimeoutRef.current setTimeout(() { if (isScanning) { console.log(自动停止扫描); stopScan(); Alert.alert( 提示, 扫描已完成共找到 foundDevices.length 个设备, ); } }, 15000); manager.startDeviceScan(null, null, (error, device) { if (error) { console.log(扫描错误:, error); stopScan(); setConnectionStatus(扫描失败); return; } if (device device.name) { console.log(发现设备:, device.name, 信号:, device.rssi); setFoundDevices(prev { const exists prev.find(d d.id device.id); if (!exists) { return [ ...prev, { id: device.id, name: device.name || 未知设备, localName: device.localName, rssi: device.rssi, isPrinter: isLikelyPrinter(device), }, ].sort((a, b) b.rssi - a.rssi); } return prev; }); } }); } catch (error) { console.log(扫描失败:, error); stopScan(); setConnectionStatus(扫描失败); } }; // 停止扫描函数 const stopScan () { try { manager.stopDeviceScan(); } catch (error) { console.log(停止扫描错误:, error); } setIsScanning(false); setConnectionStatus(扫描完成); // 清除定时器 if (scanTimeoutRef.current) { clearTimeout(scanTimeoutRef.current); scanTimeoutRef.current null; } if (scanTimerRef.current) { clearInterval(scanTimerRef.current); scanTimerRef.current null; } setScanTimeLeft(0); }; // 判断设备是否可能是打印机 const isLikelyPrinter device { if (!device.name) return false; const name device.name.toLowerCase(); const localName device.localName?.toLowerCase() || ; const printerKeywords [ printer, print, prt, pos, receipt, bt, blue, thermal, 58mm, 80mm, tsc, rtk, u26, xp, sp, ]; return printerKeywords.some( keyword name.includes(keyword) || localName.includes(keyword), ); }; const discoverAllServicesAndCharacteristics async device { try { console.log(开始发现服务和特征...); setConnectionStatus(发现服务中...); await device.discoverAllServicesAndCharacteristics(); const services await device.services(); console.log(发现的服务数量:, services.length); const allWriteCharacteristics []; for (const service of services) { try { const characteristics await device.characteristicsForService( service.uuid, ); for (const char of characteristics) { // 收集所有可写特征 if (char.isWritableWithResponse || char.isWritableWithoutResponse) { allWriteCharacteristics.push({ serviceUUID: service.uuid, characteristicUUID: char.uuid, isWritableWithResponse: char.isWritableWithResponse, isWritableWithoutResponse: char.isWritableWithoutResponse, }); } } } catch (charError) { console.log(获取服务 ${service.uuid} 特征失败:, charError); } } console.log(所有可写特征:, allWriteCharacteristics); return allWriteCharacteristics; } catch (error) { console.log(发现服务失败:, error); throw error; } }; // 连接设备不打印 const connectToDevice async (deviceId, deviceName) { try { // 连接设备前先停止扫描 stopScan(); setConnectionStatus(连接中...); Alert.alert(提示, 正在连接: ${deviceName}); console.log(连接设备:, deviceName); // 连接设备 const device await manager.connectToDevice(deviceId, { timeout: 15000, }); console.log(设备连接成功); setIsConnected(true); setCurrentDevice(device); setConnectionStatus(已连接); // 发现所有服务和特征 const characteristics await discoverAllServicesAndCharacteristics( device, ); setWriteCharacteristics(characteristics); if (characteristics.length 0) { throw new Error(未找到可写的特征); } Alert.alert( 连接成功, 已连接到: ${deviceName}\n\n发现 ${characteristics.length} 个可写特征\n现在可以点击打印按钮进行打印, ); } catch (error) { console.log(连接失败:, error); setIsConnected(false); setCurrentDevice(null); setWriteCharacteristics([]); setConnectionStatus(连接失败); throw error; } }; // 断开连接 const disconnectDevice async () { if (currentDevice) { try { await manager.cancelDeviceConnection(currentDevice.id); setIsConnected(false); setCurrentDevice(null); setWriteCharacteristics([]); setConnectionStatus(已断开); console.log(设备已断开连接); } catch (error) { console.log(断开连接错误:, error); } } }; // 打印函数 const handlePrint async () { if (!currentDevice || writeCharacteristics.length 0) { Alert.alert(错误, 请先连接打印机); return; } try { setConnectionStatus(打印中...); // 准备打印数据 const printContent [ \x1B\x40, // 初始化打印机 \x1B\x61\x01, // 居中对齐 \x1B\x45\x01, // 加粗 用水通知单 \r\n, \x1B\x45\x00, // 关闭加粗 \x1B\x61\x00, // 左对齐 本期应收: ${arrearsData?.Payment || }\r\n, 总欠费金额: ${arrearsData?.TotalPayment || }\r\n, 余额: ${arrearsData?.Balance || }\r\n, --------------------------------\r\n, \x1B\x45\x01, // 加粗 温馨提示:请及时通过微信公众号或者营业厅进行缴费\r\n, \x1B\x45\x00, // 取消加粗 打印日期: ${new Date().toLocaleString()}\r\n, \r\n, 谢谢合作\r\n, \r\n\r\n\r\n, \x1D\x56\x00, // 切纸 ].join(); // 使用 GBK 编码 const gbkBuffer iconv.encode(printContent, gbk); const base64Data gbkBuffer.toString(base64); let printSuccess false; let lastError null; // 尝试所有可写特征 for (const charInfo of writeCharacteristics) { try { console.log(尝试特征: ${charInfo.characteristicUUID}); if (charInfo.isWritableWithResponse) { await currentDevice.writeCharacteristicWithResponseForService( charInfo.serviceUUID, charInfo.characteristicUUID, base64Data, ); } else { await currentDevice.writeCharacteristicWithoutResponseForService( charInfo.serviceUUID, charInfo.characteristicUUID, base64Data, ); } printSuccess true; break; } catch (error) { console.log( 特征 ${charInfo.characteristicUUID} 失败:, error.message, ); lastError error; } } if (printSuccess) { Alert.alert(打印成功, 用水通知单已发送到打印机); setConnectionStatus(打印成功); } else { throw new Error(所有特征尝试失败: ${lastError?.message}); } } catch (error) { console.log(打印失败:, error); Alert.alert(打印失败, error.message); setConnectionStatus(打印失败); } }; const getSignalStrength rssi { if (rssi -50) return { text: 强, color: green }; if (rssi -70) return { text: 中, color: orange }; return { text: 弱, color: red }; }; return ( View style{{ padding: 20, flex: 1, backgroundColor: white }} View style{{ marginBottom: 20, padding: 10, backgroundColor: #f0f0f0, borderRadius: 5, }} Text style{{ fontWeight: bold, fontSize: 16 }} 状态: {connectionStatus} {isScanning (${scanTimeLeft}秒后自动停止)} /Text {currentDevice ( Text style{{ fontSize: 14, color: green, marginTop: 5 }} 已连接设备: {currentDevice.name} /Text )} {writeCharacteristics.length 0 ( Text style{{ fontSize: 12, color: blue, marginTop: 2 }} 发现 {writeCharacteristics.length} 个可写特征 /Text )} /View {/* 操作按钮区域 */} View style{{ flexDirection: row, flexWrap: wrap, marginBottom: 20, gap: 10, }} Button title{isScanning ? 停止扫描 (${scanTimeLeft}s) : 扫描设备} onPress{isScanning ? stopScan : scanDevices} color{isScanning ? red : #007AFF} / {isConnected ( Button title打印通知单 onPress{handlePrint} colorgreen / Button title断开连接 onPress{disconnectDevice} colororange / / )} /View ScrollView style{{ flex: 1 }} {foundDevices.length 0 ( View Text style{{ fontWeight: bold, marginBottom: 10, fontSize: 16 }} 发现的设备 ({foundDevices.length}个): /Text {foundDevices.map(device { const signal getSignalStrength(device.rssi); return ( TouchableOpacity key{device.id} style{{ marginBottom: 12, padding: 12, backgroundColor: device.isPrinter ? #e8f5e8 : #f5f5f5, borderRadius: 8, borderWidth: 1, borderColor: device.isPrinter ? green : #ddd, }} onPress{() connectToDevice(device.id, device.name)} Text style{{ fontWeight: bold, color: device.isPrinter ? green : black, fontSize: 16, }} {device.name} {device.isPrinter ? ️ : } /Text Text style{{ fontSize: 12, color: gray, marginTop: 4 }} ID: {device.id} /Text Text style{{ fontSize: 12, color: gray }} 信号强度: {device.rssi}dBm ({signal.text}) /Text {device.localName device.localName ! device.name ( Text style{{ fontSize: 12, color: gray }} 本地名: {device.localName} /Text )} Text style{{ fontSize: 12, color: device.isPrinter ? green : blue, marginTop: 4, fontWeight: bold, }} {device.isPrinter ? 点击连接此打印机 : 点击连接此设备} /Text /TouchableOpacity ); })} /View )} {!isScanning foundDevices.length 0 ( View style{{ alignItems: center, marginTop: 50 }} Text style{{ textAlign: center, color: gray, fontSize: 16, marginBottom: 10, }} 点击扫描设备开始搜索蓝牙设备 /Text Text style{{ textAlign: center, color: darkgray, fontSize: 14 }} 扫描将在15秒后自动停止 /Text /View )} {isConnected ( View style{{ marginTop: 20, padding: 15, backgroundColor: #e8f5e8, borderRadius: 8, }} Text style{{ fontWeight: bold, fontSize: 16, color: green, marginBottom: 10, }} ✅ 已连接打印机 /Text Text style{{ fontSize: 14, marginBottom: 5 }} 设备名称: {currentDevice?.name} /Text Text style{{ fontSize: 14, marginBottom: 5 }} 可写特征: {writeCharacteristics.length} 个 /Text Text style{{ fontSize: 12, color: darkgreen }} 现在可以点击打印通知单按钮进行打印 /Text /View )} /ScrollView /View ); }; export default BluetoothPrinter;2.注意使用GBK编码打印 解决打印乱码问题