update doc

This commit is contained in:
ekibun
2020-08-20 14:57:07 +08:00
parent 805f261880
commit 204f2411e5
7 changed files with 182 additions and 167 deletions

View File

@@ -3,8 +3,12 @@
* @Author: ekibun * @Author: ekibun
* @Date: 2020-08-08 08:16:50 * @Date: 2020-08-08 08:16:50
* @LastEditors: ekibun * @LastEditors: ekibun
* @LastEditTime: 2020-08-20 14:05:32 * @LastEditTime: 2020-08-20 14:55:13
--> -->
## 0.0.2
* update example.
## 0.0.1 ## 0.0.1
* initial publish. * initial publish.

View File

@@ -72,7 +72,8 @@ class CodeInputController extends TextEditingController {
InlineSpan leftSpan; InlineSpan leftSpan;
oldSpan.children?.indexWhere((element) { oldSpan.children?.indexWhere((element) {
String elementText = element.toPlainText(); String elementText = element.toPlainText();
if (start + elementText.length > splitAt || !newText.startsWith(elementText, start)) { if (start + elementText.length > splitAt ||
!newText.startsWith(elementText, start)) {
leftSpan = element; leftSpan = element;
return true; return true;
} }
@@ -86,7 +87,9 @@ class CodeInputController extends TextEditingController {
oldSpan.children?.sublist(beforeSpans.length)?.lastIndexWhere((element) { oldSpan.children?.sublist(beforeSpans.length)?.lastIndexWhere((element) {
String elementText = element.toPlainText(); String elementText = element.toPlainText();
if (splitAt + end + elementText.length >= newText.length || if (splitAt + end + elementText.length >= newText.length ||
!newText.substring(start, newText.length - end).endsWith(elementText)) { !newText
.substring(start, newText.length - end)
.endsWith(elementText)) {
rightSpan = element; rightSpan = element;
return true; return true;
} }
@@ -98,7 +101,9 @@ class CodeInputController extends TextEditingController {
return TextSpan(style: style, children: [ return TextSpan(style: style, children: [
...beforeSpans, ...beforeSpans,
TextSpan( TextSpan(
style: leftSpan != null && leftSpan == rightSpan ? leftSpan.style : style, style: leftSpan != null && leftSpan == rightSpan
? leftSpan.style
: style,
text: newText.substring(start, max(start, newText.length - end))), text: newText.substring(start, max(start, newText.length - end))),
...endSpans.reversed ...endSpans.reversed
]); ]);

View File

@@ -1,21 +1,24 @@
/* /*
* @Description: * @Description: example
* @Author: ekibun * @Author: ekibun
* @Date: 2020-08-08 08:16:51 * @Date: 2020-08-08 08:16:51
* @LastEditors: ekibun * @LastEditors: ekibun
* @LastEditTime: 2020-08-17 21:46:10 * @LastEditTime: 2020-08-20 14:42:10
*/ */
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'dart:typed_data';
import 'package:flutter_qjs_example/test.dart'; import 'package:dio/dio.dart';
import 'package:flutter_qjs/flutter_qjs.dart';
import 'code/editor.dart';
void main() { void main() {
runApp(MyApp()); runApp(MyApp());
} }
class MyApp extends StatelessWidget { class MyApp extends StatelessWidget {
const MyApp({Key key}) : super(key: key);
const MyApp({ Key key }) : super(key: key);
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@@ -24,8 +27,6 @@ class MyApp extends StatelessWidget {
debugShowCheckedModeBanner: false, debugShowCheckedModeBanner: false,
theme: ThemeData( theme: ThemeData(
appBarTheme: AppBarTheme(brightness: Brightness.dark, elevation: 0), appBarTheme: AppBarTheme(brightness: Brightness.dark, elevation: 0),
primaryColor: Color(0xfff09199),
accentColor: Color(0xffec818a),
backgroundColor: Colors.grey[300], backgroundColor: Colors.grey[300],
primaryColorBrightness: Brightness.dark, primaryColorBrightness: Brightness.dark,
), ),
@@ -35,5 +36,109 @@ class MyApp extends StatelessWidget {
initialRoute: 'home', initialRoute: 'home',
); );
} }
}
class TestPage extends StatefulWidget {
@override
State<StatefulWidget> createState() => _TestPageState();
}
class _TestPageState extends State<TestPage> {
String code, resp;
FlutterJs engine;
_createEngine() async {
if (engine != null) return;
engine = FlutterJs();
await engine.setMethodHandler((String method, List arg) async {
switch (method) {
case "http":
Response response = await Dio().get(arg[0]);
return response.data;
case "test":
return await arg[0]([
true,
1,
0.5,
"str",
{"key": "val", 0: 1},
Uint8List(2),
Int32List(2),
Int64List(2),
Float64List(2),
Float32List(2)
]);
default:
return JsMethodHandlerNotImplement();
}
});
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text("JS engine test"),
),
body: SingleChildScrollView(
padding: const EdgeInsets.all(16),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
SingleChildScrollView(
scrollDirection: Axis.horizontal,
child: Row(
children: [
FlatButton(
child: Text("create engine"), onPressed: _createEngine),
FlatButton(
child: Text("evaluate"),
onPressed: () async {
if (engine == null) {
print("please create engine first");
return;
}
try {
resp = (await engine.evaluate(code ?? '', "<eval>"))
.toString();
} catch (e) {
resp = e.toString();
}
setState(() {});
}),
FlatButton(
child: Text("close engine"),
onPressed: () async {
if (engine != null) return;
await engine.destroy();
engine = null;
}),
],
),
),
Container(
padding: const EdgeInsets.all(12),
color: Colors.grey.withOpacity(0.1),
constraints: BoxConstraints(minHeight: 200),
child: CodeEditor(
onChanged: (v) {
code = v;
},
),
),
SizedBox(height: 16),
Text("result:"),
SizedBox(height: 16),
Container(
width: double.infinity,
padding: const EdgeInsets.all(12),
color: Colors.green.withOpacity(0.05),
constraints: BoxConstraints(minHeight: 100),
child: Text(resp ?? ''),
),
],
),
),
);
}
} }

View File

@@ -1,116 +0,0 @@
/*
* @Description:
* @Author: ekibun
* @Date: 2020-07-18 23:28:55
* @LastEditors: ekibun
* @LastEditTime: 2020-08-20 13:11:33
*/
import 'dart:typed_data';
import 'package:dio/dio.dart';
import 'package:flutter/material.dart';
import 'package:flutter_qjs/flutter_qjs.dart';
import 'code/editor.dart';
class TestPage extends StatefulWidget {
@override
State<StatefulWidget> createState() => _TestPageState();
}
class _TestPageState extends State<TestPage> {
String code, resp;
FlutterJs engine;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
title: Text("JS engine test"),
),
body: SingleChildScrollView(
padding: const EdgeInsets.all(16),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
SingleChildScrollView(
scrollDirection: Axis.horizontal,
child: Row(
children: [
FlatButton(
child: Text("create engine"),
onPressed: () async {
if (engine != null) return;
engine = FlutterJs();
engine.setMethodHandler((String method, List arg) async {
switch (method) {
case "http":
Response response = await Dio().get(arg[0]);
return response.data;
case "test":
return await arg[0]([
true,
1,
0.5,
"str",
{ "key": "val", 0: 1 },
Uint8List(2),
Int32List(2),
Int64List(2),
Float64List(2),
Float32List(2)]);
default:
return JsMethodHandlerNotImplement();
}
});
}),
FlatButton(
child: Text("evaluate"),
onPressed: () async {
if (engine == null) {
print("please create engine first");
return;
}
try {
resp = "${await engine.evaluate(code ?? '', "<eval>")}";
} catch (e) {
resp = e.toString();
}
setState(() {});
}),
FlatButton(
child: Text("close engine"),
onPressed: () async {
if (engine != null) return;
await engine.destroy();
engine = null;
}),
],
),
),
Container(
padding: const EdgeInsets.all(12),
color: Colors.grey.withOpacity(0.1),
constraints: BoxConstraints(minHeight: 200),
child: CodeEditor(
onChanged: (v) {
code = v;
},
),
),
SizedBox(height: 16),
Text("result:"),
SizedBox(height: 16),
Container(
width: double.infinity,
padding: const EdgeInsets.all(12),
color: Colors.green.withOpacity(0.05),
constraints: BoxConstraints(minHeight: 100),
child: Text(resp ?? ''),
),
],
),
),
);
}
}

View File

@@ -75,7 +75,7 @@ packages:
path: ".." path: ".."
relative: true relative: true
source: path source: path
version: "0.0.1" version: "0.0.2"
flutter_test: flutter_test:
dependency: "direct dev" dependency: "direct dev"
description: flutter description: flutter

View File

@@ -1,31 +1,73 @@
/* /*
* @Description: * @Description: quickjs engine
* @Author: ekibun * @Author: ekibun
* @Date: 2020-08-08 08:29:09 * @Date: 2020-08-08 08:29:09
* @LastEditors: ekibun * @LastEditors: ekibun
* @LastEditTime: 2020-08-20 12:49:46 * @LastEditTime: 2020-08-20 14:43:40
*/ */
import 'dart:async'; import 'dart:async';
import 'dart:io'; import 'dart:io';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
/// Handle function to manage js call with `dart(method, ...args)` function.
typedef JsMethodHandler = Future<dynamic> Function(String method, List args); typedef JsMethodHandler = Future<dynamic> Function(String method, List args);
/// return this in [JsMethodHandler] to mark method not implemented.
class JsMethodHandlerNotImplement {} class JsMethodHandlerNotImplement {}
/// FlutterJs instance.
/// Each [FlutterJs] object creates a new thread that runs a simple js loop.
/// Make sure call `destroy` to terminate thread and release memory when you don't need it.
class FlutterJs {
dynamic _engine;
_ensureEngine() async {
if (_engine == null) {
_engine = await _FlutterJs.instance._channel.invokeMethod("createEngine");
print(_engine);
}
}
/// Set a handler to manage js call with `dart(method, ...args)` function.
setMethodHandler(JsMethodHandler handler) async {
await _ensureEngine();
_FlutterJs.instance.methodHandlers[_engine] = handler;
}
/// Terminate thread and release memory.
destroy() async {
if (_engine != null) {
await _FlutterJs.instance._channel
.invokeMethod("close", {"engine": _engine});
_engine = null;
}
}
/// Evaluate js script.
Future<dynamic> evaluate(String command, String name) async {
_ensureEngine();
var arguments = {"engine": _engine, "script": command, "name": name};
return _FlutterJs.instance._wrapFunctionArguments(
await _FlutterJs.instance._channel.invokeMethod("evaluate", arguments),
_engine);
}
}
class _FlutterJs { class _FlutterJs {
factory _FlutterJs() => _getInstance(); factory _FlutterJs() => _getInstance();
static _FlutterJs get instance => _getInstance(); static _FlutterJs get instance => _getInstance();
static _FlutterJs _instance; static _FlutterJs _instance;
MethodChannel _channel = const MethodChannel('soko.ekibun.flutter_qjs'); MethodChannel _channel = const MethodChannel('soko.ekibun.flutter_qjs');
Map<dynamic, JsMethodHandler> methodHandlers = Map<dynamic, JsMethodHandler>(); Map<dynamic, JsMethodHandler> methodHandlers =
Map<dynamic, JsMethodHandler>();
_FlutterJs._internal() { _FlutterJs._internal() {
_channel.setMethodCallHandler((call) async { _channel.setMethodCallHandler((call) async {
print(call.arguments); print(call.arguments);
var engine = call.arguments["engine"]; var engine = call.arguments["engine"];
var args = call.arguments["args"]; var args = call.arguments["args"];
if (methodHandlers[engine] == null) return call.noSuchMethod(null); if (methodHandlers[engine] == null) return call.noSuchMethod(null);
var ret = await methodHandlers[engine](call.method, _wrapFunctionArguments(args, engine)); var ret = await methodHandlers[engine](
call.method, _wrapFunctionArguments(args, engine));
if (ret is JsMethodHandlerNotImplement) return call.noSuchMethod(null); if (ret is JsMethodHandlerNotImplement) return call.noSuchMethod(null);
return ret; return ret;
}); });
@@ -43,8 +85,13 @@ class _FlutterJs {
if (val["__js_function__"] != null) { if (val["__js_function__"] != null) {
var functionId = val["__js_function__"]; var functionId = val["__js_function__"];
return (List<dynamic> args) async { return (List<dynamic> args) async {
var arguments = {"engine": engine, "function": functionId, "arguments": args}; var arguments = {
return _wrapFunctionArguments(await _channel.invokeMethod("call", arguments), engine); "engine": engine,
"function": functionId,
"arguments": args
};
return _wrapFunctionArguments(
await _channel.invokeMethod("call", arguments), engine);
}; };
} else } else
for (var key in val.keys) { for (var key in val.keys) {
@@ -61,33 +108,3 @@ class _FlutterJs {
return _instance; return _instance;
} }
} }
class FlutterJs {
dynamic _engine;
_ensureEngine() async {
if (_engine == null) {
_engine = await _FlutterJs.instance._channel.invokeMethod("createEngine");
print(_engine);
}
}
setMethodHandler(JsMethodHandler handler) async {
await _ensureEngine();
_FlutterJs.instance.methodHandlers[_engine] = handler;
}
destroy() async {
if (_engine != null) {
await _FlutterJs.instance._channel.invokeMethod("close", {"engine": _engine});
_engine = null;
}
}
Future<dynamic> evaluate(String command, String name) async {
_ensureEngine();
var arguments = {"engine": _engine, "script": command, "name": "<eval>"};
return _FlutterJs.instance._wrapFunctionArguments(
await _FlutterJs.instance._channel.invokeMethod("evaluate", arguments), _engine);
}
}

View File

@@ -1,6 +1,6 @@
name: flutter_qjs name: flutter_qjs
description: A new flutter plugin project. description: This plugin is a simple js engine for flutter using the `quickjs` project. Plugin currently supports Windows, Linux, and Android.
version: 0.0.1 version: 0.0.2
homepage: https://github.com/ekibun/flutter_qjs homepage: https://github.com/ekibun/flutter_qjs
environment: environment: