flutter_pcsc 0.0.4 flutter_pcsc: ^0.0.4 copied to clipboard
Flutter plugin to access PCSC readers on Linux/Windows/MacOS
import 'package:flutter/material.dart';
import 'dart:async';
import 'package:flutter_pcsc/flutter_pcsc.dart';
void main() {
MyApp? myApp;
runZonedGuarded(() async {
WidgetsFlutterBinding.ensureInitialized();
FlutterError.onError = (FlutterErrorDetails details) {
FlutterError.dumpErrorToConsole(details);
myApp?.addError(details.toString());
};
runApp(myApp = MyApp());
}, (Object error, StackTrace stack) {
myApp?.addError(error.toString());
});
}
class MyApp extends StatelessWidget {
final GlobalKey<_MyAppBodyState> _myAppKey = GlobalKey();
MyApp({Key? key}) : super(key: key);
@override
Widget build(BuildContext context) {
return MaterialApp(
home: Scaffold(
appBar: AppBar(
title: const Text('Plugin example app'),
),
body: MyAppBody(key: _myAppKey)),
);
}
void addError(String msg) {
_myAppKey.currentState?.messages.add(Message.error(msg));
}
}
class MyAppBody extends StatefulWidget {
const MyAppBody({required Key key}) : super(key: key);
@override
_MyAppBodyState createState() {
return _MyAppBodyState();
}
}
enum MessageType { info, error }
class Message {
final String content;
final MessageType type;
Message(this.type, this.content);
static info(String content) {
return Message(MessageType.info, content);
}
static error(String content) {
return Message(MessageType.error, content);
}
}
class _MyAppBodyState extends State<MyAppBody> {
static const List<int> getCardSerialNumberCommand = [
0xFF,
0xCA,
0x00,
0x00,
0x00
];
final ScrollController _scrollController = ScrollController();
final List<Message> messages = [];
@override
void initState() {
super.initState();
getCardSerialNumber();
}
Future<void> getCardSerialNumber() async {
int ctx = await Pcsc.establishContext(PcscSCope.user);
CardStruct? card;
try {
List<String> readers = await Pcsc.listReaders(ctx);
if (readers.isEmpty) {
messages.add(Message.error('Could not detect any reader'));
} else {
String reader = readers[0];
setState(() {
messages.add(Message.info('Using reader: $reader'));
});
card = await Pcsc.cardConnect(
ctx, reader, PcscShare.shared, PcscProtocol.any);
var response = await Pcsc.transmit(card, getCardSerialNumberCommand);
var sw = response.sublist(response.length - 2);
var sn = response.sublist(0, response.length - 2);
if (sw[0] != 0x90 || sw[1] != 0x00) {
setState(() {
messages
.add(Message.error('Card returned an error: ${hexDump(sw)}'));
});
} else {
setState(() {
messages.add(Message.info('Card Serial Number is: ${hexDump(sn)}'));
messages.add(Message.info('Done'));
});
}
}
} finally {
if (card != null) {
try {
await Pcsc.cardDisconnect(card.hCard, PcscDisposition.resetCard);
} on Exception catch (e) {
messages.add(Message.error(e.toString()));
}
}
try {
await Pcsc.releaseContext(ctx);
} on Exception catch (e) {
messages.add(Message.error(e.toString()));
}
}
}
static String hexDump(List<int> csn) {
return csn
.map((i) => i.toRadixString(16).padLeft(2, '0').toUpperCase())
.join(' ');
}
_scrollToBottom() {
_scrollController.jumpTo(_scrollController.position.maxScrollExtent);
}
@override
Widget build(BuildContext context) {
TextStyle errorStyle = const TextStyle(color: Colors.red);
WidgetsBinding.instance?.addPostFrameCallback((_) => _scrollToBottom());
return Row(crossAxisAlignment: CrossAxisAlignment.start, children: [
Expanded(
child: Column(children: [
Expanded(
child: ListView(
controller: _scrollController,
children: messages
.map((e) => Text(e.content,
style: e.type == MessageType.error ? errorStyle : null))
.toList())),
Container(
margin: const EdgeInsets.all(10),
child: ElevatedButton(
onPressed: () async {
await tryAgain();
},
child: const Text("Try again")))
]))
]);
}
tryAgain() async {
messages.clear();
await getCardSerialNumber();
}
}