18#if IMPELLER_SUPPORTS_RENDERING
22#include "third_party/skia/include/core/SkImage.h"
23#include "third_party/skia/include/core/SkSurface.h"
24#include "third_party/skia/include/encode/SkPngEncoder.h"
39void FinalizeSkData(
void* isolate_callback_data,
void* peer) {
40 SkData*
buffer =
reinterpret_cast<SkData*
>(peer);
44void InvokeDataCallback(std::unique_ptr<DartPersistentValue>
callback,
46 std::shared_ptr<tonic::DartState> dart_state =
callback->dart_state().lock();
52 std::string error_copy(
buffer.status().message());
53 Dart_Handle dart_string =
ToDart(error_copy);
60 void* bytes =
const_cast<void*
>(
buffer.value()->data());
62 void* peer =
reinterpret_cast<void*
>(
buffer.value().release());
63 Dart_Handle dart_data = Dart_NewExternalTypedDataWithFinalizer(
64 Dart_TypedData_kUint8, bytes,
length, peer,
length, FinalizeSkData);
69 const sk_sp<SkImage>& raster_image,
76 if (!raster_image->peekPixels(&pixmap)) {
78 "Could not copy pixels from the raster image.");
83 return SkData::MakeWithCopy(pixmap.addr(), pixmap.computeByteSize());
87 auto surface = SkSurfaces::Raster(
88 SkImageInfo::Make(raster_image->width(), raster_image->height(),
93 "Could not set up the surface for swizzle.");
96 surface->writePixels(pixmap, 0, 0);
98 if (!
surface->peekPixels(&pixmap)) {
100 "Pixel address is not available.");
103 return SkData::MakeWithCopy(pixmap.addr(), pixmap.computeByteSize());
106void EncodeImageAndInvokeDataCallback(
107 const sk_sp<DlImage>&
image,
108 std::unique_ptr<DartPersistentValue>
callback,
115 const std::shared_ptr<const fml::SyncSwitch>& is_gpu_disabled_sync_switch,
116 const std::shared_ptr<impeller::Context>& impeller_context,
117 bool is_impeller_enabled) {
121 InvokeDataCallback(std::move(
callback), std::move(encoded));
127 [callback_task = std::move(callback_task),
format,
129 if (raster_image.ok()) {
132 ui_task_runner->PostTask([callback_task = callback_task,
133 encoded = std::move(encoded)]()
mutable {
134 callback_task(std::move(encoded));
137 ui_task_runner->PostTask([callback_task = callback_task,
138 raster_image = raster_image]()
mutable {
139 callback_task(raster_image.status());
145#if IMPELLER_SUPPORTS_RENDERING
146 if (is_impeller_enabled) {
148 image, encode_task, raster_task_runner, io_task_runner,
149 snapshot_delegate, is_gpu_disabled_sync_switch, impeller_context);
155 io_task_runner, resource_context, snapshot_delegate,
156 is_gpu_disabled_sync_switch);
159 FML_LOG(FATAL) <<
"Unsupported renderer.";
167 Dart_Handle callback_handle) {
169 return ToDart(
"encode called with non-genuine Image.");
172 if (!Dart_IsClosure(callback_handle)) {
173 return ToDart(
"Callback must be a function.");
178 auto callback = std::make_unique<DartPersistentValue>(
181#if IMPELLER_SUPPORTS_RENDERING && FML_OS_IOS_SIMULATOR
182 if (canvas_image->
image()->IsFakeImage()) {
183 sk_sp<SkData>
data = SkData::MakeEmpty();
195 image_format, ui_task_runner = task_runners.GetUITaskRunner(),
196 raster_task_runner = task_runners.GetRasterTaskRunner(),
197 io_task_runner = task_runners.GetIOTaskRunner(),
200 is_impeller_enabled =
202 EncodeImageAndInvokeDataCallback(
203 image, std::move(callback), image_format, ui_task_runner,
204 raster_task_runner, io_task_runner,
205 io_manager->GetResourceContext(), snapshot_delegate,
206 io_manager->GetIsGpuDisabledSyncSwitch(),
207 io_manager->GetImpellerContext(), is_impeller_enabled);
223 auto png_image = SkPngEncoder::Encode(
nullptr, raster_image.get(), {});
225 if (png_image ==
nullptr) {
227 "Could not convert raster image to PNG.");
232 return CopyImageByteData(raster_image, kRGBA_8888_SkColorType,
233 kPremul_SkAlphaType);
236 return CopyImageByteData(raster_image, kRGBA_8888_SkColorType,
237 kUnpremul_SkAlphaType);
240 return CopyImageByteData(raster_image, raster_image->colorType(),
241 raster_image->alphaType());
243 return CopyImageByteData(raster_image, kRGBA_F32_SkColorType,
244 kUnpremul_SkAlphaType);
248 "Unknown error encoding image.");
sk_sp< DlImage > image() const
static void ConvertImageToRaster(const sk_sp< DlImage > &dl_image, std::function< void(fml::StatusOr< sk_sp< SkImage > >)> encode_task, const fml::RefPtr< fml::TaskRunner > &raster_task_runner, const fml::RefPtr< fml::TaskRunner > &io_task_runner, const fml::TaskRunnerAffineWeakPtr< SnapshotDelegate > &snapshot_delegate, const std::shared_ptr< const fml::SyncSwitch > &is_gpu_disabled_sync_switch, const std::shared_ptr< impeller::Context > &impeller_context)
fml::RefPtr< fml::TaskRunner > GetIOTaskRunner() const
fml::WeakPtr< IOManager > GetIOManager() const
const TaskRunners & GetTaskRunners() const
bool IsImpellerEnabled() const
Whether Impeller is enabled for this application.
fml::TaskRunnerAffineWeakPtr< SnapshotDelegate > GetSnapshotDelegate() const
static UIDartState * Current()
virtual void PostTask(const fml::closure &task) override
static DartState * Current()
FlutterVulkanImage * image
uint32_t uint32_t * format
FlutterDesktopBinaryReply callback
#define FML_LOG(severity)
#define FML_DCHECK(condition)
DEF_SWITCHES_START aot vmservice shared library Name of the *so containing AOT compiled Dart assets for launching the service isolate vm snapshot data
void ConvertImageToRasterSkia(const sk_sp< DlImage > &dl_image, std::function< void(sk_sp< SkImage >)> encode_task, const fml::RefPtr< fml::TaskRunner > &raster_task_runner, const fml::RefPtr< fml::TaskRunner > &io_task_runner, const fml::WeakPtr< GrDirectContext > &resource_context, const fml::TaskRunnerAffineWeakPtr< SnapshotDelegate > &snapshot_delegate, const std::shared_ptr< const fml::SyncSwitch > &is_gpu_disabled_sync_switch)
Dart_Handle EncodeImage(CanvasImage *canvas_image, int format, Dart_Handle callback_handle)
DEF_SWITCHES_START aot vmservice shared library Name of the *so containing AOT compiled Dart assets for launching the service isolate vm snapshot The VM snapshot data that will be memory mapped as read only SnapshotAssetPath must be present isolate snapshot The isolate snapshot data that will be memory mapped as read only SnapshotAssetPath must be present cache dir Path to the cache directory This is different from the persistent_cache_path in embedder which is used for Skia shader cache icu native lib Path to the library file that exports the ICU data vm service The hostname IP address on which the Dart VM Service should be served If not defaults to or::depending on whether ipv6 is specified disable vm Disable the Dart VM Service The Dart VM Service is never available in release mode Bind to the IPv6 localhost address for the Dart VM Service Ignored if vm service host is set profile Make the profiler discard new samples once the profiler sample buffer is full When this flag is not the profiler sample buffer is used as a ring buffer
internal::CopyableLambda< T > MakeCopyable(T lambda)
Dart_Handle ToDart(const T &object)
Dart_Handle DartInvoke(Dart_Handle closure, std::initializer_list< Dart_Handle > args)
#define TRACE_EVENT0(category_group, name)