veilid/veilid-flutter/example/lib/main.dart

343 lines
11 KiB
Dart
Raw Normal View History

2022-01-16 11:19:01 -05:00
import 'dart:async';
2022-03-08 22:32:12 -05:00
import 'dart:typed_data';
2022-06-15 21:51:38 -04:00
import 'dart:convert';
2022-12-09 18:59:31 -05:00
import 'dart:io';
2022-01-16 11:19:01 -05:00
import 'package:flutter/material.dart';
2022-12-09 18:59:31 -05:00
import 'package:flutter/foundation.dart';
2022-01-16 11:19:01 -05:00
import 'package:veilid/veilid.dart';
2022-12-09 18:59:31 -05:00
//import 'package:flutter_loggy/flutter_loggy.dart';
2022-03-03 20:45:39 -05:00
import 'package:loggy/loggy.dart';
2022-12-09 18:59:31 -05:00
import 'platform_menu.dart';
import 'package:flutter/foundation.dart';
import 'package:flutter/material.dart';
import 'package:flutter/services.dart';
import 'package:flutter_acrylic/flutter_acrylic.dart';
import 'package:xterm/xterm.dart';
import 'home.dart';
2022-02-15 13:40:17 -05:00
2022-03-03 20:45:39 -05:00
import 'config.dart';
// Loggy tools
2022-09-09 16:27:13 -04:00
const LogLevel traceLevel = LogLevel('Trace', 1);
2022-03-03 20:45:39 -05:00
2022-03-10 09:51:53 -05:00
class ConsolePrinter extends LoggyPrinter {
ConsolePrinter(this.childPrinter) : super();
final LoggyPrinter childPrinter;
@override
void onLog(LogRecord record) {
debugPrint(record.toString());
childPrinter.onLog(record);
}
}
2022-03-03 20:45:39 -05:00
extension TraceLoggy on Loggy {
void trace(dynamic message, [Object? error, StackTrace? stackTrace]) =>
log(traceLevel, message, error, stackTrace);
}
LogOptions getLogOptions(LogLevel? level) {
return LogOptions(
level ?? LogLevel.all,
stackTraceLevel: LogLevel.error,
);
}
2022-06-10 17:07:10 -04:00
VeilidConfigLogLevel convertToVeilidConfigLogLevel(LogLevel? level) {
if (level == null) {
return VeilidConfigLogLevel.off;
}
switch (level) {
case LogLevel.error:
return VeilidConfigLogLevel.error;
case LogLevel.warning:
return VeilidConfigLogLevel.warn;
case LogLevel.info:
return VeilidConfigLogLevel.info;
case LogLevel.debug:
return VeilidConfigLogLevel.debug;
case traceLevel:
return VeilidConfigLogLevel.trace;
}
return VeilidConfigLogLevel.off;
}
2022-03-03 20:45:39 -05:00
void setRootLogLevel(LogLevel? level) {
Loggy('').level = getLogOptions(level);
2022-07-01 12:13:52 -04:00
Veilid.instance.changeLogLevel("all", convertToVeilidConfigLogLevel(level));
2022-03-03 20:45:39 -05:00
}
void initLoggy() {
2022-12-09 18:59:31 -05:00
// Loggy.initLoggy(
// logPrinter: StreamPrinter(ConsolePrinter(
// const PrettyDeveloperPrinter(),
// )),
// logOptions: getLogOptions(null),
// );
}
/////////////////////////////// Acrylic
bool get isDesktop {
if (kIsWeb) return false;
return [
TargetPlatform.windows,
TargetPlatform.linux,
TargetPlatform.macOS,
].contains(defaultTargetPlatform);
}
Future<void> setupAcrylic() async {
await Window.initialize();
await Window.makeTitlebarTransparent();
await Window.setEffect(effect: WindowEffect.aero, color: Color(0xFFFFFFFF));
await Window.setBlurViewState(MacOSBlurViewState.active);
2022-02-15 13:40:17 -05:00
}
2022-12-09 18:59:31 -05:00
/////////////////////////////// Entrypoint
2022-01-16 11:19:01 -05:00
void main() {
2022-03-03 20:45:39 -05:00
WidgetsFlutterBinding.ensureInitialized();
initLoggy();
2022-06-15 21:51:38 -04:00
if (kIsWeb) {
var platformConfig = VeilidWASMConfig(
logging: VeilidWASMConfigLogging(
performance: VeilidWASMConfigLoggingPerformance(
enabled: true,
2022-07-01 12:13:52 -04:00
level: VeilidConfigLogLevel.debug,
2022-06-15 21:51:38 -04:00
logsInTimings: true,
2022-07-01 12:13:52 -04:00
logsInConsole: true),
api: VeilidWASMConfigLoggingApi(
enabled: true, level: VeilidConfigLogLevel.info)));
2022-07-01 16:20:43 -04:00
Veilid.instance.initializeVeilidCore(platformConfig.json);
2022-06-15 21:51:38 -04:00
} else {
var platformConfig = VeilidFFIConfig(
logging: VeilidFFIConfigLogging(
terminal: VeilidFFIConfigLoggingTerminal(
enabled: false,
2022-07-01 12:13:52 -04:00
level: VeilidConfigLogLevel.debug,
2022-06-15 21:51:38 -04:00
),
otlp: VeilidFFIConfigLoggingOtlp(
enabled: false,
2022-07-01 12:13:52 -04:00
level: VeilidConfigLogLevel.trace,
2022-06-15 21:51:38 -04:00
grpcEndpoint: "localhost:4317",
2022-07-01 12:13:52 -04:00
serviceName: "VeilidExample"),
api: VeilidFFIConfigLoggingApi(
enabled: true, level: VeilidConfigLogLevel.info)));
2022-07-01 16:20:43 -04:00
Veilid.instance.initializeVeilidCore(platformConfig.json);
2022-06-15 21:51:38 -04:00
}
2022-03-03 20:45:39 -05:00
runApp(MaterialApp(
title: 'Veilid Plugin Demo',
theme: ThemeData(
2022-12-09 18:59:31 -05:00
primarySwatch: '#6667AB',
2022-03-03 20:45:39 -05:00
visualDensity: VisualDensity.adaptivePlatformDensity,
),
home: const MyApp()));
2022-01-16 11:19:01 -05:00
}
2022-02-15 13:40:17 -05:00
// Main App
2022-01-16 11:19:01 -05:00
class MyApp extends StatefulWidget {
const MyApp({Key? key}) : super(key: key);
@override
State<MyApp> createState() => _MyAppState();
}
2022-03-03 20:45:39 -05:00
class _MyAppState extends State<MyApp> with UiLoggy {
String _veilidVersion = 'Unknown';
2022-03-03 20:45:39 -05:00
Stream<VeilidUpdate>? _updateStream;
2022-03-08 22:32:12 -05:00
Future<void>? _updateProcessor;
2022-01-16 11:19:01 -05:00
@override
void initState() {
super.initState();
2022-06-15 21:51:38 -04:00
setRootLogLevel(LogLevel.info);
2022-01-16 11:19:01 -05:00
initPlatformState();
}
// Platform messages are asynchronous, so we initialize in an async method.
Future<void> initPlatformState() async {
String veilidVersion;
2022-01-16 11:19:01 -05:00
// Platform messages may fail, so we use a try/catch PlatformException.
// We also handle the message potentially returning null.
try {
2022-03-17 10:31:10 -04:00
veilidVersion = Veilid.instance.veilidVersionString();
2022-03-11 07:35:41 -05:00
} on Exception {
veilidVersion = 'Failed to get veilid version.';
2022-01-16 11:19:01 -05:00
}
2022-03-11 07:35:41 -05:00
// In case of hot restart shut down first
try {
await Veilid.instance.shutdownVeilidCore();
} on Exception {
//
}
2022-01-16 11:19:01 -05:00
// If the widget was removed from the tree while the asynchronous platform
// message was in flight, we want to discard the reply rather than calling
// setState to update our non-existent appearance.
if (!mounted) return;
setState(() {
_veilidVersion = veilidVersion;
2022-01-16 11:19:01 -05:00
});
}
Future<void> processLog(VeilidLog log) async {
2022-09-07 10:33:14 -04:00
StackTrace? stackTrace;
Object? error;
final backtrace = log.backtrace;
2022-09-07 10:33:14 -04:00
if (backtrace != null) {
stackTrace =
StackTrace.fromString("$backtrace\n${StackTrace.current.toString()}");
error = 'embedded stack trace for ${log.logLevel} ${log.message}';
2022-09-07 10:33:14 -04:00
}
switch (log.logLevel) {
2022-03-08 22:32:12 -05:00
case VeilidLogLevel.error:
loggy.error(log.message, error, stackTrace);
2022-03-08 22:32:12 -05:00
break;
case VeilidLogLevel.warn:
loggy.warning(log.message, error, stackTrace);
2022-03-08 22:32:12 -05:00
break;
case VeilidLogLevel.info:
loggy.info(log.message, error, stackTrace);
2022-03-08 22:32:12 -05:00
break;
case VeilidLogLevel.debug:
loggy.debug(log.message, error, stackTrace);
2022-03-08 22:32:12 -05:00
break;
case VeilidLogLevel.trace:
loggy.trace(log.message, error, stackTrace);
2022-03-08 22:32:12 -05:00
break;
}
}
Future<void> processUpdates() async {
var stream = _updateStream;
if (stream != null) {
await for (final update in stream) {
if (update is VeilidLog) {
await processLog(update);
} else if (update is VeilidAppMessage) {
loggy.info("AppMessage: ${update.json}");
} else if (update is VeilidAppCall) {
loggy.info("AppCall: ${update.json}");
2022-03-08 22:32:12 -05:00
} else {
2022-09-06 16:49:43 -04:00
loggy.trace("Update: ${update.json}");
2022-03-08 22:32:12 -05:00
}
}
}
}
2022-01-16 11:19:01 -05:00
@override
Widget build(BuildContext context) {
2022-03-03 20:45:39 -05:00
final ButtonStyle buttonStyle =
ElevatedButton.styleFrom(textStyle: const TextStyle(fontSize: 20));
return Scaffold(
2022-01-16 11:19:01 -05:00
appBar: AppBar(
2022-02-15 13:40:17 -05:00
title: Text('Veilid Plugin Version $_veilidVersion'),
2022-01-16 11:19:01 -05:00
),
2022-03-03 20:45:39 -05:00
body: Column(children: [
Expanded(
child: Container(
color: ThemeData.dark().scaffoldBackgroundColor,
height: MediaQuery.of(context).size.height * 0.4,
2022-12-09 18:59:31 -05:00
child: SafeArea(
child: TerminalView(
terminal,
controller: terminalController,
autofocus: true,
backgroundOpacity: 0.7,
onSecondaryTapDown: (details, offset) async {
final selection = terminalController.selection;
if (selection != null) {
final text = terminal.buffer.getText(selection);
terminalController.clearSelection();
await Clipboard.setData(ClipboardData(text: text));
} else {
final data = await Clipboard.getData('text/plain');
final text = data?.text;
if (text != null) {
terminal.paste(text);
}
}
},
),
),
2022-03-03 20:45:39 -05:00
)),
Container(
padding: const EdgeInsets.fromLTRB(8, 8, 8, 12),
child: Row(children: [
ElevatedButton(
style: buttonStyle,
2022-03-08 22:32:12 -05:00
onPressed: _updateStream != null
? null
: () async {
2022-09-09 16:27:13 -04:00
var updateStream = await Veilid.instance
.startupVeilidCore(
await getDefaultVeilidConfig());
2022-03-08 22:32:12 -05:00
setState(() {
_updateStream = updateStream;
_updateProcessor = processUpdates();
});
2022-09-06 18:59:41 -04:00
await Veilid.instance.attach();
2022-03-08 22:32:12 -05:00
},
2022-03-03 20:45:39 -05:00
child: const Text('Startup'),
),
ElevatedButton(
style: buttonStyle,
2022-03-08 22:32:12 -05:00
onPressed: _updateStream == null
? null
: () async {
await Veilid.instance.shutdownVeilidCore();
if (_updateProcessor != null) {
await _updateProcessor;
}
setState(() {
_updateProcessor = null;
_updateStream = null;
});
},
2022-03-03 20:45:39 -05:00
child: const Text('Shutdown'),
),
])),
Row(children: [
Expanded(
child: TextField(
decoration: const InputDecoration(
border: OutlineInputBorder(),
labelText: 'Debug Command'),
textInputAction: TextInputAction.send,
onSubmitted: (String v) async {
2022-08-24 20:59:09 -04:00
loggy.info(await Veilid.instance.debug(v));
2022-03-03 20:45:39 -05:00
})),
DropdownButton<LogLevel>(
value: loggy.level.logLevel,
onChanged: (LogLevel? newLevel) {
setState(() {
setRootLogLevel(newLevel);
});
},
items: const [
DropdownMenuItem<LogLevel>(
value: LogLevel.error, child: Text("Error")),
DropdownMenuItem<LogLevel>(
value: LogLevel.warning, child: Text("Warning")),
DropdownMenuItem<LogLevel>(
value: LogLevel.info, child: Text("Info")),
DropdownMenuItem<LogLevel>(
value: LogLevel.debug, child: Text("Debug")),
DropdownMenuItem<LogLevel>(
value: traceLevel, child: Text("Trace")),
DropdownMenuItem<LogLevel>(
value: LogLevel.all, child: Text("All")),
])
]),
]));
2022-01-16 11:19:01 -05:00
}
}