2025-08-13 12:55:27 +07:00

471 lines
13 KiB
Dart

import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:image_picker/image_picker.dart';
import 'dart:io';
import '../../../common/theme/theme.dart';
// Model untuk question item
class TaskQuestion {
final String id;
final String question;
bool? answer;
File? photo;
TaskQuestion({
required this.id,
required this.question,
this.answer,
this.photo,
});
}
// Model untuk task section
class TaskSection {
final String title;
final List<TaskQuestion> questions;
TaskSection({required this.title, required this.questions});
}
@RoutePage()
class DailyTasksFormPage extends StatefulWidget {
const DailyTasksFormPage({super.key});
@override
State<DailyTasksFormPage> createState() => _DailyTasksFormPageState();
}
class _DailyTasksFormPageState extends State<DailyTasksFormPage>
with SingleTickerProviderStateMixin {
late TabController _tabController;
final ImagePicker _picker = ImagePicker();
// Sample data untuk OPEN dan CLOSING tasks
List<TaskSection> taskSections = [
TaskSection(
title: "OPENING",
questions: [
TaskQuestion(id: "open_1", question: "Apakah meja sudah dibersihkan?"),
TaskQuestion(
id: "open_2",
question: "Apakah alat kerja sudah disiapkan?",
),
TaskQuestion(
id: "open_3",
question: "Apakah ruangan sudah dalam kondisi bersih?",
),
TaskQuestion(
id: "open_4",
question: "Apakah penerangan sudah memadai?",
),
],
),
TaskSection(
title: "CLOSING",
questions: [
TaskQuestion(
id: "close_1",
question: "Apakah meja sudah dibersihkan kembali?",
),
TaskQuestion(
id: "close_2",
question: "Apakah alat kerja sudah disimpan dengan rapi?",
),
TaskQuestion(id: "close_3", question: "Apakah lampu sudah dimatikan?"),
TaskQuestion(id: "close_4", question: "Apakah pintu sudah dikunci?"),
],
),
];
@override
void initState() {
super.initState();
_tabController = TabController(length: 2, vsync: this);
}
@override
void dispose() {
_tabController.dispose();
super.dispose();
}
Future<void> _pickImage(TaskQuestion question) async {
try {
final XFile? image = await _picker.pickImage(
source: ImageSource.camera,
maxWidth: 1920,
maxHeight: 1080,
imageQuality: 85,
);
if (image != null) {
setState(() {
question.photo = File(image.path);
});
}
} catch (e) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text('Error mengambil foto: $e'),
backgroundColor: AppColor.error,
),
);
}
}
void _removePhoto(TaskQuestion question) {
setState(() {
question.photo = null;
});
}
bool _isFormComplete() {
for (var section in taskSections) {
for (var question in section.questions) {
if (question.answer == null || question.photo == null) {
return false;
}
}
}
return true;
}
void _submitForm() {
if (!_isFormComplete()) {
ScaffoldMessenger.of(context).showSnackBar(
const SnackBar(
content: Text('Mohon lengkapi semua pertanyaan dan foto'),
backgroundColor: AppColor.error,
),
);
return;
}
// Logic untuk submit form
ScaffoldMessenger.of(context).showSnackBar(
const SnackBar(
content: Text('Daily tasks berhasil disimpan!'),
backgroundColor: AppColor.success,
),
);
}
@override
Widget build(BuildContext context) {
return Scaffold(
backgroundColor: AppColor.background,
appBar: AppBar(
title: const Text(
'Daily Tasks',
style: TextStyle(
color: AppColor.textWhite,
fontWeight: FontWeight.bold,
),
),
backgroundColor: AppColor.primary,
elevation: 0,
bottom: TabBar(
controller: _tabController,
indicatorColor: AppColor.textWhite,
labelColor: AppColor.textWhite,
unselectedLabelColor: AppColor.textWhite.withOpacity(0.7),
labelStyle: const TextStyle(fontWeight: FontWeight.bold),
tabs: taskSections
.map((section) => Tab(text: section.title))
.toList(),
),
),
body: Column(
children: [
Expanded(
child: TabBarView(
controller: _tabController,
children: taskSections.map((section) {
return SingleChildScrollView(
padding: const EdgeInsets.all(16),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
_buildSectionHeader(section.title),
const SizedBox(height: 16),
...section.questions.map((question) {
return _buildQuestionCard(question);
}).toList(),
],
),
);
}).toList(),
),
),
_buildSubmitButton(),
],
),
);
}
Widget _buildSectionHeader(String title) {
return Container(
padding: const EdgeInsets.symmetric(vertical: 12, horizontal: 16),
decoration: BoxDecoration(
gradient: LinearGradient(
colors: [AppColor.primary, AppColor.primaryLight],
begin: Alignment.centerLeft,
end: Alignment.centerRight,
),
borderRadius: BorderRadius.circular(12),
),
child: Row(
children: [
Icon(
title == "OPENING" ? Icons.wb_sunny : Icons.nightlight_round,
color: AppColor.textWhite,
size: 24,
),
const SizedBox(width: 12),
Text(
'$title CHECKLIST',
style: const TextStyle(
color: AppColor.textWhite,
fontSize: 18,
fontWeight: FontWeight.bold,
),
),
],
),
);
}
Widget _buildQuestionCard(TaskQuestion question) {
return Container(
margin: const EdgeInsets.only(bottom: 16),
decoration: BoxDecoration(
color: AppColor.white,
borderRadius: BorderRadius.circular(12),
boxShadow: [
BoxShadow(
color: AppColor.black.withOpacity(0.05),
blurRadius: 8,
offset: const Offset(0, 2),
),
],
),
child: Padding(
padding: const EdgeInsets.all(16),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
question.question,
style: const TextStyle(
fontSize: 16,
fontWeight: FontWeight.w600,
color: AppColor.textPrimary,
),
),
const SizedBox(height: 16),
// Yes/No buttons
Row(
children: [
Expanded(
child: _buildAnswerButton(
question: question,
value: true,
label: "YES",
icon: Icons.check_circle,
color: AppColor.success,
),
),
const SizedBox(width: 12),
Expanded(
child: _buildAnswerButton(
question: question,
value: false,
label: "NO",
icon: Icons.cancel,
color: AppColor.error,
),
),
],
),
const SizedBox(height: 16),
// Photo section
_buildPhotoSection(question),
],
),
),
);
}
Widget _buildAnswerButton({
required TaskQuestion question,
required bool value,
required String label,
required IconData icon,
required Color color,
}) {
bool isSelected = question.answer == value;
return GestureDetector(
onTap: () {
setState(() {
question.answer = value;
});
},
child: Container(
padding: const EdgeInsets.symmetric(vertical: 12, horizontal: 16),
decoration: BoxDecoration(
color: isSelected ? color : AppColor.backgroundLight,
border: Border.all(
color: isSelected ? color : AppColor.border,
width: 1.5,
),
borderRadius: BorderRadius.circular(8),
),
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(
icon,
color: isSelected ? AppColor.white : AppColor.textSecondary,
size: 20,
),
const SizedBox(width: 8),
Text(
label,
style: TextStyle(
color: isSelected ? AppColor.white : AppColor.textSecondary,
fontWeight: FontWeight.w600,
),
),
],
),
),
);
}
Widget _buildPhotoSection(TaskQuestion question) {
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
children: [
const Icon(
Icons.camera_alt,
color: AppColor.textSecondary,
size: 18,
),
const SizedBox(width: 8),
const Text(
'Foto Bukti',
style: TextStyle(
fontSize: 14,
fontWeight: FontWeight.w600,
color: AppColor.textSecondary,
),
),
const Text(' *', style: TextStyle(color: AppColor.error)),
],
),
const SizedBox(height: 8),
if (question.photo == null)
GestureDetector(
onTap: () => _pickImage(question),
child: Container(
height: 120,
width: double.infinity,
decoration: BoxDecoration(
color: AppColor.borderLight,
borderRadius: BorderRadius.circular(8),
border: Border.all(
color: AppColor.border,
style: BorderStyle.solid,
),
),
child: const Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(Icons.add_a_photo, size: 32, color: AppColor.textLight),
SizedBox(height: 8),
Text(
'Tap untuk mengambil foto',
style: TextStyle(color: AppColor.textLight, fontSize: 14),
),
],
),
),
)
else
Stack(
children: [
ClipRRect(
borderRadius: BorderRadius.circular(8),
child: Image.file(
question.photo!,
height: 200,
width: double.infinity,
fit: BoxFit.cover,
),
),
Positioned(
top: 8,
right: 8,
child: GestureDetector(
onTap: () => _removePhoto(question),
child: Container(
padding: const EdgeInsets.all(4),
decoration: const BoxDecoration(
color: AppColor.error,
shape: BoxShape.circle,
),
child: const Icon(
Icons.close,
color: AppColor.white,
size: 16,
),
),
),
),
],
),
],
);
}
Widget _buildSubmitButton() {
return Container(
padding: const EdgeInsets.all(16),
decoration: const BoxDecoration(
color: AppColor.white,
border: Border(top: BorderSide(color: AppColor.border, width: 1)),
),
child: SizedBox(
width: double.infinity,
child: ElevatedButton(
onPressed: _submitForm,
style: ElevatedButton.styleFrom(
backgroundColor: AppColor.primary,
padding: const EdgeInsets.symmetric(vertical: 16),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(8),
),
elevation: 2,
),
child: const Text(
'SUBMIT DAILY TASKS',
style: TextStyle(
color: AppColor.textWhite,
fontSize: 16,
fontWeight: FontWeight.bold,
),
),
),
),
);
}
}