Florin Malita | a831655 | 2018-11-09 16:19:44 -0500 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2018 Google Inc. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license that can be |
| 5 | * found in the LICENSE file. |
| 6 | */ |
| 7 | |
Mike Klein | c0bd9f9 | 2019-04-23 12:05:21 -0500 | [diff] [blame] | 8 | #include "modules/skottie/utils/SkottieUtils.h" |
Florin Malita | a831655 | 2018-11-09 16:19:44 -0500 | [diff] [blame] | 9 | |
Florin Malita | a831655 | 2018-11-09 16:19:44 -0500 | [diff] [blame] | 10 | namespace skottie_utils { |
| 11 | |
Florin Malita | 91af8d8 | 2018-11-30 16:46:45 -0500 | [diff] [blame] | 12 | class CustomPropertyManager::PropertyInterceptor final : public skottie::PropertyObserver { |
| 13 | public: |
| 14 | explicit PropertyInterceptor(CustomPropertyManager* mgr) : fMgr(mgr) {} |
Florin Malita | 8ac81b7 | 2018-11-28 11:39:39 -0500 | [diff] [blame] | 15 | |
Florin Malita | 91af8d8 | 2018-11-30 16:46:45 -0500 | [diff] [blame] | 16 | void onColorProperty(const char node_name[], |
| 17 | const LazyHandle<skottie::ColorPropertyHandle>& c) override { |
Florin Malita | 3f45e4b | 2020-08-20 14:51:41 -0400 | [diff] [blame] | 18 | const auto key = fMgr->acceptKey(node_name, ".Color"); |
| 19 | if (!key.empty()) { |
| 20 | fMgr->fColorMap[key].push_back(c()); |
| 21 | } |
Florin Malita | 8ac81b7 | 2018-11-28 11:39:39 -0500 | [diff] [blame] | 22 | } |
Florin Malita | 8ac81b7 | 2018-11-28 11:39:39 -0500 | [diff] [blame] | 23 | |
Florin Malita | 91af8d8 | 2018-11-30 16:46:45 -0500 | [diff] [blame] | 24 | void onOpacityProperty(const char node_name[], |
| 25 | const LazyHandle<skottie::OpacityPropertyHandle>& o) override { |
Florin Malita | 3f45e4b | 2020-08-20 14:51:41 -0400 | [diff] [blame] | 26 | const auto key = fMgr->acceptKey(node_name, ".Opacity"); |
| 27 | if (!key.empty()) { |
| 28 | fMgr->fOpacityMap[key].push_back(o()); |
| 29 | } |
Florin Malita | 8ac81b7 | 2018-11-28 11:39:39 -0500 | [diff] [blame] | 30 | } |
Florin Malita | 8ac81b7 | 2018-11-28 11:39:39 -0500 | [diff] [blame] | 31 | |
Florin Malita | 91af8d8 | 2018-11-30 16:46:45 -0500 | [diff] [blame] | 32 | void onTransformProperty(const char node_name[], |
| 33 | const LazyHandle<skottie::TransformPropertyHandle>& t) override { |
Florin Malita | 3f45e4b | 2020-08-20 14:51:41 -0400 | [diff] [blame] | 34 | const auto key = fMgr->acceptKey(node_name, ".Transform"); |
| 35 | if (!key.empty()) { |
| 36 | fMgr->fTransformMap[key].push_back(t()); |
| 37 | } |
| 38 | } |
| 39 | |
| 40 | void onTextProperty(const char node_name[], |
| 41 | const LazyHandle<skottie::TextPropertyHandle>& t) override { |
| 42 | const auto key = fMgr->acceptKey(node_name, ".Text"); |
| 43 | if (!key.empty()) { |
| 44 | fMgr->fTextMap[key].push_back(t()); |
| 45 | } |
Shachar Langbeheim | be28d2e | 2020-02-11 09:48:50 +0200 | [diff] [blame] | 46 | } |
| 47 | |
| 48 | void onEnterNode(const char node_name[]) override { |
| 49 | fMgr->fCurrentNode = |
| 50 | fMgr->fCurrentNode.empty() ? node_name : fMgr->fCurrentNode + "." + node_name; |
| 51 | } |
| 52 | |
| 53 | void onLeavingNode(const char node_name[]) override { |
| 54 | auto length = strlen(node_name); |
| 55 | fMgr->fCurrentNode = |
| 56 | fMgr->fCurrentNode.length() > length |
| 57 | ? fMgr->fCurrentNode.substr( |
| 58 | 0, fMgr->fCurrentNode.length() - strlen(node_name) - 1) |
| 59 | : ""; |
Florin Malita | 8ac81b7 | 2018-11-28 11:39:39 -0500 | [diff] [blame] | 60 | } |
Florin Malita | 8ac81b7 | 2018-11-28 11:39:39 -0500 | [diff] [blame] | 61 | |
Florin Malita | 91af8d8 | 2018-11-30 16:46:45 -0500 | [diff] [blame] | 62 | private: |
| 63 | CustomPropertyManager* fMgr; |
| 64 | }; |
| 65 | |
| 66 | class CustomPropertyManager::MarkerInterceptor final : public skottie::MarkerObserver { |
| 67 | public: |
| 68 | explicit MarkerInterceptor(CustomPropertyManager* mgr) : fMgr(mgr) {} |
| 69 | |
| 70 | void onMarker(const char name[], float t0, float t1) override { |
Florin Malita | 3f45e4b | 2020-08-20 14:51:41 -0400 | [diff] [blame] | 71 | // collect all markers |
| 72 | fMgr->fMarkers.push_back({ std::string(name), t0, t1 }); |
Florin Malita | 91af8d8 | 2018-11-30 16:46:45 -0500 | [diff] [blame] | 73 | } |
| 74 | |
| 75 | private: |
| 76 | CustomPropertyManager* fMgr; |
| 77 | }; |
| 78 | |
Florin Malita | 3f45e4b | 2020-08-20 14:51:41 -0400 | [diff] [blame] | 79 | CustomPropertyManager::CustomPropertyManager(Mode mode, const char* prefix) |
| 80 | : fMode(mode) |
| 81 | , fPrefix(prefix ? prefix : "$") |
| 82 | , fPropertyInterceptor(sk_make_sp<PropertyInterceptor>(this)) |
Florin Malita | 91af8d8 | 2018-11-30 16:46:45 -0500 | [diff] [blame] | 83 | , fMarkerInterceptor(sk_make_sp<MarkerInterceptor>(this)) {} |
Florin Malita | 8ac81b7 | 2018-11-28 11:39:39 -0500 | [diff] [blame] | 84 | |
| 85 | CustomPropertyManager::~CustomPropertyManager() = default; |
| 86 | |
Florin Malita | 3f45e4b | 2020-08-20 14:51:41 -0400 | [diff] [blame] | 87 | std::string CustomPropertyManager::acceptKey(const char* name, const char* suffix) const { |
| 88 | if (!SkStrStartsWith(name, fPrefix.c_str())) { |
| 89 | return std::string(); |
| 90 | } |
| 91 | |
| 92 | return fMode == Mode::kCollapseProperties |
| 93 | ? std::string(name) |
| 94 | : fCurrentNode + suffix; |
| 95 | } |
| 96 | |
Florin Malita | 91af8d8 | 2018-11-30 16:46:45 -0500 | [diff] [blame] | 97 | sk_sp<skottie::PropertyObserver> CustomPropertyManager::getPropertyObserver() const { |
| 98 | return fPropertyInterceptor; |
| 99 | } |
| 100 | |
| 101 | sk_sp<skottie::MarkerObserver> CustomPropertyManager::getMarkerObserver() const { |
| 102 | return fMarkerInterceptor; |
| 103 | } |
| 104 | |
Florin Malita | 8ac81b7 | 2018-11-28 11:39:39 -0500 | [diff] [blame] | 105 | template <typename T> |
| 106 | std::vector<CustomPropertyManager::PropKey> |
| 107 | CustomPropertyManager::getProps(const PropMap<T>& container) const { |
| 108 | std::vector<PropKey> props; |
| 109 | |
| 110 | for (const auto& prop_list : container) { |
| 111 | SkASSERT(!prop_list.second.empty()); |
| 112 | props.push_back(prop_list.first); |
| 113 | } |
| 114 | |
| 115 | return props; |
| 116 | } |
| 117 | |
| 118 | template <typename V, typename T> |
| 119 | V CustomPropertyManager::get(const PropKey& key, const PropMap<T>& container) const { |
| 120 | auto prop_group = container.find(key); |
| 121 | |
| 122 | return prop_group == container.end() |
| 123 | ? V() |
| 124 | : prop_group->second.front()->get(); |
| 125 | } |
| 126 | |
| 127 | template <typename V, typename T> |
| 128 | bool CustomPropertyManager::set(const PropKey& key, const V& val, const PropMap<T>& container) { |
| 129 | auto prop_group = container.find(key); |
| 130 | |
| 131 | if (prop_group == container.end()) { |
| 132 | return false; |
| 133 | } |
| 134 | |
| 135 | for (auto& handle : prop_group->second) { |
| 136 | handle->set(val); |
| 137 | } |
| 138 | |
| 139 | return true; |
| 140 | } |
| 141 | |
| 142 | std::vector<CustomPropertyManager::PropKey> |
| 143 | CustomPropertyManager::getColorProps() const { |
| 144 | return this->getProps(fColorMap); |
| 145 | } |
| 146 | |
| 147 | skottie::ColorPropertyValue CustomPropertyManager::getColor(const PropKey& key) const { |
| 148 | return this->get<skottie::ColorPropertyValue>(key, fColorMap); |
| 149 | } |
| 150 | |
| 151 | bool CustomPropertyManager::setColor(const PropKey& key, const skottie::ColorPropertyValue& c) { |
| 152 | return this->set(key, c, fColorMap); |
| 153 | } |
| 154 | |
| 155 | std::vector<CustomPropertyManager::PropKey> |
| 156 | CustomPropertyManager::getOpacityProps() const { |
| 157 | return this->getProps(fOpacityMap); |
| 158 | } |
| 159 | |
| 160 | skottie::OpacityPropertyValue CustomPropertyManager::getOpacity(const PropKey& key) const { |
| 161 | return this->get<skottie::OpacityPropertyValue>(key, fOpacityMap); |
| 162 | } |
| 163 | |
| 164 | bool CustomPropertyManager::setOpacity(const PropKey& key, const skottie::OpacityPropertyValue& o) { |
| 165 | return this->set(key, o, fOpacityMap); |
| 166 | } |
| 167 | |
| 168 | std::vector<CustomPropertyManager::PropKey> |
| 169 | CustomPropertyManager::getTransformProps() const { |
| 170 | return this->getProps(fTransformMap); |
| 171 | } |
| 172 | |
Florin Malita | 7c7cd30 | 2020-01-16 18:39:44 -0500 | [diff] [blame] | 173 | skottie::TransformPropertyValue CustomPropertyManager::getTransform(const PropKey& key) const { |
| 174 | return this->get<skottie::TransformPropertyValue>(key, fTransformMap); |
| 175 | } |
| 176 | |
Florin Malita | 8ac81b7 | 2018-11-28 11:39:39 -0500 | [diff] [blame] | 177 | bool CustomPropertyManager::setTransform(const PropKey& key, |
| 178 | const skottie::TransformPropertyValue& t) { |
| 179 | return this->set(key, t, fTransformMap); |
| 180 | } |
| 181 | |
Florin Malita | 7c7cd30 | 2020-01-16 18:39:44 -0500 | [diff] [blame] | 182 | std::vector<CustomPropertyManager::PropKey> |
| 183 | CustomPropertyManager::getTextProps() const { |
| 184 | return this->getProps(fTextMap); |
| 185 | } |
| 186 | |
| 187 | skottie::TextPropertyValue CustomPropertyManager::getText(const PropKey& key) const { |
| 188 | return this->get<skottie::TextPropertyValue>(key, fTextMap); |
| 189 | } |
| 190 | |
| 191 | bool CustomPropertyManager::setText(const PropKey& key, const skottie::TextPropertyValue& o) { |
| 192 | return this->set(key, o, fTextMap); |
| 193 | } |
| 194 | |
Florin Malita | fbddfbb | 2020-05-06 15:55:18 -0400 | [diff] [blame] | 195 | namespace { |
| 196 | |
| 197 | class ExternalAnimationLayer final : public skottie::ExternalLayer { |
| 198 | public: |
| 199 | ExternalAnimationLayer(sk_sp<skottie::Animation> anim, const SkSize& size) |
| 200 | : fAnimation(std::move(anim)) |
| 201 | , fSize(size) {} |
| 202 | |
| 203 | private: |
| 204 | void render(SkCanvas* canvas, double t) override { |
| 205 | fAnimation->seekFrameTime(t); |
| 206 | |
| 207 | const auto dst_rect = SkRect::MakeSize(fSize); |
| 208 | fAnimation->render(canvas, &dst_rect); |
| 209 | } |
| 210 | |
| 211 | const sk_sp<skottie::Animation> fAnimation; |
| 212 | const SkSize fSize; |
| 213 | }; |
| 214 | |
| 215 | } // namespace |
| 216 | |
| 217 | ExternalAnimationPrecompInterceptor::ExternalAnimationPrecompInterceptor( |
| 218 | sk_sp<skresources::ResourceProvider> rprovider, |
| 219 | const char prefixp[]) |
| 220 | : fResourceProvider(std::move(rprovider)) |
| 221 | , fPrefix(prefixp) {} |
| 222 | |
| 223 | ExternalAnimationPrecompInterceptor::~ExternalAnimationPrecompInterceptor() = default; |
| 224 | |
| 225 | sk_sp<skottie::ExternalLayer> ExternalAnimationPrecompInterceptor::onLoadPrecomp( |
| 226 | const char[], const char name[], const SkSize& size) { |
John Stiles | c1c3c6d | 2020-08-15 23:22:53 -0400 | [diff] [blame] | 227 | if (0 != strncmp(name, fPrefix.c_str(), fPrefix.size())) { |
Florin Malita | fbddfbb | 2020-05-06 15:55:18 -0400 | [diff] [blame] | 228 | return nullptr; |
| 229 | } |
| 230 | |
| 231 | auto data = fResourceProvider->load("", name + fPrefix.size()); |
| 232 | if (!data) { |
| 233 | return nullptr; |
| 234 | } |
| 235 | |
| 236 | auto anim = skottie::Animation::Builder() |
| 237 | .setPrecompInterceptor(sk_ref_sp(this)) |
| 238 | .make(static_cast<const char*>(data->data()), data->size()); |
| 239 | |
| 240 | return anim ? sk_make_sp<ExternalAnimationLayer>(std::move(anim), size) |
| 241 | : nullptr; |
| 242 | } |
| 243 | |
Florin Malita | a831655 | 2018-11-09 16:19:44 -0500 | [diff] [blame] | 244 | } // namespace skottie_utils |