144 lines
4.4 KiB
Dart
144 lines
4.4 KiB
Dart
|
import 'dart:convert';
|
|||
|
import 'dart:io';
|
|||
|
import 'dart:typed_data';
|
|||
|
import 'package:flutter/material.dart';
|
|||
|
import 'package:nfc_manager/nfc_manager.dart';
|
|||
|
import 'package:nfc_manager/ndef_record.dart';
|
|||
|
import 'package:nfc_manager/nfc_manager_android.dart';
|
|||
|
import 'package:nfc_manager/nfc_manager_ios.dart';
|
|||
|
import 'package:nfc_manager_ndef/nfc_manager_ndef.dart';
|
|||
|
|
|||
|
class NfcTestPage extends StatefulWidget {
|
|||
|
const NfcTestPage({Key? key}) : super(key: key);
|
|||
|
|
|||
|
@override
|
|||
|
_NfcTestPageState createState() => _NfcTestPageState();
|
|||
|
}
|
|||
|
|
|||
|
class _NfcTestPageState extends State<NfcTestPage> {
|
|||
|
String _nfcData = '未扫描到标签。';
|
|||
|
bool _scanning = false;
|
|||
|
|
|||
|
void _safeSetState(String data) {
|
|||
|
if (!mounted) return;
|
|||
|
setState(() => _nfcData = data);
|
|||
|
}
|
|||
|
|
|||
|
Future<void> _startNfcSession() async {
|
|||
|
setState(() {
|
|||
|
_scanning = true;
|
|||
|
_nfcData = '扫描中...';
|
|||
|
});
|
|||
|
|
|||
|
final available = await NfcManager.instance.isAvailable();
|
|||
|
if (!available) {
|
|||
|
_safeSetState('此设备不支持 NFC');
|
|||
|
setState(() => _scanning = false);
|
|||
|
return;
|
|||
|
}
|
|||
|
|
|||
|
NfcManager.instance.startSession(
|
|||
|
onDiscovered: (NfcTag tag) async {
|
|||
|
String text = '未知';
|
|||
|
|
|||
|
try {
|
|||
|
final ndef = Ndef.from(tag);
|
|||
|
if (ndef == null) {
|
|||
|
_safeSetState('标签不支持 NDEF');
|
|||
|
} else {
|
|||
|
final msg = ndef.cachedMessage;
|
|||
|
if (msg == null || msg.records.isEmpty) {
|
|||
|
_safeSetState('未找到任何 NDEF 记录');
|
|||
|
} else {
|
|||
|
// 第一条记录
|
|||
|
final record = msg.records.first;
|
|||
|
text = _parseTextRecord(record);
|
|||
|
_safeSetState('读取到文本: "$text"');
|
|||
|
|
|||
|
}
|
|||
|
}
|
|||
|
String uid = '未知';
|
|||
|
try {
|
|||
|
if (Platform.isAndroid) {
|
|||
|
final nfcA = NfcAAndroid.from(tag);
|
|||
|
if (nfcA != null) {
|
|||
|
// 正确地通过 nfcA.tag.identifier 拿到 List<int>
|
|||
|
final bytes = nfcA.tag.id;
|
|||
|
uid = bytes
|
|||
|
.map((b) => b.toRadixString(16).padLeft(2, '0'))
|
|||
|
.join(':')
|
|||
|
.toUpperCase();
|
|||
|
}
|
|||
|
} else if (Platform.isIOS) {
|
|||
|
// iOS 平台,用 MiFareIos(对于大多数 ISO14443-A,也可以用 Iso7816Ios)
|
|||
|
final mifare = MiFareIos.from(tag);
|
|||
|
final bytes = mifare!.identifier;
|
|||
|
uid = bytes.map((b) => b.toRadixString(16).padLeft(2, '0')).join(':').toUpperCase();
|
|||
|
}
|
|||
|
} catch (e) {
|
|||
|
debugPrint('取 UID 错误: $e');
|
|||
|
} finally {
|
|||
|
setState(() => _nfcData = '标签 UID: $uid' + '\n读取到文本: "$text"');
|
|||
|
await NfcManager.instance.stopSession();
|
|||
|
}
|
|||
|
} catch (e) {
|
|||
|
_safeSetState('解析时出错: $e');
|
|||
|
} finally {
|
|||
|
await NfcManager.instance.stopSession();
|
|||
|
if (mounted) setState(() => _scanning = false);
|
|||
|
}
|
|||
|
},
|
|||
|
pollingOptions: {NfcPollingOption.iso14443},
|
|||
|
);
|
|||
|
}
|
|||
|
|
|||
|
/// 解析一个 NDEF Text 记录的 payload,返回纯文本内容
|
|||
|
String _parseTextRecord(NdefRecord record) {
|
|||
|
final payload = record.payload;
|
|||
|
if (payload.isEmpty) return '';
|
|||
|
|
|||
|
final statusByte = payload[0];
|
|||
|
final langCodeLen = statusByte & 0x3F;
|
|||
|
// 文本数据开始的位置
|
|||
|
final textBytes = payload.sublist(1 + langCodeLen);
|
|||
|
|
|||
|
try {
|
|||
|
return utf8.decode(textBytes); // 默认 UTF‑8
|
|||
|
} catch (e) {
|
|||
|
// 解码失败时,退回到直接展示 bytes
|
|||
|
return String.fromCharCodes(textBytes);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
@override
|
|||
|
void dispose() {
|
|||
|
NfcManager.instance.stopSession();
|
|||
|
super.dispose();
|
|||
|
}
|
|||
|
|
|||
|
@override
|
|||
|
Widget build(BuildContext context) {
|
|||
|
return Scaffold(
|
|||
|
appBar: AppBar(title: const Text('NFC 测试')),
|
|||
|
body: Padding(
|
|||
|
padding: const EdgeInsets.all(16.0),
|
|||
|
child: Column(
|
|||
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
|||
|
children: [
|
|||
|
ElevatedButton(
|
|||
|
onPressed: _scanning ? null : _startNfcSession,
|
|||
|
child: Text(_scanning ? '扫描中...' : '开始扫描 NFC'),
|
|||
|
),
|
|||
|
const SizedBox(height: 24),
|
|||
|
Expanded(
|
|||
|
child: SingleChildScrollView(
|
|||
|
child: Text(_nfcData, style: const TextStyle(fontSize: 16)),
|
|||
|
),
|
|||
|
),
|
|||
|
],
|
|||
|
),
|
|||
|
),
|
|||
|
);
|
|||
|
}
|
|||
|
}
|