This commit is contained in:
ekibun
2020-09-13 22:50:14 +08:00
parent 89ea0222be
commit 60466a5c7d
11 changed files with 146 additions and 10 deletions

View File

@@ -3,7 +3,7 @@
* @Author: ekibun
* @Date: 2020-08-08 08:16:51
* @LastEditors: ekibun
* @LastEditTime: 2020-08-27 20:39:32
* @LastEditTime: 2020-09-06 19:44:32
*/
import 'package:flutter/material.dart';
import 'dart:typed_data';
@@ -46,13 +46,13 @@ class TestPage extends StatefulWidget {
class _TestPageState extends State<TestPage> {
String resp;
FlutterJs engine;
FlutterQjs engine;
CodeInputController _controller = CodeInputController();
_createEngine() async {
if (engine != null) return;
engine = FlutterJs();
engine = FlutterQjs();
await engine.setMethodHandler((String method, List arg) async {
switch (method) {
case "http":

View File

@@ -57,6 +57,13 @@ packages:
url: "https://pub.flutter-io.cn"
source: hosted
version: "1.1.0-nullsafety"
ffi:
dependency: transitive
description:
name: ffi
url: "https://pub.flutter-io.cn"
source: hosted
version: "0.1.3"
flutter:
dependency: "direct main"
description: flutter

View File

@@ -0,0 +1,27 @@
// This is a basic Flutter widget test.
//
// To perform an interaction with a widget in your test, use the WidgetTester
// utility that Flutter provides. For example, you can send tap and scroll
// gestures. You can also use WidgetTester to find child widgets in the widget
// tree, read text, and verify that the values of widget properties are correct.
import 'package:flutter/material.dart';
import 'package:flutter_test/flutter_test.dart';
import 'package:flutter_qjs_example/main.dart';
void main() {
testWidgets('Verify Platform version', (WidgetTester tester) async {
// Build our app and trigger a frame.
await tester.pumpWidget(MyApp());
// Verify that platform version is retrieved.
expect(
find.byWidgetPredicate(
(Widget widget) => widget is Text &&
widget.data.startsWith('Running on:'),
),
findsOneWidget,
);
});
}

View File

@@ -3,7 +3,7 @@
* @Author: ekibun
* @Date: 2020-08-08 08:29:09
* @LastEditors: ekibun
* @LastEditTime: 2020-08-28 10:45:14
* @LastEditTime: 2020-09-06 13:03:56
*/
import 'dart:async';
import 'dart:io';
@@ -19,9 +19,9 @@ typedef JsModuleHandler = Future<String> Function(String name);
class JsMethodHandlerNotImplement {}
/// FlutterJs instance.
/// Each [FlutterJs] object creates a new thread that runs a simple js loop.
/// Each [FlutterQjs] 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 {
class FlutterQjs {
dynamic _engine;
dynamic get pointer => _engine;

View File

@@ -50,6 +50,13 @@ packages:
url: "https://pub.flutter-io.cn"
source: hosted
version: "1.1.0-nullsafety"
ffi:
dependency: "direct main"
description:
name: ffi
url: "https://pub.flutter-io.cn"
source: hosted
version: "0.1.3"
flutter:
dependency: "direct main"
description: flutter

View File

@@ -10,6 +10,7 @@ environment:
dependencies:
flutter:
sdk: flutter
ffi: 0.1.3
dev_dependencies:
flutter_test:

View File

@@ -0,0 +1,33 @@
/*
* @Description:
* @Author: ekibun
* @Date: 2020-09-06 13:02:46
* @LastEditors: ekibun
* @LastEditTime: 2020-09-13 17:29:26
*/
import 'dart:ffi';
import 'package:ffi/ffi.dart';
void main() {
final DynamicLibrary qjsLib = DynamicLibrary.open("test/lib/build/Debug/ffi_library.dll");
print(qjsLib);
// JSRuntime *js_NewRuntime(void);
final Pointer Function() jsNewRuntime =
qjsLib.lookup<NativeFunction<Pointer Function()>>("jsNewRuntime").asFunction();
final rt = jsNewRuntime();
print(rt);
// JSContext *js_NewContext(JSRuntime *rt);
final Pointer Function(Pointer rt) jsNewContext =
qjsLib.lookup<NativeFunction<Pointer Function(Pointer)>>("jsNewContext").asFunction();
final ctx = jsNewContext(rt);
print(ctx);
// JSValue *js_Eval(JSContext *ctx, const char *input, const char *filename, int eval_flags)
final Pointer Function(Pointer rt, Pointer<Utf8> input, Pointer<Utf8> filename, int evalFlags) jsEval =
qjsLib.lookup<NativeFunction<Pointer Function(Pointer,Pointer<Utf8>,Pointer<Utf8>, Int32)>>("jsEval").asFunction();
final jsval = jsEval(ctx, Utf8.toUtf8("1+1"), Utf8.toUtf8("<eval>"), 0);
// const char *js_ToCString(JSContext *ctx, JSValue *val)
final Pointer<Utf8> Function(Pointer rt, Pointer val) jsToCString =
qjsLib.lookup<NativeFunction<Pointer<Utf8> Function(Pointer,Pointer)>>("jsToCString").asFunction();
final str = Utf8.fromUtf8(jsToCString(ctx, jsval));
print(str);
}

19
test/lib/CMakeLists.txt Normal file
View File

@@ -0,0 +1,19 @@
cmake_minimum_required(VERSION 3.7 FATAL_ERROR)
project(ffi_library LANGUAGES CXX)
add_library(ffi_library SHARED ffi.cpp)
# quickjs
set(QUICK_JS_LIB_DIR ${CMAKE_CURRENT_SOURCE_DIR}/../../cxx/quickjs)
file (STRINGS "${QUICK_JS_LIB_DIR}/VERSION" QUICKJS_VERSION)
add_library(libquickjs STATIC
${QUICK_JS_LIB_DIR}/cutils.c
${QUICK_JS_LIB_DIR}/libregexp.c
${QUICK_JS_LIB_DIR}/libunicode.c
${QUICK_JS_LIB_DIR}/quickjs.c
)
project(libquickjs LANGUAGES C)
target_compile_options(libquickjs PRIVATE "-DCONFIG_VERSION=\"${QUICKJS_VERSION}\"")
target_compile_options(libquickjs PRIVATE "-DDUMP_LEAKS")
target_link_libraries(ffi_library PRIVATE libquickjs)

39
test/lib/ffi.cpp Normal file
View File

@@ -0,0 +1,39 @@
/*
* @Description:
* @Author: ekibun
* @Date: 2020-09-06 18:32:45
* @LastEditors: ekibun
* @LastEditTime: 2020-09-13 17:26:29
*/
#include "../../cxx/quickjs/quickjs.h"
#include <cstring>
#ifdef _MSC_VER
#define DLLEXPORT __declspec(dllexport)
#else
#define DLLEXPORT __attribute__((visibility("default")))
#endif
extern "C"
{
DLLEXPORT JSRuntime *jsNewRuntime()
{
return JS_NewRuntime();
}
DLLEXPORT JSContext *jsNewContext(JSRuntime *rt)
{
return JS_NewContext(rt);
}
DLLEXPORT JSValue *jsEval(JSContext *ctx, const char *input, const char *filename, int eval_flags)
{
return new JSValue{JS_Eval(ctx, input, strlen(input), filename, eval_flags)};
}
DLLEXPORT const char *jsToCString(JSContext *ctx, JSValue *val)
{
return JS_ToCString(ctx, *val);
}
}

5
test/lib/make.cmd Normal file
View File

@@ -0,0 +1,5 @@
cd %~dp0
set CMAKE="C:\Program Files (x86)\Microsoft Visual Studio\2019\BuildTools\Common7\IDE\CommonExtensions\Microsoft\CMake\CMake\bin\cmake.exe"
set BUILD_DIR="./build"
%CMAKE% -S ./ -B %BUILD_DIR%
%CMAKE% --build %BUILD_DIR% --verbose

View File

@@ -121,7 +121,7 @@ namespace
qjs::Engine *engine = (qjs::Engine *)std::get<int64_t>(ValueOrNull(args, "engine"));
std::string script = std::get<std::string>(ValueOrNull(args, "script"));
std::string name = std::get<std::string>(ValueOrNull(args, "name"));
auto presult = result.release();
std::shared_ptr<flutter::MethodResult<flutter::EncodableValue>> presult = std::move(result);
engine->commit(qjs::EngineTask{
[script, name](qjs::Context &ctx) {
return ctx.eval(script, name.c_str(), JS_EVAL_TYPE_GLOBAL);
@@ -129,11 +129,9 @@ namespace
[presult](qjs::Value resolve) {
flutter::EncodableValue response = qjs::jsToDart(resolve);
presult->Success(&response);
delete presult;
},
[presult](qjs::Value reject) {
presult->Error("FlutterJSException", qjs::getStackTrack(reject));
delete presult;
}});
}
else if (method_call.method_name().compare("call") == 0)
@@ -142,7 +140,7 @@ namespace
qjs::Engine *engine = (qjs::Engine *)std::get<int64_t>(ValueOrNull(args, "engine"));
qjs::JSValue *function = (qjs::JSValue *)std::get<int64_t>(ValueOrNull(args, "function"));
flutter::EncodableList arguments = std::get<flutter::EncodableList>(ValueOrNull(args, "arguments"));
auto presult = result.release();
std::shared_ptr<flutter::MethodResult<flutter::EncodableValue>> presult = std::move(result);
engine->commit(qjs::EngineTask{
[function, arguments](qjs::Context &ctx) {
size_t argscount = arguments.size();