2019-03-06 14:32:55 +00:00
|
|
|
/*
|
|
|
|
* Copyright 2019 Google LLC
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
|
|
|
*/
|
|
|
|
|
2019-04-23 17:05:21 +00:00
|
|
|
#include "include/core/SkCanvas.h"
|
|
|
|
#include "include/core/SkTypes.h"
|
|
|
|
#include "include/utils/SkRandom.h"
|
|
|
|
#include "modules/particles/include/SkParticleEffect.h"
|
|
|
|
#include "modules/particles/include/SkParticleSerialization.h"
|
2019-11-27 15:34:18 +00:00
|
|
|
#include "modules/skresources/include/SkResources.h"
|
2021-04-13 14:41:57 +00:00
|
|
|
#include "src/sksl/codegen/SkSLVMCodeGenerator.h"
|
2019-03-06 14:32:55 +00:00
|
|
|
|
|
|
|
#include <string>
|
|
|
|
|
2020-03-26 13:27:48 +00:00
|
|
|
#include "modules/canvaskit/WasmCommon.h"
|
2020-02-21 13:26:59 +00:00
|
|
|
|
2019-03-06 14:32:55 +00:00
|
|
|
#include <emscripten.h>
|
|
|
|
#include <emscripten/bind.h>
|
|
|
|
|
|
|
|
using namespace emscripten;
|
|
|
|
|
2019-11-27 15:34:18 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
class ParticleAssetProvider : public skresources::ResourceProvider {
|
|
|
|
public:
|
|
|
|
~ParticleAssetProvider() override = default;
|
|
|
|
|
|
|
|
// Tried using a map, but that gave strange errors like
|
|
|
|
// https://emscripten.org/docs/porting/guidelines/function_pointer_issues.html
|
|
|
|
// Not entirely sure why, but perhaps the iterator in the map was
|
|
|
|
// confusing enscripten.
|
|
|
|
using AssetVec = std::vector<std::pair<SkString, sk_sp<SkData>>>;
|
|
|
|
|
|
|
|
static sk_sp<ParticleAssetProvider> Make(AssetVec assets) {
|
|
|
|
if (assets.empty()) {
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
return sk_sp<ParticleAssetProvider>(new ParticleAssetProvider(std::move(assets)));
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<skresources::ImageAsset> loadImageAsset(const char[] /* path */,
|
|
|
|
const char name[],
|
|
|
|
const char[] /* id */) const override {
|
|
|
|
// For CK we ignore paths & IDs, and identify images based solely on name.
|
|
|
|
if (auto data = this->findAsset(name)) {
|
|
|
|
return skresources::MultiFrameImageAsset::Make(std::move(data));
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
sk_sp<SkData> loadFont(const char name[], const char[] /* url */) const override {
|
|
|
|
// Same as images paths, we ignore font URLs.
|
|
|
|
return this->findAsset(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
explicit ParticleAssetProvider(AssetVec assets) : fAssets(std::move(assets)) {}
|
|
|
|
|
|
|
|
sk_sp<SkData> findAsset(const char name[]) const {
|
|
|
|
for (const auto& asset : fAssets) {
|
|
|
|
if (asset.first.equals(name)) {
|
|
|
|
return asset.second;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SkDebugf("Could not find %s\n", name);
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
const AssetVec fAssets;
|
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-02-03 21:12:09 +00:00
|
|
|
struct ParticleUniform {
|
2020-02-21 13:26:59 +00:00
|
|
|
int columns;
|
|
|
|
int rows;
|
|
|
|
int slot; // the index into the uniforms array that this uniform begins.
|
|
|
|
};
|
|
|
|
|
2021-02-03 21:12:09 +00:00
|
|
|
ParticleUniform fromUniform(const SkSL::UniformInfo::Uniform& u) {
|
|
|
|
ParticleUniform su;
|
2020-02-21 13:26:59 +00:00
|
|
|
su.columns = u.fColumns;
|
|
|
|
su.rows = u.fRows;
|
|
|
|
su.slot = u.fSlot;
|
|
|
|
return su;
|
|
|
|
}
|
|
|
|
|
2019-03-06 14:32:55 +00:00
|
|
|
EMSCRIPTEN_BINDINGS(Particles) {
|
2020-10-07 20:09:22 +00:00
|
|
|
class_<SkParticleEffect>("ParticleEffect")
|
2019-03-06 14:32:55 +00:00
|
|
|
.smart_ptr<sk_sp<SkParticleEffect>>("sk_sp<SkParticleEffect>")
|
|
|
|
.function("draw", &SkParticleEffect::draw, allow_raw_pointers())
|
2021-01-29 19:32:09 +00:00
|
|
|
.function("_uniformPtr", optional_override([](SkParticleEffect& self)->uintptr_t {
|
|
|
|
return reinterpret_cast<uintptr_t>(self.uniformData());
|
2020-02-21 13:26:59 +00:00
|
|
|
}))
|
2021-01-29 19:32:09 +00:00
|
|
|
.function("getUniformCount", optional_override([](SkParticleEffect& self)->int {
|
|
|
|
auto info = self.uniformInfo();
|
|
|
|
if (!info) {
|
2020-02-21 13:26:59 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2021-01-29 19:32:09 +00:00
|
|
|
return info->fUniforms.size();
|
2020-02-21 13:26:59 +00:00
|
|
|
}))
|
2021-01-29 19:32:09 +00:00
|
|
|
.function("getUniformFloatCount", optional_override([](SkParticleEffect& self)->int {
|
|
|
|
auto info = self.uniformInfo();
|
|
|
|
if (!info) {
|
2020-02-21 13:26:59 +00:00
|
|
|
return -1;
|
|
|
|
}
|
2021-01-29 19:32:09 +00:00
|
|
|
return info->fUniformSlotCount;
|
2020-02-21 13:26:59 +00:00
|
|
|
}))
|
2021-01-29 19:32:09 +00:00
|
|
|
.function("getUniformName", optional_override([](SkParticleEffect& self, int i)->JSString {
|
|
|
|
auto info = self.uniformInfo();
|
|
|
|
if (!info) {
|
2020-02-21 13:26:59 +00:00
|
|
|
return emscripten::val::null();
|
|
|
|
}
|
2021-01-29 19:32:09 +00:00
|
|
|
return emscripten::val(info->fUniforms[i].fName.c_str());
|
2020-02-21 13:26:59 +00:00
|
|
|
}))
|
2021-02-03 21:12:09 +00:00
|
|
|
.function("getUniform", optional_override([](SkParticleEffect& self, int i)->ParticleUniform {
|
|
|
|
ParticleUniform su;
|
2021-01-29 19:32:09 +00:00
|
|
|
auto info = self.uniformInfo();
|
|
|
|
if (!info) {
|
2020-02-21 13:26:59 +00:00
|
|
|
return su;
|
|
|
|
}
|
2021-01-29 19:32:09 +00:00
|
|
|
su = fromUniform(info->fUniforms[i]);
|
2020-02-21 13:26:59 +00:00
|
|
|
return su;
|
|
|
|
}))
|
2021-02-02 13:18:11 +00:00
|
|
|
.function("_setPosition", optional_override([](SkParticleEffect& self,
|
|
|
|
SkScalar x, SkScalar y)->void {
|
|
|
|
self.setPosition({x, y});
|
|
|
|
}))
|
2020-02-21 13:26:59 +00:00
|
|
|
.function("setRate", select_overload<void (float)>(&SkParticleEffect::setRate))
|
|
|
|
.function("start", select_overload<void (double, bool)>(&SkParticleEffect::start))
|
|
|
|
.function("update", select_overload<void (double)>(&SkParticleEffect::update));
|
|
|
|
|
2021-02-03 21:12:09 +00:00
|
|
|
value_object<ParticleUniform>("ParticleUniform")
|
|
|
|
.field("columns", &ParticleUniform::columns)
|
|
|
|
.field("rows", &ParticleUniform::rows)
|
|
|
|
.field("slot", &ParticleUniform::slot);
|
2019-03-06 14:32:55 +00:00
|
|
|
|
2019-11-27 15:34:18 +00:00
|
|
|
function("_MakeParticles", optional_override([](std::string json,
|
|
|
|
size_t assetCount,
|
|
|
|
uintptr_t /* char** */ nptr,
|
|
|
|
uintptr_t /* uint8_t** */ dptr,
|
|
|
|
uintptr_t /* size_t* */ sptr)
|
|
|
|
->sk_sp<SkParticleEffect> {
|
|
|
|
// See the comment in canvaskit_bindings.cpp about the use of uintptr_t
|
2019-03-06 14:32:55 +00:00
|
|
|
static bool didInit = false;
|
|
|
|
if (!didInit) {
|
2019-08-30 14:59:47 +00:00
|
|
|
SkParticleEffect::RegisterParticleTypes();
|
2019-03-06 14:32:55 +00:00
|
|
|
didInit = true;
|
|
|
|
}
|
2019-11-27 15:34:18 +00:00
|
|
|
|
|
|
|
const auto assetNames = reinterpret_cast<char** >(nptr);
|
|
|
|
const auto assetDatas = reinterpret_cast<uint8_t**>(dptr);
|
|
|
|
const auto assetSizes = reinterpret_cast<size_t* >(sptr);
|
|
|
|
|
|
|
|
ParticleAssetProvider::AssetVec assets;
|
|
|
|
assets.reserve(assetCount);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < assetCount; i++) {
|
|
|
|
auto name = SkString(assetNames[i]);
|
|
|
|
auto bytes = SkData::MakeFromMalloc(assetDatas[i], assetSizes[i]);
|
|
|
|
assets.push_back(std::make_pair(std::move(name), std::move(bytes)));
|
|
|
|
}
|
|
|
|
|
2019-03-06 14:32:55 +00:00
|
|
|
sk_sp<SkParticleEffectParams> params(new SkParticleEffectParams());
|
|
|
|
skjson::DOM dom(json.c_str(), json.length());
|
|
|
|
SkFromJsonVisitor fromJson(dom.root());
|
|
|
|
params->visitFields(&fromJson);
|
2019-12-02 21:52:51 +00:00
|
|
|
params->prepare(skresources::DataURIResourceProviderProxy::Make(
|
|
|
|
ParticleAssetProvider::Make(std::move(assets))).get());
|
2020-02-10 18:45:22 +00:00
|
|
|
return sk_sp<SkParticleEffect>(new SkParticleEffect(std::move(params)));
|
2019-03-06 14:32:55 +00:00
|
|
|
}));
|
|
|
|
constant("particles", true);
|
|
|
|
|
|
|
|
}
|