This commit is contained in:
2025-09-16 17:41:53 +07:00
commit be2ad0a8fd
78 changed files with 3275 additions and 0 deletions

View File

@@ -0,0 +1,190 @@
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
class DataScreen extends StatelessWidget {
final List<String> scannedHistory;
const DataScreen({
super.key,
required this.scannedHistory,
});
void _copyToClipboard(BuildContext context, String text) {
Clipboard.setData(ClipboardData(text: text));
ScaffoldMessenger.of(context).showSnackBar(
const SnackBar(
content: Text('Copied to clipboard'),
duration: Duration(seconds: 2),
),
);
}
void _shareAllData(BuildContext context) {
final allData = scannedHistory.join('\n');
Clipboard.setData(ClipboardData(text: allData));
ScaffoldMessenger.of(context).showSnackBar(
const SnackBar(
content: Text('All data copied to clipboard'),
duration: Duration(seconds: 2),
),
);
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: const Text('Scanned Data'),
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
actions: [
IconButton(
icon: const Icon(Icons.share),
onPressed: scannedHistory.isEmpty
? null
: () => _shareAllData(context),
tooltip: 'Copy all data',
),
],
),
body: scannedHistory.isEmpty
? const Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
Icons.inbox,
size: 80,
color: Colors.grey,
),
SizedBox(height: 20),
Text(
'No scanned data yet',
style: TextStyle(
fontSize: 18,
color: Colors.grey,
),
),
SizedBox(height: 10),
Text(
'Start scanning barcodes from the home screen',
style: TextStyle(
fontSize: 14,
color: Colors.grey,
),
),
],
),
)
: Column(
children: [
Container(
padding: const EdgeInsets.all(16),
color: Colors.blue[50],
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
'Total Items: ${scannedHistory.length}',
style: const TextStyle(
fontSize: 16,
fontWeight: FontWeight.bold,
),
),
ElevatedButton.icon(
onPressed: () => _shareAllData(context),
icon: const Icon(Icons.copy, size: 18),
label: const Text('Copy All'),
style: ElevatedButton.styleFrom(
padding: const EdgeInsets.symmetric(
horizontal: 12,
vertical: 8,
),
),
),
],
),
),
Expanded(
child: ListView.builder(
padding: const EdgeInsets.all(16),
itemCount: scannedHistory.length,
itemBuilder: (context, index) {
final item = scannedHistory[index];
return Card(
elevation: 2,
margin: const EdgeInsets.only(bottom: 12),
child: ListTile(
leading: Container(
width: 40,
height: 40,
decoration: BoxDecoration(
color: Theme.of(context).colorScheme.primary,
borderRadius: BorderRadius.circular(20),
),
child: Center(
child: Text(
'${index + 1}',
style: const TextStyle(
color: Colors.white,
fontWeight: FontWeight.bold,
),
),
),
),
title: Text(
item,
style: const TextStyle(
fontSize: 16,
fontWeight: FontWeight.w500,
),
),
subtitle: Text(
'Scanned item #${index + 1}',
style: TextStyle(
color: Colors.grey[600],
),
),
trailing: Row(
mainAxisSize: MainAxisSize.min,
children: [
IconButton(
icon: const Icon(Icons.copy, size: 20),
onPressed: () => _copyToClipboard(context, item),
tooltip: 'Copy',
),
Container(
padding: const EdgeInsets.symmetric(
horizontal: 8,
vertical: 4,
),
decoration: BoxDecoration(
color: Colors.green[100],
borderRadius: BorderRadius.circular(12),
),
child: Text(
'Active',
style: TextStyle(
color: Colors.green[800],
fontSize: 12,
fontWeight: FontWeight.bold,
),
),
),
],
),
),
);
},
),
),
],
),
floatingActionButton: FloatingActionButton.extended(
onPressed: () => Navigator.pop(context),
icon: const Icon(Icons.qr_code_scanner),
label: const Text('Scan More'),
backgroundColor: Theme.of(context).colorScheme.primary,
),
);
}
}

View File

@@ -0,0 +1,304 @@
import 'package:flutter/material.dart';
import 'package:mobile_scanner/mobile_scanner.dart';
import 'data_screen.dart';
class HomeScreen extends StatefulWidget {
const HomeScreen({super.key});
@override
State<HomeScreen> createState() => _HomeScreenState();
}
class _HomeScreenState extends State<HomeScreen> {
String? scannedData;
List<String> scannedHistory = [];
void _navigateToDataScreen() {
Navigator.push(
context,
MaterialPageRoute(
builder: (context) => DataScreen(scannedHistory: scannedHistory),
),
);
}
void _openScanner() {
Navigator.push(
context,
MaterialPageRoute(
builder: (context) => ScannerScreen(
onScan: (String code) {
setState(() {
scannedData = code;
scannedHistory.add(code);
});
},
),
),
);
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: const Text('Barcode Scanner'),
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
),
body: Column(
children: [
Expanded(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Container(
padding: const EdgeInsets.all(20),
child: ElevatedButton.icon(
onPressed: _openScanner,
icon: const Icon(Icons.qr_code_scanner, size: 32),
label: const Text(
'Scan Barcode',
style: TextStyle(fontSize: 18),
),
style: ElevatedButton.styleFrom(
padding: const EdgeInsets.symmetric(
horizontal: 40,
vertical: 15,
),
backgroundColor: Theme.of(context).colorScheme.primary,
foregroundColor: Colors.white,
),
),
),
const SizedBox(height: 30),
if (scannedData != null)
Container(
margin: const EdgeInsets.symmetric(horizontal: 20),
padding: const EdgeInsets.all(20),
decoration: BoxDecoration(
color: Colors.grey[100],
borderRadius: BorderRadius.circular(10),
border: Border.all(color: Colors.grey[300]!),
),
child: Column(
children: [
const Text(
'Last Scanned:',
style: TextStyle(
fontSize: 16,
fontWeight: FontWeight.bold,
),
),
const SizedBox(height: 10),
Text(
scannedData!,
style: const TextStyle(fontSize: 18),
textAlign: TextAlign.center,
),
],
),
),
const SizedBox(height: 20),
if (scannedHistory.isNotEmpty)
Expanded(
child: Container(
margin: const EdgeInsets.symmetric(horizontal: 20),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
'Scan History (${scannedHistory.length} items)',
style: const TextStyle(
fontSize: 16,
fontWeight: FontWeight.bold,
),
),
const SizedBox(height: 10),
Expanded(
child: ListView.builder(
itemCount: scannedHistory.length,
itemBuilder: (context, index) {
final reversedIndex =
scannedHistory.length - 1 - index;
return Card(
child: ListTile(
leading: CircleAvatar(
child: Text('${reversedIndex + 1}'),
),
title: Text(scannedHistory[reversedIndex]),
subtitle: Text('Item ${reversedIndex + 1}'),
),
);
},
),
),
],
),
),
),
],
),
),
Container(
padding: const EdgeInsets.all(20),
decoration: BoxDecoration(
color: Colors.white,
boxShadow: [
BoxShadow(
color: Colors.grey.withOpacity(0.3),
spreadRadius: 1,
blurRadius: 5,
offset: const Offset(0, -3),
),
],
),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
ElevatedButton.icon(
onPressed: scannedHistory.isEmpty ? null : () {
setState(() {
scannedHistory.clear();
scannedData = null;
});
},
icon: const Icon(Icons.clear_all),
label: const Text('Clear'),
style: ElevatedButton.styleFrom(
backgroundColor: Colors.red,
foregroundColor: Colors.white,
),
),
ElevatedButton.icon(
onPressed: scannedHistory.isEmpty ? null : _navigateToDataScreen,
icon: const Icon(Icons.list),
label: const Text('View All Data'),
style: ElevatedButton.styleFrom(
backgroundColor: Colors.green,
foregroundColor: Colors.white,
),
),
],
),
),
],
),
);
}
}
class ScannerScreen extends StatefulWidget {
final Function(String) onScan;
const ScannerScreen({super.key, required this.onScan});
@override
State<ScannerScreen> createState() => _ScannerScreenState();
}
class _ScannerScreenState extends State<ScannerScreen> {
MobileScannerController cameraController = MobileScannerController(
formats: [BarcodeFormat.code128]
);
bool hasScanned = false;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: const Text('Scan Barcode'),
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
actions: [
// IconButton(
// icon: ValueListenableBuilder(
// valueListenable: cameraController.torchStateNotifier,
// builder: (context, state, child) {
// switch (state) {
// case TorchState.off:
// return const Icon(Icons.flash_off, color: Colors.grey);
// case TorchState.on:
// return const Icon(Icons.flash_on, color: Colors.yellow);
// }
// },
// ),
// onPressed: () => cameraController.toggleTorch(),
// ),
// IconButton(
// icon: ValueListenableBuilder(
// valueListenable: cameraController.cameraFacingState,
// builder: (context, state, child) {
// switch (state) {
// case CameraFacing.front:
// return const Icon(Icons.camera_front);
// case CameraFacing.back:
// return const Icon(Icons.camera_rear);
// }
// },
// ),
// onPressed: () => cameraController.switchCamera(),
// ),
],
),
body: Stack(
children: [
MobileScanner(
controller: cameraController,
onDetect: (capture) {
if (!hasScanned) {
final List<Barcode> barcodes = capture.barcodes;
for (final barcode in barcodes) {
if (barcode.rawValue != null) {
hasScanned = true;
widget.onScan(barcode.rawValue!);
Navigator.pop(context);
break;
}
}
}
},
),
Center(
child: Container(
width: 300,
height: 300,
decoration: BoxDecoration(
border: Border.all(
color: Colors.green,
width: 2,
),
borderRadius: BorderRadius.circular(12),
),
),
),
Positioned(
bottom: 100,
left: 0,
right: 0,
child: Center(
child: Container(
padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 10),
decoration: BoxDecoration(
color: Colors.black54,
borderRadius: BorderRadius.circular(8),
),
child: const Text(
'Align barcode within the frame',
style: TextStyle(
color: Colors.white,
fontSize: 16,
),
),
),
),
),
],
),
);
}
@override
void dispose() {
cameraController.dispose();
super.dispose();
}
}