import 'dart:ui'; import 'package:flutter/material.dart'; import 'package:provider/provider.dart'; import 'package:flutter/services.dart'; // 复制功能需要 import 'package:win_text_editor/app/providers/editor_provider.dart'; import 'package:file_picker/file_picker.dart'; import 'dart:io'; class TextTab extends StatefulWidget { final String tabId; const TextTab({super.key, required this.tabId}); @override State createState() => _TextTabState(); } class _TextTabState extends State { late TextEditingController _controller; late EditorProvider _provider; late FocusNode _focusNode; late ScrollController _scrollController; @override @override void initState() { super.initState(); _provider = Provider.of(context, listen: false); _controller = TextEditingController(text: _getCurrentContent()); _focusNode = FocusNode(); _scrollController = ScrollController(); WidgetsBinding.instance.addPostFrameCallback((_) { FocusScope.of(context).requestFocus(_focusNode); }); } String _getCurrentContent() { return _provider.tabs.firstWhere((t) => t.id == widget.tabId).content; } @override void didUpdateWidget(TextTab oldWidget) { super.didUpdateWidget(oldWidget); if (oldWidget.tabId != widget.tabId) { _controller.text = _getCurrentContent(); } } @override void dispose() { _controller.dispose(); _focusNode.dispose(); _scrollController.dispose(); // 添加这行 super.dispose(); } @override Widget build(BuildContext context) { final tab = _provider.tabs.firstWhere((t) => t.id == widget.tabId); return Column( children: [ Container( height: 40, padding: const EdgeInsets.symmetric(horizontal: 16), decoration: BoxDecoration( color: Colors.grey[100], border: Border(bottom: BorderSide(color: Colors.grey[300]!)), ), child: Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ Text( '源文本${tab.content.isEmpty ? '' : ' (${tab.fileName!.isEmpty ? '' : '${tab.fileName},'}${tab.content.length}字符)'}', style: const TextStyle(fontWeight: FontWeight.bold), ), Row( children: [ IconButton( icon: const Icon(Icons.folder_open, size: 20), tooltip: '打开文件', onPressed: () => _openFile(context), ), IconButton( icon: const Icon(Icons.content_copy, size: 20), tooltip: '复制内容', onPressed: tab.content.isEmpty ? null : () => _copyToClipboard(context, tab.content), ), IconButton( icon: const Icon(Icons.save, size: 20), tooltip: '保存到文件', onPressed: tab.content.isEmpty ? null : () => _saveFile(context, tab.content), ), ], ), ], ), ), Expanded( child: ScrollConfiguration( behavior: ScrollConfiguration.of(context).copyWith( scrollbars: true, dragDevices: {PointerDeviceKind.touch, PointerDeviceKind.mouse}, ), child: SingleChildScrollView( controller: _scrollController, // 添加滚动控制器 child: Stack( children: [ // 实际可编辑的TextField TextField( controller: _controller, focusNode: _focusNode, maxLines: null, onChanged: (text) => _provider.updateContent(widget.tabId, text, tab.fileName), decoration: const InputDecoration( border: InputBorder.none, contentPadding: EdgeInsets.all(16), ), style: TextStyle( fontFamily: 'monospace', fontSize: 14, color: Theme.of(context).textTheme.bodyLarge?.color, // 使用主题文本颜色 ), ), ], ), ), ), ), ], ); } Future _openFile(BuildContext context) async { try { final result = await FilePicker.platform.pickFiles(type: FileType.any, allowMultiple: false); if (result != null && result.files.single.path != null) { final file = File(result.files.single.path!); final content = await file.readAsString(); // 更新provider和控制器 _provider.updateContent(widget.tabId, content, result.files.first.name); _controller.text = content; // 确保更新控制器 if (context.mounted) { ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('已加载: ${file.path}'))); } } } on FormatException { ScaffoldMessenger.of(context).showSnackBar(const SnackBar(content: Text('这不是可读的文本文件'))); } on FileSystemException catch (e) { ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('文件访问错误: ${e.message}'))); } catch (e) { if (context.mounted) { ScaffoldMessenger.of( context, ).showSnackBar(SnackBar(content: Text('读取失败: ${e.toString()}'))); } } } // 复制到剪贴板 Future _copyToClipboard(BuildContext context, String content) async { await Clipboard.setData(ClipboardData(text: content)); if (context.mounted) { ScaffoldMessenger.of(context).showSnackBar(const SnackBar(content: Text('已复制到剪贴板'))); } } // 保存文件模拟功能 // 保存文件功能 Future _saveFile(BuildContext context, String content) async { try { // 让用户选择保存位置 String? outputPath = await FilePicker.platform.saveFile( dialogTitle: '保存文件', fileName: 'untitled.txt', allowedExtensions: ['txt'], type: FileType.any, ); if (outputPath == null) return; final file = File(outputPath); // 检查文件是否存在 if (await file.exists()) { // 如果存在,显示确认对话框 final shouldOverwrite = await showDialog( context: context, builder: (context) => AlertDialog( title: const Text('文件已存在'), content: const Text('要覆盖现有文件吗?'), actions: [ TextButton( onPressed: () => Navigator.pop(context, false), child: const Text('取消'), ), TextButton( onPressed: () => Navigator.pop(context, true), child: const Text('覆盖'), ), ], ), ); if (shouldOverwrite != true) return; } // 写入文件 await file.writeAsString(content); if (context.mounted) { ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('已保存到: ${file.path}'))); } } on FileSystemException catch (e) { if (context.mounted) { ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text('保存失败: ${e.message}'))); } } catch (e) { if (context.mounted) { ScaffoldMessenger.of( context, ).showSnackBar(SnackBar(content: Text('保存失败: ${e.toString()}'))); } } } }