Files
cup_edit/lib/filament_controller_ffi.dart
2023-11-07 13:15:12 +08:00

1052 lines
33 KiB
Dart

import 'dart:async';
import 'dart:ffi';
import 'dart:io';
import 'dart:ui' as ui;
import 'dart:developer' as dev;
import 'package:flutter/services.dart';
import 'package:ffi/ffi.dart';
import 'package:flutter/widgets.dart';
import 'package:flutter_filament/filament_controller.dart';
import 'package:flutter_filament/animations/animation_data.dart';
import 'package:flutter_filament/generated_bindings.dart';
import 'package:flutter_filament/rendering_surface.dart';
import 'package:vector_math/vector_math_64.dart';
// ignore: constant_identifier_names
const FilamentEntity _FILAMENT_ASSET_ERROR = 0;
class FilamentControllerFFI extends FilamentController {
final _channel = const MethodChannel("app.polyvox.filament/event");
bool _usesBackingWindow = false;
@override
bool get requiresTextureWidget => !_usesBackingWindow;
double _pixelRatio = 1.0;
late Pointer<Void>? _assetManager;
Pointer<Void>? _viewer;
final String? uberArchivePath;
Pointer<Void> _driver = nullptr.cast<Void>();
@override
final rect = ValueNotifier<Rect?>(null);
@override
final hasViewer = ValueNotifier<bool>(false);
@override
Stream<FilamentEntity> get pickResult => _pickResultController.stream;
final _pickResultController = StreamController<FilamentEntity>.broadcast();
int? _resizingWidth;
int? _resizingHeight;
Timer? _resizeTimer;
///
/// This controller uses platform channels to bridge Dart with the C/C++ code for the Filament API.
/// Setting up the context/texture (since this is platform-specific) and the render ticker are platform-specific; all other methods are passed through by the platform channel to the methods specified in FlutterFilamentApi.h.
///
FilamentControllerFFI({this.uberArchivePath}) {
// on some platforms, we ignore the resize event raised by the Flutter RenderObserver
// in favour of a window-level event passed via the method channel.
// (this is because there is no apparent way to exactly synchronize resizing a Flutter widget and resizing a pixel buffer, so we need
// to handle the latter first and rebuild the swapchain appropriately).
_channel.setMethodCallHandler((call) async {
if (call.arguments[0] == _resizingWidth &&
call.arguments[1] == _resizingHeight) {
return;
}
_resizeTimer?.cancel();
_resizingWidth = call.arguments[0];
_resizingHeight = call.arguments[1];
_resizeTimer = Timer(const Duration(milliseconds: 500), () async {
rect.value = Offset.zero &
ui.Size(_resizingWidth!.toDouble(), _resizingHeight!.toDouble());
await resize();
});
});
late DynamicLibrary dl;
if (Platform.isIOS || Platform.isMacOS || Platform.isWindows) {
dl = DynamicLibrary.process();
} else {
dl = DynamicLibrary.open("libflutter_filament_android.so");
}
if (Platform.isWindows) {
_channel.invokeMethod("usesBackingWindow").then((result) {
_usesBackingWindow = result;
});
}
}
bool _rendering = false;
@override
bool get rendering => _rendering;
@override
Future setRendering(bool render) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
_rendering = render;
set_rendering_ffi(_viewer!, render);
}
@override
Future render() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
render_ffi(_viewer!);
}
@override
Future setFrameRate(int framerate) async {
set_frame_interval_ffi(1.0 / framerate);
}
@override
Future setDimensions(Rect rect, double pixelRatio) async {
this.rect.value = Rect.fromLTWH(rect.left, rect.top,
rect.width * _pixelRatio, rect.height * _pixelRatio);
_pixelRatio = pixelRatio;
}
@override
Future destroy() async {
await destroyViewer();
await destroyTexture();
}
@override
Future destroyViewer() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
var viewer = _viewer;
_viewer = null;
_assetManager = null;
destroy_filament_viewer_ffi(viewer!);
hasViewer.value = false;
}
@override
Future destroyTexture() async {
if (textureDetails.value != null) {
await _channel.invokeMethod(
"destroyTexture", textureDetails.value!.textureId);
}
dev.log("Texture destroyed");
}
///
/// Called by `FilamentWidget`. You do not need to call this yourself.
///
@override
Future createViewer() async {
if (rect.value == null) {
throw Exception(
"Dimensions have not yet been set by FilamentWidget. You need to wait for at least one frame after FilamentWidget has been inserted into the hierarchy");
}
if (_viewer != null) {
throw Exception(
"Viewer already exists, make sure you call destroyViewer first");
}
if (textureDetails.value != null) {
throw Exception(
"Texture already exists, make sure you call destroyTexture first");
}
var loader = Pointer<ResourceLoaderWrapper>.fromAddress(
await _channel.invokeMethod("getResourceLoaderWrapper"));
if (loader == nullptr) {
throw Exception("Failed to get resource loader");
}
if (Platform.isWindows && requiresTextureWidget) {
_driver = Pointer<Void>.fromAddress(
await _channel.invokeMethod("getDriverPlatform"));
}
var renderCallbackResult = await _channel.invokeMethod("getRenderCallback");
var renderCallback =
Pointer<NativeFunction<Void Function(Pointer<Void>)>>.fromAddress(
renderCallbackResult[0]);
var renderCallbackOwner =
Pointer<Void>.fromAddress(renderCallbackResult[1]);
var renderingSurface = await _createRenderingSurface();
dev.log("Got rendering surface");
_viewer = create_filament_viewer_ffi(
Pointer<Void>.fromAddress(renderingSurface.sharedContext),
_driver,
uberArchivePath?.toNativeUtf8().cast<Char>() ?? nullptr,
loader,
renderCallback,
renderCallbackOwner);
dev.log("Created viewer");
if (_viewer!.address == 0) {
throw Exception("Failed to create viewer. Check logs for details");
}
_assetManager = get_asset_manager(_viewer!);
create_swap_chain_ffi(_viewer!, renderingSurface.surface,
rect.value!.width.toInt(), rect.value!.height.toInt());
dev.log("Created swap chain");
if (renderingSurface.textureHandle != 0) {
dev.log(
"Creating render target from native texture ${renderingSurface.textureHandle}");
create_render_target_ffi(_viewer!, renderingSurface.textureHandle,
rect.value!.width.toInt(), rect.value!.height.toInt());
}
textureDetails.value = TextureDetails(
textureId: renderingSurface.flutterTextureId,
width: rect.value!.width.toInt(),
height: rect.value!.height.toInt());
dev.log("texture details ${textureDetails.value}");
update_viewport_and_camera_projection_ffi(
_viewer!, rect.value!.width.toInt(), rect.value!.height.toInt(), 1.0);
hasViewer.value = true;
}
Future<RenderingSurface> _createRenderingSurface() async {
return RenderingSurface.from(await _channel.invokeMethod("createTexture", [
rect.value!.width,
rect.value!.height,
rect.value!.left,
rect.value!.top
]));
}
///
/// When a FilamentWidget is resized, it will call the [resize] method below, which will tear down/recreate the swapchain.
/// For "once-off" resizes, this is fine; however, this can be problematic for consecutive resizes
/// (e.g. dragging to expand/contract the parent window on desktop, or animating the size of the FilamentWidget itself).
/// It is too expensive to recreate the swapchain multiple times per second.
/// We therefore add a timer to FilamentWidget so that the call to [resize] is delayed (e.g. 500ms).
/// Any subsequent resizes before the delay window elapses will cancel the earlier call.
///
/// The overall process looks like this:
/// 1) the window is resized
/// 2) (Windows only) the Flutter engine requests PixelBufferTexture to provide a new pixel buffer with a new size (we return an empty texture, blanking the Texture widget)
/// 3) After Xms, [resize] is invoked
/// 4) the viewer is instructed to stop rendering (synchronous)
/// 5) the existing Filament swapchain is destroyed (synchronous)
/// 6) (where a Texture widget is used), the Flutter texture is unregistered
/// a) this is asynchronous, but
/// b) *** SEE NOTE BELOW ON WINDOWS *** by passing the method channel result through to the callback, we make this synchronous from the Flutter side,
/// c) in this async callback, the glTexture is destroyed
/// 7) (where a backing window is used), the window is resized
/// 7) (where a Texture widget is used), a new Flutter/OpenGL texture is created (synchronous)
/// 8) a new swapchain is created (synchronous)
/// 9) if the viewer was rendering prior to the resize, the viewer is instructed to recommence rendering
/// 10) (where a Texture widget is used) the new texture ID is pushed to the FilamentWidget
/// 11) the FilamentWidget updates the Texture widget with the new texture.
///
/// #### (Windows-only) ############################################################
/// # As soon as the widget/window is resized, the PixelBufferTexture will be
/// # requested to provide a new pixel buffer for the new size.
/// # Even with zero delay to the call to [resize], this will be triggered *before*
/// # we have had a chance to anything else (like tear down the swapchain).
/// # On the backend, we deal with this by simply returning an empty texture as soon
/// # as the size changes, and will rely on the followup call to [resize] to actually
/// # destroy/recreate the pixel buffer and Flutter texture.
///
/// NOTE RE ASYNC CALLBACK
/// # The bigger problem is a race condition when resize is called multiple times in quick succession (e.g dragging to resize on Windows).
/// # It looks like occasionally, the backend OpenGL texture is being destroyed while its corresponding swapchain is still active, causing a crash.
/// # I'm not exactly sure how/where this is occurring, but something clearly isn't synchronized between destroy_swap_chain_ffi and
/// # the asynchronous callback passed to FlutterTextureRegistrar::UnregisterTexture.
/// # Theoretically this could occur if resize_2 starts before resize_1 completes, i.e.
/// # 1) resize_1 destroys swapchain/texture and creates new texture
/// # 2) resize_2 destroys swapchain/texture
/// # 3) resize_1 creates new swapchain but texture isn't available, ergo crash
/// #
/// # I don't think this should happen if:
/// # 1) we add a flag on the Flutter side to ensure only one call to destroy/recreate the swapchain/texture is active at any given time, and
/// # 2) on the Flutter side, we are sure that calling destroyTexture only returns once the async callback on the native side has completed.
/// # For (1), checking if textureId is null at the entrypoint should be sufficient.
/// # For (2), we invoke flutter::MethodResult<flutter::EncodableValue>->Success in the UnregisterTexture callback.
/// #
/// # Maybe (2) doesn't actually make Flutter wait?
/// #
/// # The other possibility is that both (1) and (2) are fine and the issue is elsewhere.
/// #
/// # Either way, the current solution is to basically setup a double-buffer on resize.
/// # When destroyTexture is called, the active texture isn't destroyed yet, it's only marked as inactive.
/// # On subsequent calls to destroyTexture, the inactive texture is destroyed.
/// # This seems to work fine.
///
/// # Another option is to only use a single large (e.g. 4k) texture and simply crop whenever a resize is requested.
/// # This might be preferable for other reasons (e.g. don't need to destroy/recreate the pixel buffer or swapchain).
/// # Given we don't do this on other platforms, I'm OK to stick with the existing solution for the time being.
/// ############################################################################
///
bool _resizing = false;
@override
Future resize() async {
if (_viewer == null) {
throw Exception("Cannot resize without active viewer");
}
if (_resizing) {
throw Exception("Resize currently underway, ignoring");
}
_resizing = true;
set_rendering_ffi(_viewer!, false);
if (!_usesBackingWindow) {
destroy_swap_chain_ffi(_viewer!);
}
if (requiresTextureWidget) {
if (textureDetails.value != null) {
await _channel.invokeMethod(
"destroyTexture", textureDetails.value!.textureId);
}
} else if (Platform.isWindows) {
dev.log("Resizing window with rect $rect");
await _channel.invokeMethod("resizeWindow", [
rect.value!.width,
rect.value!.height,
rect.value!.left,
rect.value!.top
]);
}
var renderingSurface = await _createRenderingSurface();
if (_viewer!.address == 0) {
throw Exception("Failed to create viewer. Check logs for details");
}
_assetManager = get_asset_manager(_viewer!);
if (!_usesBackingWindow) {
create_swap_chain_ffi(_viewer!, renderingSurface.surface,
rect.value!.width.toInt(), rect.value!.height.toInt());
}
if (renderingSurface.textureHandle != 0) {
dev.log(
"Creating render target from native texture ${renderingSurface.textureHandle}");
create_render_target_ffi(_viewer!, renderingSurface.textureHandle,
rect.value!.width.toInt(), rect.value!.height.toInt());
}
textureDetails.value = TextureDetails(
textureId: renderingSurface.flutterTextureId,
width: rect.value!.width.toInt(),
height: rect.value!.height.toInt());
update_viewport_and_camera_projection_ffi(
_viewer!, rect.value!.width.toInt(), rect.value!.height.toInt(), 1.0);
await setRendering(_rendering);
_resizing = false;
}
@override
Future clearBackgroundImage() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
clear_background_image_ffi(_viewer!);
}
@override
Future setBackgroundImage(String path, {bool fillHeight = false}) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_background_image_ffi(
_viewer!, path.toNativeUtf8().cast<Char>(), fillHeight);
}
@override
Future setBackgroundColor(Color color) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_background_color_ffi(
_viewer!,
color.red.toDouble() / 255.0,
color.green.toDouble() / 255.0,
color.blue.toDouble() / 255.0,
color.alpha.toDouble() / 255.0);
}
@override
Future setBackgroundImagePosition(double x, double y,
{bool clamp = false}) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_background_image_position_ffi(_viewer!, x, y, clamp);
}
@override
Future loadSkybox(String skyboxPath) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
load_skybox_ffi(_viewer!, skyboxPath.toNativeUtf8().cast<Char>());
}
@override
Future loadIbl(String lightingPath, {double intensity = 30000}) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
load_ibl_ffi(_viewer!, lightingPath.toNativeUtf8().cast<Char>(), intensity);
}
@override
Future removeSkybox() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
remove_skybox_ffi(_viewer!);
}
@override
Future removeIbl() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
remove_ibl_ffi(_viewer!);
}
@override
Future<FilamentEntity> addLight(
int type,
double colour,
double intensity,
double posX,
double posY,
double posZ,
double dirX,
double dirY,
double dirZ,
bool castShadows) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
var entity = add_light_ffi(_viewer!, type, colour, intensity, posX, posY,
posZ, dirX, dirY, dirZ, castShadows);
return entity;
}
@override
Future removeLight(FilamentEntity light) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
remove_light_ffi(_viewer!, light);
}
@override
Future clearLights() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
clear_lights_ffi(_viewer!);
}
@override
Future<FilamentEntity> loadGlb(String path, {bool unlit = false}) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
if (unlit) {
throw Exception("Not yet implemented");
}
var asset =
load_glb_ffi(_assetManager!, path.toNativeUtf8().cast<Char>(), unlit);
if (asset == _FILAMENT_ASSET_ERROR) {
throw Exception("An error occurred loading the asset at $path");
}
return asset;
}
@override
Future<FilamentEntity> loadGltf(String path, String relativeResourcePath,
{bool force = false}) async {
if (Platform.isWindows && !force) {
throw Exception(
"loadGltf has a race condition on Windows which is likely to crash your program. If you really want to try, pass force=true to loadGltf");
}
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
var asset = load_gltf_ffi(_assetManager!, path.toNativeUtf8().cast<Char>(),
relativeResourcePath.toNativeUtf8().cast<Char>());
if (asset == _FILAMENT_ASSET_ERROR) {
throw Exception("An error occurred loading the asset at $path");
}
return asset;
}
@override
Future panStart(double x, double y) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
grab_begin(_viewer!, x * _pixelRatio, y * _pixelRatio, true);
}
@override
Future panUpdate(double x, double y) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
grab_update(_viewer!, x * _pixelRatio, y * _pixelRatio);
}
@override
Future panEnd() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
grab_end(_viewer!);
}
@override
Future rotateStart(double x, double y) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
grab_begin(_viewer!, x * _pixelRatio, y * _pixelRatio, false);
}
@override
Future rotateUpdate(double x, double y) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
grab_update(_viewer!, x * _pixelRatio, y * _pixelRatio);
}
@override
Future rotateEnd() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
grab_end(_viewer!);
}
@override
Future setMorphTargetWeights(
FilamentEntity entity, String meshName, List<double> weights) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
var weightsPtr = calloc<Float>(weights.length);
for (int i = 0; i < weights.length; i++) {
weightsPtr.elementAt(i).value = weights[i];
}
set_morph_target_weights_ffi(_assetManager!, entity,
meshName.toNativeUtf8().cast<Char>(), weightsPtr, weights.length);
calloc.free(weightsPtr);
}
@override
Future<List<String>> getMorphTargetNames(
FilamentEntity entity, String meshName) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
var names = <String>[];
var count = get_morph_target_name_count_ffi(
_assetManager!, entity, meshName.toNativeUtf8().cast<Char>());
var outPtr = calloc<Char>(255);
for (int i = 0; i < count; i++) {
get_morph_target_name(_assetManager!, entity,
meshName.toNativeUtf8().cast<Char>(), outPtr, i);
names.add(outPtr.cast<Utf8>().toDartString());
}
calloc.free(outPtr);
return names.cast<String>();
}
@override
Future<List<String>> getAnimationNames(FilamentEntity entity) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
var animationCount = get_animation_count(_assetManager!, entity);
var names = <String>[];
var outPtr = calloc<Char>(255);
for (int i = 0; i < animationCount; i++) {
get_animation_name_ffi(_assetManager!, entity, outPtr, i);
names.add(outPtr.cast<Utf8>().toDartString());
}
return names;
}
@override
Future<double> getAnimationDuration(
FilamentEntity entity, int animationIndex) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
var duration =
get_animation_duration(_assetManager!, entity, animationIndex);
return duration;
}
@override
Future setMorphAnimationData(
FilamentEntity entity, MorphAnimationData animation) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
var dataPtr = calloc<Float>(animation.data.length);
for (int i = 0; i < animation.data.length; i++) {
dataPtr.elementAt(i).value = animation.data[i];
}
// the morph targets in [animation] might be a subset of those that actually exist in the mesh (and might not have the same order)
// we don't want to reorder the data (?? or do we? this is probably more efficient for the backend?)
// so let's get the actual list of morph targets from the mesh and pass the relevant indices to the native side.
var meshMorphTargets =
await getMorphTargetNames(entity, animation.meshName);
Pointer<Int> idxPtr = calloc<Int>(animation.morphTargets.length);
for (int i = 0; i < animation.numMorphTargets; i++) {
var index = meshMorphTargets.indexOf(animation.morphTargets[i]);
if (index == -1) {
calloc.free(dataPtr);
calloc.free(idxPtr);
throw Exception(
"Morph target ${animation.morphTargets[i]} is specified in the animation but could not be found in the mesh ${animation.meshName} under entity $entity");
}
idxPtr.elementAt(i).value = index;
}
set_morph_animation(
_assetManager!,
entity,
animation.meshName.toNativeUtf8().cast<Char>(),
dataPtr,
idxPtr,
animation.numMorphTargets,
animation.numFrames,
(animation.frameLengthInMs));
calloc.free(dataPtr);
calloc.free(idxPtr);
}
@override
Future setBoneAnimation(
FilamentEntity entity, BoneAnimationData animation) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
// var data = calloc<Float>(animation.frameData.length);
// int offset = 0;
// var numFrames = animation.frameData.length ~/ 7;
// var boneNames = calloc<Pointer<Char>>(1);
// boneNames.elementAt(0).value =
// animation.boneName.toNativeUtf8().cast<Char>();
// var meshNames = calloc<Pointer<Char>>(animation.meshNames.length);
// for (int i = 0; i < animation.meshNames.length; i++) {
// meshNames.elementAt(i).value =
// animation.meshNames[i].toNativeUtf8().cast<Char>();
// }
// for (int i = 0; i < animation.frameData.length; i++) {
// data.elementAt(offset).value = animation.frameData[i];
// offset += 1;
// }
// await _channel.invokeMethod("setBoneAnimation", [
// _assetManager!,
// asset,
// data,
// numFrames,
// 1,
// boneNames,
// meshNames,
// animation.meshNames.length,
// animation.frameLengthInMs
// ]);
// calloc.free(data);
}
@override
Future removeAsset(FilamentEntity entity) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
remove_asset_ffi(_viewer!, entity);
}
@override
Future clearAssets() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
clear_assets_ffi(_viewer!);
}
@override
Future zoomBegin() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
scroll_begin(_viewer!);
}
@override
Future zoomUpdate(double x, double y, double z) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
scroll_update(_viewer!, x, y, z);
}
@override
Future zoomEnd() async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
scroll_end(_viewer!);
}
@override
Future playAnimation(FilamentEntity entity, int index,
{bool loop = false,
bool reverse = false,
bool replaceActive = true,
double crossfade = 0.0}) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
play_animation_ffi(
_assetManager!, entity, index, loop, reverse, replaceActive, crossfade);
}
@override
Future setAnimationFrame(
FilamentEntity entity, int index, int animationFrame) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_animation_frame(_assetManager!, entity, index, animationFrame);
}
@override
Future stopAnimation(FilamentEntity entity, int animationIndex) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
stop_animation(_assetManager!, entity, animationIndex);
}
@override
Future setCamera(FilamentEntity entity, String? name) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
var result = set_camera(
_viewer!, entity, name?.toNativeUtf8().cast<Char>() ?? nullptr);
if (!result) {
throw Exception("Failed to set camera");
}
}
@override
Future setToneMapping(ToneMapper mapper) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_tone_mapping_ffi(_viewer!, mapper.index);
}
@override
Future setPostProcessing(bool enabled) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_post_processing_ffi(_viewer!, enabled);
}
@override
Future setBloom(double bloom) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_bloom_ffi(_viewer!, bloom);
}
@override
Future setCameraFocalLength(double focalLength) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_camera_focal_length(_viewer!, focalLength);
}
@override
Future setCameraFocusDistance(double focusDistance) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_camera_focus_distance(_viewer!, focusDistance);
}
@override
Future setCameraPosition(double x, double y, double z) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_camera_position(_viewer!, x, y, z);
}
@override
Future moveCameraToAsset(FilamentEntity entity) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
move_camera_to_asset(_viewer!, entity);
}
@override
Future setViewFrustumCulling(bool enabled) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_view_frustum_culling(_viewer!, enabled);
}
@override
Future setCameraExposure(
double aperture, double shutterSpeed, double sensitivity) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_camera_exposure(_viewer!, aperture, shutterSpeed, sensitivity);
}
@override
Future setCameraRotation(double rads, double x, double y, double z) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_camera_rotation(_viewer!, rads, x, y, z);
}
@override
Future setCameraModelMatrix(List<double> matrix) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
assert(matrix.length == 16);
var ptr = calloc<Float>(16);
for (int i = 0; i < 16; i++) {
ptr.elementAt(i).value = matrix[i];
}
set_camera_model_matrix(_viewer!, ptr);
calloc.free(ptr);
}
@override
Future setMaterialColor(FilamentEntity entity, String meshName,
int materialIndex, Color color) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
var result = set_material_color(
_assetManager!,
entity,
meshName.toNativeUtf8().cast<Char>(),
materialIndex,
color.red.toDouble() / 255.0,
color.green.toDouble() / 255.0,
color.blue.toDouble() / 255.0,
color.alpha.toDouble() / 255.0);
if (!result) {
throw Exception("Failed to set material color");
}
}
@override
Future transformToUnitCube(FilamentEntity entity) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
transform_to_unit_cube(_assetManager!, entity);
}
@override
Future setPosition(
FilamentEntity entity, double x, double y, double z) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_position(_assetManager!, entity, x, y, z);
}
@override
Future setScale(FilamentEntity entity, double scale) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_scale(_assetManager!, entity, scale);
}
@override
Future setRotation(
FilamentEntity entity, double rads, double x, double y, double z) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
set_rotation(_assetManager!, entity, rads, x, y, z);
}
@override
Future hide(FilamentEntity entity, String meshName) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
if (hide_mesh(
_assetManager!, entity, meshName.toNativeUtf8().cast<Char>()) !=
1) {}
}
@override
Future reveal(FilamentEntity entity, String meshName) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
if (reveal_mesh(
_assetManager!, entity, meshName.toNativeUtf8().cast<Char>()) !=
1) {
throw Exception("Failed to reveal mesh $meshName");
}
}
@override
String? getNameForEntity(FilamentEntity entity) {
final result = get_name_for_entity(_assetManager!, entity);
if (result == nullptr) {
return null;
}
return result.cast<Utf8>().toDartString();
}
@override
void pick(int x, int y) async {
if (_viewer == null) {
throw Exception("No viewer available, ignoring");
}
final outPtr = calloc<EntityId>(1);
outPtr.value = 0;
pick_ffi(_viewer!, x, textureDetails.value!.height - y, outPtr);
int wait = 0;
while (outPtr.value == 0) {
await Future.delayed(const Duration(milliseconds: 32));
wait++;
if (wait > 10) {
calloc.free(outPtr);
throw Exception("Failed to get picking result");
}
}
var entityId = outPtr.value;
_pickResultController.add(entityId);
calloc.free(outPtr);
}
@override
Future<Matrix4> getCameraViewMatrix() async {
if (_viewer == null) {
throw Exception("No viewer available");
}
var arrayPtr = get_camera_view_matrix(_viewer!);
var viewMatrix = Matrix4.fromList(arrayPtr.asTypedList(16));
calloc.free(arrayPtr);
return viewMatrix;
}
@override
Future<Matrix4> getCameraModelMatrix() async {
if (_viewer == null) {
throw Exception("No viewer available");
}
var arrayPtr = get_camera_model_matrix(_viewer!);
var modelMatrix = Matrix4.fromList(arrayPtr.asTypedList(16));
calloc.free(arrayPtr);
return modelMatrix;
}
@override
Future<Vector3> getCameraPosition() async {
if (_viewer == null) {
throw Exception("No viewer available");
}
var arrayPtr = get_camera_model_matrix(_viewer!);
var doubleList = arrayPtr.asTypedList(16);
var modelMatrix = Matrix4.fromFloat64List(doubleList);
var position = modelMatrix.getColumn(3).xyz;
calloc.free(arrayPtr);
return position;
}
@override
Future<Matrix3> getCameraRotation() async {
if (_viewer == null) {
throw Exception("No viewer available");
}
var arrayPtr = get_camera_model_matrix(_viewer!);
var doubleList = arrayPtr.asTypedList(16);
var modelMatrix = Matrix4.fromFloat64List(doubleList);
var rotationMatrix = Matrix3.identity();
modelMatrix.copyRotation(rotationMatrix);
calloc.free(arrayPtr);
return rotationMatrix;
}
@override
Future setCameraManipulatorOptions(
{ManipulatorMode mode = ManipulatorMode.ORBIT,
double orbitSpeedX = 0.01,
double orbitSpeedY = 0.01,
double zoomSpeed = 0.01}) async {
if (_viewer == null) {
throw Exception("No viewer available");
}
if (mode != ManipulatorMode.ORBIT) {
throw Exception("Manipulator mode $mode not yet implemented");
}
set_camera_manipulator_options(
_viewer!, mode.index, orbitSpeedX, orbitSpeedX, zoomSpeed);
}
}