// // KasenAPIExample.cpp // plugins/KasenAPIExample/src // // Created by Kasen IO on 2019.07.14 | realities.dev | kasenvr@gmail.com // Copyright 2019 Kasen IO // // Distributed under the Apache License, Version 2.0. // See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html // // Example of prototyping new JS APIs by leveraging the existing plugin system. #include "ExampleScriptPlugin.h" #include #include #include #include #include #include #include namespace custom_api_example { QLoggingCategory logger{ "custom_api_example" }; class KasenAPIExample : public example::ScriptPlugin { Q_OBJECT Q_PLUGIN_METADATA(IID "KasenAPIExample" FILE "plugin.json") public: KasenAPIExample() : example::ScriptPlugin("KasenAPIExample", "0.0.1") { qCInfo(logger) << "plugin loaded" << qApp << toString() << QThread::currentThread(); } public slots: /**jsdoc * Returns current microseconds (usecs) since Epoch. note: 1000usecs == 1ms * @example Measure current setTimeout accuracy. * var expected = 1000; * var start = KasenAPIExample.now(); * Script.setTimeout(function () { * var elapsed = (KasenAPIExample.now() - start)/1000; * print("expected (ms):", expected, "actual (ms):", elapsed); * }, expected); */ QVariant now() const { return usecTimestampNow(); } /**jsdoc * Returns the available blendshape names for an avatar. * @example Get blendshape names * print(JSON.stringify(KasenAPIExample.getBlendshapeNames(MyAvatar.sessionUUID))); */ QStringList getBlendshapeNames(const QUuid& avatarID) const { QVector out; if (auto head = getAvatarHead(avatarID)) { for (const auto& kv : head->getBlendshapeMap().toStdMap()) { if (kv.second >= out.size()) out.resize(kv.second+1); out[kv.second] = kv.first; } } return out.toList(); } /**jsdoc * Returns a key-value object with active (non-zero) blendshapes. * eg: { JawOpen: 1.0, ... } * @example Get active blendshape map * print(JSON.stringify(KasenAPIExample.getActiveBlendshapes(MyAvatar.sessionUUID))); */ QVariant getActiveBlendshapes(const QUuid& avatarID) const { if (auto head = getAvatarHead(avatarID)) { return head->toJson()["blendShapes"].toVariant(); } return {}; } QVariant getBlendshapeMapping(const QUuid& avatarID) const { QVariantMap out; if (auto head = getAvatarHead(avatarID)) { for (const auto& kv : head->getBlendshapeMap().toStdMap()) { out[kv.first] = kv.second; } } return out; } QVariant getBlendshapes(const QUuid& avatarID) const { QVariantMap result; if (auto head = getAvatarHead(avatarID)) { QStringList names = getBlendshapeNames(avatarID); auto states = head->getBlendshapeStates(); result = { { "base", zipNonZeroValues(names, states.base) }, { "summed", zipNonZeroValues(names, states.summed) }, { "transient", zipNonZeroValues(names, states.transient) }, }; } return result; } private: static QVariantMap zipNonZeroValues(const QStringList& keys, const QVector& values) { QVariantMap out; for (int i=1; i < values.size(); i++) { if (fabs(values[i]) > 1.0e-6) { out[keys.value(i)] = values[i]; } } return out; } struct _HeadHelper : public HeadData { QMap getBlendshapeMap() const { return _blendshapeLookupMap; } struct States { QVector base, summed, transient; }; States getBlendshapeStates() const { return { _blendshapeCoefficients, _summedBlendshapeCoefficients, _transientBlendshapeCoefficients }; } }; static const _HeadHelper* getAvatarHead(const QUuid& avatarID) { auto avatars = DependencyManager::get(); auto avatar = avatars ? avatars->getAvatarBySessionID(avatarID) : nullptr; auto head = avatar ? avatar->getHeadData() : nullptr; return reinterpret_cast(head); } }; } const QLoggingCategory& example::logger{ custom_api_example::logger }; #include "KasenAPIExample.moc"