This commit is contained in:
sjk
2025-11-17 13:39:05 +08:00
commit d4cfe2b9de
479 changed files with 109324 additions and 0 deletions

View File

@@ -0,0 +1,437 @@
import 'package:flutter/material.dart';
import 'package:provider/provider.dart';
import 'dart:async';
import '../models/writing_task.dart';
import '../models/writing_submission.dart';
import '../providers/writing_provider.dart';
class WritingTaskScreen extends StatefulWidget {
final WritingTask task;
const WritingTaskScreen({
super.key,
required this.task,
});
@override
State<WritingTaskScreen> createState() => _WritingTaskScreenState();
}
class _WritingTaskScreenState extends State<WritingTaskScreen> {
final TextEditingController _contentController = TextEditingController();
final ScrollController _scrollController = ScrollController();
Timer? _timer;
int _elapsedSeconds = 0;
bool _isSubmitting = false;
bool _showInstructions = true;
int _wordCount = 0;
@override
void initState() {
super.initState();
_startTimer();
_contentController.addListener(_updateWordCount);
}
@override
void dispose() {
_timer?.cancel();
_contentController.dispose();
_scrollController.dispose();
super.dispose();
}
void _startTimer() {
_timer = Timer.periodic(const Duration(seconds: 1), (timer) {
setState(() {
_elapsedSeconds++;
});
// 检查时间限制
if (widget.task.timeLimit != null &&
_elapsedSeconds >= widget.task.timeLimit! * 60) {
_showTimeUpDialog();
}
});
}
void _updateWordCount() {
final text = _contentController.text;
final words = text.trim().split(RegExp(r'\s+'));
setState(() {
_wordCount = text.trim().isEmpty ? 0 : words.length;
});
}
void _showTimeUpDialog() {
showDialog(
context: context,
barrierDismissible: false,
builder: (context) => AlertDialog(
title: const Text('时间到!'),
content: const Text('写作时间已结束,请提交您的作品。'),
actions: [
TextButton(
onPressed: () {
Navigator.of(context).pop();
_submitWriting();
},
child: const Text('提交'),
),
],
),
);
}
Future<void> _submitWriting() async {
if (_isSubmitting) return;
final content = _contentController.text.trim();
if (content.isEmpty) {
ScaffoldMessenger.of(context).showSnackBar(
const SnackBar(content: Text('请输入写作内容')),
);
return;
}
setState(() {
_isSubmitting = true;
});
try {
final provider = Provider.of<WritingProvider>(context, listen: false);
// 首先开始任务(如果还没有开始)
if (provider.currentSubmission == null) {
await provider.startTask(widget.task.id);
}
// 更新内容
provider.updateContent(content);
provider.updateTimeSpent(_elapsedSeconds);
// 提交写作
final success = await provider.submitWriting();
if (mounted && success) {
Navigator.of(context).pop(true);
ScaffoldMessenger.of(context).showSnackBar(
const SnackBar(content: Text('写作已提交,正在批改中...')),
);
} else if (mounted && !success) {
ScaffoldMessenger.of(context).showSnackBar(
const SnackBar(content: Text('提交失败,请重试')),
);
}
} catch (e) {
if (mounted) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(content: Text('提交失败: $e')),
);
}
} finally {
if (mounted) {
setState(() {
_isSubmitting = false;
});
}
}
}
String _formatTime(int seconds) {
final minutes = seconds ~/ 60;
final remainingSeconds = seconds % 60;
return '${minutes.toString().padLeft(2, '0')}:${remainingSeconds.toString().padLeft(2, '0')}';
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text(widget.task.title),
actions: [
IconButton(
icon: Icon(_showInstructions ? Icons.visibility_off : Icons.visibility),
onPressed: () {
setState(() {
_showInstructions = !_showInstructions;
});
},
),
IconButton(
icon: const Icon(Icons.help_outline),
onPressed: _showHelpDialog,
),
],
),
body: Column(
children: [
// 状态栏
Container(
padding: const EdgeInsets.all(16),
decoration: BoxDecoration(
color: Colors.grey[100],
border: Border(
bottom: BorderSide(color: Colors.grey[300]!),
),
),
child: Row(
children: [
Icon(
Icons.timer,
size: 20,
color: _getTimeColor(),
),
const SizedBox(width: 8),
Text(
_formatTime(_elapsedSeconds),
style: TextStyle(
fontSize: 16,
fontWeight: FontWeight.bold,
color: _getTimeColor(),
),
),
if (widget.task.timeLimit != null)
Text(
' / ${widget.task.timeLimit}分钟',
style: const TextStyle(fontSize: 14, color: Colors.grey),
),
const Spacer(),
Icon(
Icons.text_fields,
size: 20,
color: _getWordCountColor(_wordCount),
),
const SizedBox(width: 8),
Text(
'$_wordCount',
style: TextStyle(
fontSize: 16,
fontWeight: FontWeight.bold,
color: _getWordCountColor(_wordCount),
),
),
if (widget.task.wordLimit != null)
Text(
' / ${widget.task.wordLimit}',
style: const TextStyle(fontSize: 14, color: Colors.grey),
),
],
),
),
// 任务说明
if (_showInstructions)
Container(
width: double.infinity,
padding: const EdgeInsets.all(16),
decoration: BoxDecoration(
color: Colors.blue[50],
border: Border(
bottom: BorderSide(color: Colors.grey[300]!),
),
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
children: [
Icon(
Icons.assignment,
color: Colors.blue[700],
size: 20,
),
const SizedBox(width: 8),
Text(
'任务要求',
style: TextStyle(
fontSize: 16,
fontWeight: FontWeight.bold,
color: Colors.blue[700],
),
),
],
),
const SizedBox(height: 8),
Text(
widget.task.description,
style: const TextStyle(fontSize: 14),
),
if (widget.task.requirements.isNotEmpty) ...[
const SizedBox(height: 12),
const Text(
'具体要求:',
style: TextStyle(
fontSize: 14,
fontWeight: FontWeight.w600,
),
),
const SizedBox(height: 4),
...widget.task.requirements.map((req) => Padding(
padding: const EdgeInsets.only(left: 16, bottom: 2),
child: Text(
'$req',
style: const TextStyle(fontSize: 13),
),
)).toList(),
],
if (widget.task.keywords.isNotEmpty) ...[
const SizedBox(height: 12),
const Text(
'关键词:',
style: TextStyle(
fontSize: 14,
fontWeight: FontWeight.w600,
),
),
const SizedBox(height: 4),
Wrap(
spacing: 8,
children: widget.task.keywords.map((keyword) => Chip(
label: Text(
keyword,
style: const TextStyle(fontSize: 12),
),
backgroundColor: Colors.blue[100],
materialTapTargetSize: MaterialTapTargetSize.shrinkWrap,
)).toList(),
),
],
if (widget.task.prompt != null && widget.task.prompt!.isNotEmpty) ...[
const SizedBox(height: 12),
const Text(
'提示:',
style: TextStyle(
fontSize: 14,
fontWeight: FontWeight.w600,
),
),
const SizedBox(height: 4),
Padding(
padding: const EdgeInsets.only(left: 16, bottom: 2),
child: Text(
'💡 ${widget.task.prompt}',
style: const TextStyle(fontSize: 13),
),
),
],
],
),
),
// 写作区域
Expanded(
child: Padding(
padding: const EdgeInsets.all(16),
child: TextField(
controller: _contentController,
maxLines: null,
expands: true,
decoration: const InputDecoration(
hintText: '请在此处开始写作...',
border: OutlineInputBorder(),
contentPadding: EdgeInsets.all(16),
),
style: const TextStyle(fontSize: 16, height: 1.5),
),
),
),
],
),
bottomNavigationBar: Container(
padding: const EdgeInsets.all(16),
decoration: BoxDecoration(
color: Colors.white,
boxShadow: [
BoxShadow(
color: Colors.grey.withOpacity(0.2),
spreadRadius: 1,
blurRadius: 3,
offset: const Offset(0, -1),
),
],
),
child: Row(
children: [
Expanded(
child: OutlinedButton(
onPressed: () => Navigator.of(context).pop(),
child: const Text('保存草稿'),
),
),
const SizedBox(width: 16),
Expanded(
child: ElevatedButton(
onPressed: _isSubmitting ? null : _submitWriting,
child: _isSubmitting
? const SizedBox(
width: 20,
height: 20,
child: CircularProgressIndicator(strokeWidth: 2),
)
: const Text('提交'),
),
),
],
),
),
);
}
Color _getTimeColor() {
if (widget.task.timeLimit == null) return Colors.blue;
final remainingMinutes = (widget.task.timeLimit! * 60 - _elapsedSeconds) / 60;
if (remainingMinutes <= 5) return Colors.red;
if (remainingMinutes <= 10) return Colors.orange;
return Colors.blue;
}
Color _getWordCountColor(int wordCount) {
if (widget.task.wordLimit == null) return Colors.green;
final ratio = wordCount / widget.task.wordLimit!;
if (ratio > 1.1) return Colors.red;
if (ratio > 0.9) return Colors.green;
if (ratio > 0.5) return Colors.orange;
return Colors.grey;
}
void _showHelpDialog() {
showDialog(
context: context,
builder: (context) => AlertDialog(
title: const Text('写作帮助'),
content: const SingleChildScrollView(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
mainAxisSize: MainAxisSize.min,
children: [
Text(
'写作技巧:',
style: TextStyle(fontWeight: FontWeight.bold),
),
SizedBox(height: 8),
Text('• 仔细阅读任务要求,确保理解题意'),
Text('• 合理安排时间,留出检查和修改的时间'),
Text('• 注意文章结构,包括开头、主体和结尾'),
Text('• 使用多样化的词汇和句式'),
Text('• 检查语法、拼写和标点符号'),
SizedBox(height: 12),
Text(
'评分标准:',
style: TextStyle(fontWeight: FontWeight.bold),
),
SizedBox(height: 8),
Text('• 内容相关性和完整性'),
Text('• 语言准确性和流畅性'),
Text('• 词汇丰富度和语法复杂性'),
Text('• 文章结构和逻辑性'),
],
),
),
actions: [
TextButton(
onPressed: () => Navigator.of(context).pop(),
child: const Text('知道了'),
),
],
),
);
}
}