Flutter伪终端交互插件dart_pty的使用
Flutter伪终端交互插件dart_pty的使用
dart_pty
Language: 中文简体 | English
创建终端的 dart 实现。
当前已经实现纯 dart 动态调用底层库获得终端,不再需要依赖项目的 so 库
介绍
创建一个伪终端并执行一个子进程。也是终端模拟器的底层实现。
与一些语言提供的执行进程不同,它不仅能够无缓冲地拿到 stdout、stderr 输出,还能拿到所有的终端序列,也能对进程进行交互。
举个简单的例子
我们在 shell 执行以下命令:
echo -e "\033[31m hello \033[0m"
这个时候会看见终端打印了红色的“hello”,因为终端模拟器不仅拿到了字符“hello”,还拿到了前后改变颜色的控制序列。
我们通过 Java 的 runtime.exec
函数,或者 Python 的 system
函数,C 语言的 system
函数,这一系列高级语言所提供的执行进程的函数,我们执行 Python,会发现无论如何也拿不到命令的输出,其原因在于终端与管道的缓冲机制。
所以此项目在于实现获得进程实时的输出流,并提供与进程交互的功能。
使用方法
参考 test.dart
import 'dart:io';
import 'package:dart_pty/src/interface/pseudo_terminal_interface.dart';
Future<void> main() async {
// 设置环境变量
final Map<String, String> environment = {'TEST': 'TEST_VALUE'};
// 指定可执行文件,根据操作系统选择
String executable = 'sh';
if (Platform.isWindows) {
executable = 'wsl'; // 在 Windows 上使用 WSL
}
// 创建伪终端实例
final PseudoTerminal pseudoTerminal = PseudoTerminal(
executable: executable,
environment: environment,
workingDirectory: '/', // 工作目录
);
// 延迟等待伪终端初始化完成
await Future<void>.delayed(const Duration(milliseconds: 100));
// 监听伪终端的输出
pseudoTerminal.out!.listen((line) {
print('\x1b[31m' + '>' * 40); // 红色分隔符
print('\x1b[32m$line'); // 绿色输出
print('\x1b[31m' + '<' * 40); // 红色分隔符
});
// 启动伪终端的轮询
pseudoTerminal.startPolling();
// 启动输入循环
await Future.delayed(const Duration(milliseconds: 100), () async {
while (true) {
// 读取用户输入
final String input = stdin.readLineSync()!;
// 将输入逐字符发送到伪终端
input.split('').forEach((element) {
pseudoTerminal.write(element);
});
// 添加换行符
pseudoTerminal.write('\n');
// 强制伪终端读取数据
pseudoTerminal.schedulingRead();
// 延迟以避免高频率操作
await Future<void>.delayed(const Duration(milliseconds: 100));
}
});
}
更多关于Flutter伪终端交互插件dart_pty的使用的实战系列教程也可以访问 https://www.itying.com/category-92-b0.html
更多关于Flutter伪终端交互插件dart_pty的使用的实战系列教程也可以访问 https://www.itying.com/category-92-b0.html
dart_pty
是一个用于在 Flutter 应用中创建伪终端(PTY)的插件。它允许你在应用中运行命令行程序,并与它们进行交互。这对于需要在应用中嵌入终端模拟器或执行命令行任务的场景非常有用。
安装 dart_pty
首先,你需要在 pubspec.yaml
文件中添加 dart_pty
依赖:
dependencies:
flutter:
sdk: flutter
dart_pty: ^0.0.1 # 请检查最新版本
然后运行 flutter pub get
来安装依赖。
使用 dart_pty
以下是一个简单的示例,展示如何在 Flutter 应用中使用 dart_pty
来创建一个伪终端并与之交互。
import 'package:flutter/material.dart';
import 'package:dart_pty/dart_pty.dart';
import 'dart:io';
void main() {
runApp(MyApp());
}
class MyApp extends StatelessWidget {
[@override](/user/override)
Widget build(BuildContext context) {
return MaterialApp(
home: TerminalScreen(),
);
}
}
class TerminalScreen extends StatefulWidget {
[@override](/user/override)
_TerminalScreenState createState() => _TerminalScreenState();
}
class _TerminalScreenState extends State<TerminalScreen> {
late Pty pty;
late StreamSubscription<List<int>> _outputSubscription;
String output = '';
[@override](/user/override)
void initState() {
super.initState();
_startPty();
}
void _startPty() async {
pty = Pty.start(
shell: Platform.isWindows ? 'cmd.exe' : 'bash',
columns: 80,
rows: 24,
);
_outputSubscription = pty.output.listen((data) {
setState(() {
output += String.fromCharCodes(data);
});
});
}
void _sendCommand(String command) {
pty.write(command + '\n');
}
[@override](/user/override)
void dispose() {
_outputSubscription.cancel();
pty.kill();
super.dispose();
}
[@override](/user/override)
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text('Flutter PTY Example'),
),
body: Column(
children: [
Expanded(
child: SingleChildScrollView(
child: Text(output),
),
),
TextField(
decoration: InputDecoration(
hintText: 'Enter command...',
),
onSubmitted: (command) {
_sendCommand(command);
},
),
],
),
);
}
}