jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 1 | /* |
| 2 | * Copyright 2016 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 | |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 8 | #include "Viewer.h" |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 9 | |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 10 | #include "GMSlide.h" |
| 11 | #include "SKPSlide.h" |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 12 | |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 13 | #include "SkCanvas.h" |
| 14 | #include "SkCommonFlags.h" |
| 15 | #include "SkOSFile.h" |
| 16 | #include "SkRandom.h" |
| 17 | #include "SkStream.h" |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 18 | |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 19 | using namespace sk_app; |
| 20 | |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 21 | Application* Application::Create(int argc, char** argv, void* platformData) { |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 22 | return new Viewer(argc, argv, platformData); |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 23 | } |
| 24 | |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 25 | static void on_paint_handler(SkCanvas* canvas, void* userData) { |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 26 | Viewer* vv = reinterpret_cast<Viewer*>(userData); |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 27 | |
| 28 | return vv->onPaint(canvas); |
| 29 | } |
| 30 | |
jvanverth | 814e38d | 2016-06-06 08:48:47 -0700 | [diff] [blame] | 31 | static bool on_touch_handler(intptr_t owner, Window::InputState state, float x, float y, void* userData) |
liyuqian | d3cdbca | 2016-05-17 12:44:20 -0700 | [diff] [blame] | 32 | { |
| 33 | Viewer* viewer = reinterpret_cast<Viewer*>(userData); |
| 34 | |
| 35 | return viewer->onTouch(owner, state, x, y); |
| 36 | } |
| 37 | |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 38 | static void on_ui_state_changed_handler(const SkString& stateName, const SkString& stateValue, void* userData) { |
| 39 | Viewer* viewer = reinterpret_cast<Viewer*>(userData); |
| 40 | |
| 41 | return viewer->onUIStateChanged(stateName, stateValue); |
| 42 | } |
| 43 | |
| 44 | DEFINE_bool2(fullscreen, f, true, "Run fullscreen."); |
egdaniel | f533f11 | 2016-06-13 11:30:10 -0700 | [diff] [blame^] | 45 | DEFINE_string(key, "", "Space-separated key/value pairs to add to JSON identifying this builder."); |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 46 | DEFINE_string2(match, m, nullptr, |
| 47 | "[~][^]substring[$] [...] of bench name to run.\n" |
| 48 | "Multiple matches may be separated by spaces.\n" |
| 49 | "~ causes a matching bench to always be skipped\n" |
| 50 | "^ requires the start of the bench to match\n" |
| 51 | "$ requires the end of the bench to match\n" |
| 52 | "^ and $ requires an exact match\n" |
| 53 | "If a bench does not match any list entry,\n" |
| 54 | "it is skipped unless some list entry starts with ~"); |
| 55 | DEFINE_string(skps, "skps", "Directory to read skps from."); |
liyuqian | 71491dc | 2016-06-09 12:02:34 -0700 | [diff] [blame] | 56 | #ifdef SK_BUILD_FOR_ANDROID |
| 57 | DEFINE_bool(vulkan, false, "Run with Vulkan."); |
| 58 | #else |
jvanverth | 85f758c | 2016-05-27 06:47:08 -0700 | [diff] [blame] | 59 | DEFINE_bool(vulkan, true, "Run with Vulkan."); |
liyuqian | 71491dc | 2016-06-09 12:02:34 -0700 | [diff] [blame] | 60 | #endif |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 61 | |
jvanverth | af236b5 | 2016-05-20 06:01:06 -0700 | [diff] [blame] | 62 | const char *kBackendTypeStrings[sk_app::Window::kBackendTypeCount] = { |
| 63 | " [OpenGL]", |
liyuqian | d94ad58 | 2016-06-07 14:22:37 -0700 | [diff] [blame] | 64 | " [Vulkan]", |
| 65 | " [Raster]" |
jvanverth | af236b5 | 2016-05-20 06:01:06 -0700 | [diff] [blame] | 66 | }; |
| 67 | |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 68 | const char* kName = "name"; |
| 69 | const char* kValue = "value"; |
| 70 | const char* kOptions = "options"; |
| 71 | const char* kSlideStateName = "Slide"; |
| 72 | const char* kBackendStateName = "Backend"; |
liyuqian | b73c24b | 2016-06-03 08:47:23 -0700 | [diff] [blame] | 73 | const char* kSoftkeyStateName = "Softkey"; |
| 74 | const char* kSoftkeyHint = "Please select a softkey"; |
liyuqian | 1f508fd | 2016-06-07 06:57:40 -0700 | [diff] [blame] | 75 | const char* kFpsStateName = "FPS"; |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 76 | |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 77 | Viewer::Viewer(int argc, char** argv, void* platformData) |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 78 | : fCurrentMeasurement(0) |
| 79 | , fDisplayStats(false) |
jvanverth | af236b5 | 2016-05-20 06:01:06 -0700 | [diff] [blame] | 80 | , fBackendType(sk_app::Window::kVulkan_BackendType) |
egdaniel | 2a0bb0a | 2016-04-11 08:30:40 -0700 | [diff] [blame] | 81 | , fZoomCenterX(0.0f) |
| 82 | , fZoomCenterY(0.0f) |
| 83 | , fZoomLevel(0.0f) |
| 84 | , fZoomScale(SK_Scalar1) |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 85 | { |
jvanverth | 3d6ed3a | 2016-04-07 11:09:51 -0700 | [diff] [blame] | 86 | memset(fMeasurements, 0, sizeof(fMeasurements)); |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 87 | |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 88 | SkDebugf("Command line arguments: "); |
| 89 | for (int i = 1; i < argc; ++i) { |
| 90 | SkDebugf("%s ", argv[i]); |
| 91 | } |
| 92 | SkDebugf("\n"); |
| 93 | |
| 94 | SkCommandLineFlags::Parse(argc, argv); |
| 95 | |
jvanverth | 85f758c | 2016-05-27 06:47:08 -0700 | [diff] [blame] | 96 | fBackendType = FLAGS_vulkan ? sk_app::Window::kVulkan_BackendType |
| 97 | : sk_app::Window::kNativeGL_BackendType; |
| 98 | |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 99 | fWindow = Window::CreateNativeWindow(platformData); |
jvanverth | af236b5 | 2016-05-20 06:01:06 -0700 | [diff] [blame] | 100 | fWindow->attach(fBackendType, DisplayParams()); |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 101 | |
| 102 | // register callbacks |
brianosman | 622c8d5 | 2016-05-10 06:50:49 -0700 | [diff] [blame] | 103 | fCommands.attach(fWindow); |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 104 | fWindow->registerPaintFunc(on_paint_handler, this); |
liyuqian | d3cdbca | 2016-05-17 12:44:20 -0700 | [diff] [blame] | 105 | fWindow->registerTouchFunc(on_touch_handler, this); |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 106 | fWindow->registerUIStateChangedFunc(on_ui_state_changed_handler, this); |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 107 | |
brianosman | 622c8d5 | 2016-05-10 06:50:49 -0700 | [diff] [blame] | 108 | // add key-bindings |
| 109 | fCommands.addCommand('s', "Overlays", "Toggle stats display", [this]() { |
| 110 | this->fDisplayStats = !this->fDisplayStats; |
| 111 | fWindow->inval(); |
| 112 | }); |
| 113 | fCommands.addCommand('c', "Modes", "Toggle sRGB color mode", [this]() { |
| 114 | DisplayParams params = fWindow->getDisplayParams(); |
| 115 | params.fProfileType = (kLinear_SkColorProfileType == params.fProfileType) |
| 116 | ? kSRGB_SkColorProfileType : kLinear_SkColorProfileType; |
| 117 | fWindow->setDisplayParams(params); |
| 118 | this->updateTitle(); |
| 119 | fWindow->inval(); |
| 120 | }); |
| 121 | fCommands.addCommand(Window::Key::kRight, "Right", "Navigation", "Next slide", [this]() { |
| 122 | int previousSlide = fCurrentSlide; |
| 123 | fCurrentSlide++; |
| 124 | if (fCurrentSlide >= fSlides.count()) { |
| 125 | fCurrentSlide = 0; |
| 126 | } |
| 127 | this->setupCurrentSlide(previousSlide); |
| 128 | }); |
| 129 | fCommands.addCommand(Window::Key::kLeft, "Left", "Navigation", "Previous slide", [this]() { |
| 130 | int previousSlide = fCurrentSlide; |
| 131 | fCurrentSlide--; |
| 132 | if (fCurrentSlide < 0) { |
| 133 | fCurrentSlide = fSlides.count() - 1; |
| 134 | } |
| 135 | this->setupCurrentSlide(previousSlide); |
| 136 | }); |
| 137 | fCommands.addCommand(Window::Key::kUp, "Up", "Transform", "Zoom in", [this]() { |
| 138 | this->changeZoomLevel(1.f / 32.f); |
| 139 | fWindow->inval(); |
| 140 | }); |
| 141 | fCommands.addCommand(Window::Key::kDown, "Down", "Transform", "Zoom out", [this]() { |
| 142 | this->changeZoomLevel(-1.f / 32.f); |
| 143 | fWindow->inval(); |
| 144 | }); |
jvanverth | 85f758c | 2016-05-27 06:47:08 -0700 | [diff] [blame] | 145 | #if 0 // this doesn't seem to work on any platform right now |
jvanverth | af236b5 | 2016-05-20 06:01:06 -0700 | [diff] [blame] | 146 | #ifndef SK_BUILD_FOR_ANDROID |
| 147 | fCommands.addCommand('d', "Modes", "Change rendering backend", [this]() { |
| 148 | fWindow->detach(); |
| 149 | |
| 150 | if (sk_app::Window::kVulkan_BackendType == fBackendType) { |
| 151 | fBackendType = sk_app::Window::kNativeGL_BackendType; |
| 152 | } |
jvanverth | 85f758c | 2016-05-27 06:47:08 -0700 | [diff] [blame] | 153 | // TODO: get Vulkan -> OpenGL working on Windows without swapchain creation failure |
jvanverth | af236b5 | 2016-05-20 06:01:06 -0700 | [diff] [blame] | 154 | //else if (sk_app::Window::kNativeGL_BackendType == fBackendType) { |
| 155 | // fBackendType = sk_app::Window::kVulkan_BackendType; |
| 156 | //} |
| 157 | |
| 158 | fWindow->attach(fBackendType, DisplayParams()); |
| 159 | this->updateTitle(); |
jvanverth | 85f758c | 2016-05-27 06:47:08 -0700 | [diff] [blame] | 160 | fWindow->inval(); |
jvanverth | af236b5 | 2016-05-20 06:01:06 -0700 | [diff] [blame] | 161 | }); |
| 162 | #endif |
jvanverth | 85f758c | 2016-05-27 06:47:08 -0700 | [diff] [blame] | 163 | #endif |
brianosman | 622c8d5 | 2016-05-10 06:50:49 -0700 | [diff] [blame] | 164 | |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 165 | // set up slides |
| 166 | this->initSlides(); |
| 167 | |
djsollen | 12d62a7 | 2016-04-21 07:59:44 -0700 | [diff] [blame] | 168 | fAnimTimer.run(); |
| 169 | |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 170 | // set up first frame |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 171 | fCurrentSlide = 0; |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 172 | setupCurrentSlide(-1); |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 173 | |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 174 | fWindow->show(); |
| 175 | } |
| 176 | |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 177 | void Viewer::initSlides() { |
liyuqian | 1f508fd | 2016-06-07 06:57:40 -0700 | [diff] [blame] | 178 | fAllSlideNames = Json::Value(Json::arrayValue); |
| 179 | |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 180 | const skiagm::GMRegistry* gms(skiagm::GMRegistry::Head()); |
| 181 | while (gms) { |
| 182 | SkAutoTDelete<skiagm::GM> gm(gms->factory()(nullptr)); |
| 183 | |
| 184 | if (!SkCommandLineFlags::ShouldSkip(FLAGS_match, gm->getName())) { |
| 185 | sk_sp<Slide> slide(new GMSlide(gm.release())); |
| 186 | fSlides.push_back(slide); |
| 187 | } |
| 188 | |
| 189 | gms = gms->next(); |
| 190 | } |
| 191 | |
| 192 | // reverse array |
| 193 | for (int i = 0; i < fSlides.count()/2; ++i) { |
| 194 | sk_sp<Slide> temp = fSlides[i]; |
| 195 | fSlides[i] = fSlides[fSlides.count() - i - 1]; |
| 196 | fSlides[fSlides.count() - i - 1] = temp; |
| 197 | } |
| 198 | |
| 199 | // SKPs |
| 200 | for (int i = 0; i < FLAGS_skps.count(); i++) { |
| 201 | if (SkStrEndsWith(FLAGS_skps[i], ".skp")) { |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 202 | if (SkCommandLineFlags::ShouldSkip(FLAGS_match, FLAGS_skps[i])) { |
| 203 | continue; |
| 204 | } |
| 205 | |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 206 | SkString path(FLAGS_skps[i]); |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 207 | sk_sp<SKPSlide> slide(new SKPSlide(SkOSPath::Basename(path.c_str()), path)); |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 208 | if (slide) { |
| 209 | fSlides.push_back(slide); |
| 210 | } |
| 211 | } else { |
| 212 | SkOSFile::Iter it(FLAGS_skps[i], ".skp"); |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 213 | SkString skpName; |
| 214 | while (it.next(&skpName)) { |
| 215 | if (SkCommandLineFlags::ShouldSkip(FLAGS_match, skpName.c_str())) { |
| 216 | continue; |
| 217 | } |
| 218 | |
| 219 | SkString path = SkOSPath::Join(FLAGS_skps[i], skpName.c_str()); |
| 220 | sk_sp<SKPSlide> slide(new SKPSlide(skpName, path)); |
jvanverth | 2bb3b6d | 2016-04-08 07:24:09 -0700 | [diff] [blame] | 221 | if (slide) { |
| 222 | fSlides.push_back(slide); |
| 223 | } |
| 224 | } |
| 225 | } |
| 226 | } |
| 227 | } |
| 228 | |
| 229 | |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 230 | Viewer::~Viewer() { |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 231 | fWindow->detach(); |
| 232 | delete fWindow; |
| 233 | } |
| 234 | |
brianosman | 05de216 | 2016-05-06 13:28:57 -0700 | [diff] [blame] | 235 | void Viewer::updateTitle() { |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 236 | SkString title("Viewer: "); |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 237 | title.append(fSlides[fCurrentSlide]->getName()); |
brianosman | 05de216 | 2016-05-06 13:28:57 -0700 | [diff] [blame] | 238 | if (kSRGB_SkColorProfileType == fWindow->getDisplayParams().fProfileType) { |
| 239 | title.append(" sRGB"); |
| 240 | } |
jvanverth | af236b5 | 2016-05-20 06:01:06 -0700 | [diff] [blame] | 241 | title.append(kBackendTypeStrings[fBackendType]); |
brianosman | 05de216 | 2016-05-06 13:28:57 -0700 | [diff] [blame] | 242 | fWindow->setTitle(title.c_str()); |
| 243 | } |
| 244 | |
| 245 | void Viewer::setupCurrentSlide(int previousSlide) { |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 246 | if (fCurrentSlide == previousSlide) { |
| 247 | return; // no change; do nothing |
| 248 | } |
| 249 | |
liyuqian | e46e4f0 | 2016-05-20 07:32:19 -0700 | [diff] [blame] | 250 | fGesture.reset(); |
| 251 | fDefaultMatrix.reset(); |
| 252 | fDefaultMatrixInv.reset(); |
| 253 | |
| 254 | if (fWindow->supportsContentRect() && fWindow->scaleContentToFit()) { |
| 255 | const SkRect contentRect = fWindow->getContentRect(); |
| 256 | const SkISize slideSize = fSlides[fCurrentSlide]->getDimensions(); |
| 257 | const SkRect slideBounds = SkRect::MakeIWH(slideSize.width(), slideSize.height()); |
| 258 | if (contentRect.width() > 0 && contentRect.height() > 0) { |
| 259 | fDefaultMatrix.setRectToRect(slideBounds, contentRect, SkMatrix::kStart_ScaleToFit); |
liyuqian | beb1c67 | 2016-05-20 11:41:01 -0700 | [diff] [blame] | 260 | SkAssertResult(fDefaultMatrix.invert(&fDefaultMatrixInv)); |
liyuqian | e46e4f0 | 2016-05-20 07:32:19 -0700 | [diff] [blame] | 261 | } |
| 262 | } |
| 263 | |
| 264 | if (fWindow->supportsContentRect()) { |
| 265 | const SkISize slideSize = fSlides[fCurrentSlide]->getDimensions(); |
| 266 | SkRect windowRect = fWindow->getContentRect(); |
| 267 | fDefaultMatrixInv.mapRect(&windowRect); |
jvanverth | 1e305ba | 2016-06-01 09:39:15 -0700 | [diff] [blame] | 268 | fGesture.setTransLimit(SkRect::MakeWH(SkIntToScalar(slideSize.width()), |
| 269 | SkIntToScalar(slideSize.height())), |
| 270 | windowRect); |
liyuqian | e46e4f0 | 2016-05-20 07:32:19 -0700 | [diff] [blame] | 271 | } |
| 272 | |
brianosman | 05de216 | 2016-05-06 13:28:57 -0700 | [diff] [blame] | 273 | this->updateTitle(); |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 274 | this->updateUIState(); |
egdaniel | f533f11 | 2016-06-13 11:30:10 -0700 | [diff] [blame^] | 275 | fSlides[fCurrentSlide]->load(); |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 276 | if (previousSlide >= 0) { |
| 277 | fSlides[previousSlide]->unload(); |
| 278 | } |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 279 | fWindow->inval(); |
| 280 | } |
| 281 | |
| 282 | #define MAX_ZOOM_LEVEL 8 |
| 283 | #define MIN_ZOOM_LEVEL -8 |
| 284 | |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 285 | void Viewer::changeZoomLevel(float delta) { |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 286 | fZoomLevel += delta; |
| 287 | if (fZoomLevel > 0) { |
| 288 | fZoomLevel = SkMinScalar(fZoomLevel, MAX_ZOOM_LEVEL); |
| 289 | fZoomScale = fZoomLevel + SK_Scalar1; |
| 290 | } else if (fZoomLevel < 0) { |
| 291 | fZoomLevel = SkMaxScalar(fZoomLevel, MIN_ZOOM_LEVEL); |
| 292 | fZoomScale = SK_Scalar1 / (SK_Scalar1 - fZoomLevel); |
| 293 | } else { |
| 294 | fZoomScale = SK_Scalar1; |
| 295 | } |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 296 | } |
| 297 | |
liyuqian | d3cdbca | 2016-05-17 12:44:20 -0700 | [diff] [blame] | 298 | SkMatrix Viewer::computeMatrix() { |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 299 | SkMatrix m; |
| 300 | m.reset(); |
| 301 | |
| 302 | if (fZoomLevel) { |
| 303 | SkPoint center; |
| 304 | //m = this->getLocalMatrix();//.invert(&m); |
| 305 | m.mapXY(fZoomCenterX, fZoomCenterY, ¢er); |
| 306 | SkScalar cx = center.fX; |
| 307 | SkScalar cy = center.fY; |
| 308 | |
| 309 | m.setTranslate(-cx, -cy); |
| 310 | m.postScale(fZoomScale, fZoomScale); |
| 311 | m.postTranslate(cx, cy); |
| 312 | } |
| 313 | |
liyuqian | d3cdbca | 2016-05-17 12:44:20 -0700 | [diff] [blame] | 314 | m.preConcat(fGesture.localM()); |
| 315 | m.preConcat(fGesture.globalM()); |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 316 | |
liyuqian | d3cdbca | 2016-05-17 12:44:20 -0700 | [diff] [blame] | 317 | return m; |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 318 | } |
| 319 | |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 320 | void Viewer::onPaint(SkCanvas* canvas) { |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 321 | int count = canvas->save(); |
djsollen | 12d62a7 | 2016-04-21 07:59:44 -0700 | [diff] [blame] | 322 | |
| 323 | if (fWindow->supportsContentRect()) { |
| 324 | SkRect contentRect = fWindow->getContentRect(); |
| 325 | canvas->clipRect(contentRect); |
| 326 | canvas->translate(contentRect.fLeft, contentRect.fTop); |
| 327 | } |
| 328 | |
| 329 | canvas->clear(SK_ColorWHITE); |
liyuqian | e46e4f0 | 2016-05-20 07:32:19 -0700 | [diff] [blame] | 330 | canvas->concat(fDefaultMatrix); |
liyuqian | d3cdbca | 2016-05-17 12:44:20 -0700 | [diff] [blame] | 331 | canvas->concat(computeMatrix()); |
jvanverth | 3d6ed3a | 2016-04-07 11:09:51 -0700 | [diff] [blame] | 332 | |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 333 | fSlides[fCurrentSlide]->draw(canvas); |
| 334 | canvas->restoreToCount(count); |
| 335 | |
| 336 | if (fDisplayStats) { |
| 337 | drawStats(canvas); |
| 338 | } |
brianosman | 622c8d5 | 2016-05-10 06:50:49 -0700 | [diff] [blame] | 339 | fCommands.drawHelp(canvas); |
jvanverth | 3d6ed3a | 2016-04-07 11:09:51 -0700 | [diff] [blame] | 340 | } |
| 341 | |
jvanverth | 814e38d | 2016-06-06 08:48:47 -0700 | [diff] [blame] | 342 | bool Viewer::onTouch(intptr_t owner, Window::InputState state, float x, float y) { |
liyuqian | d3cdbca | 2016-05-17 12:44:20 -0700 | [diff] [blame] | 343 | void* castedOwner = reinterpret_cast<void*>(owner); |
liyuqian | e46e4f0 | 2016-05-20 07:32:19 -0700 | [diff] [blame] | 344 | SkPoint touchPoint = fDefaultMatrixInv.mapXY(x, y); |
liyuqian | d3cdbca | 2016-05-17 12:44:20 -0700 | [diff] [blame] | 345 | switch (state) { |
| 346 | case Window::kUp_InputState: { |
| 347 | fGesture.touchEnd(castedOwner); |
| 348 | break; |
| 349 | } |
| 350 | case Window::kDown_InputState: { |
liyuqian | e46e4f0 | 2016-05-20 07:32:19 -0700 | [diff] [blame] | 351 | fGesture.touchBegin(castedOwner, touchPoint.fX, touchPoint.fY); |
liyuqian | d3cdbca | 2016-05-17 12:44:20 -0700 | [diff] [blame] | 352 | break; |
| 353 | } |
| 354 | case Window::kMove_InputState: { |
liyuqian | e46e4f0 | 2016-05-20 07:32:19 -0700 | [diff] [blame] | 355 | fGesture.touchMoved(castedOwner, touchPoint.fX, touchPoint.fY); |
liyuqian | d3cdbca | 2016-05-17 12:44:20 -0700 | [diff] [blame] | 356 | break; |
| 357 | } |
| 358 | } |
| 359 | fWindow->inval(); |
| 360 | return true; |
| 361 | } |
| 362 | |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 363 | void Viewer::drawStats(SkCanvas* canvas) { |
jvanverth | 3d6ed3a | 2016-04-07 11:09:51 -0700 | [diff] [blame] | 364 | static const float kPixelPerMS = 2.0f; |
| 365 | static const int kDisplayWidth = 130; |
| 366 | static const int kDisplayHeight = 100; |
| 367 | static const int kDisplayPadding = 10; |
| 368 | static const int kGraphPadding = 3; |
| 369 | static const SkScalar kBaseMS = 1000.f / 60.f; // ms/frame to hit 60 fps |
| 370 | |
| 371 | SkISize canvasSize = canvas->getDeviceSize(); |
| 372 | SkRect rect = SkRect::MakeXYWH(SkIntToScalar(canvasSize.fWidth-kDisplayWidth-kDisplayPadding), |
| 373 | SkIntToScalar(kDisplayPadding), |
| 374 | SkIntToScalar(kDisplayWidth), SkIntToScalar(kDisplayHeight)); |
| 375 | SkPaint paint; |
| 376 | canvas->save(); |
| 377 | |
djsollen | 12d62a7 | 2016-04-21 07:59:44 -0700 | [diff] [blame] | 378 | if (fWindow->supportsContentRect()) { |
| 379 | SkRect contentRect = fWindow->getContentRect(); |
| 380 | canvas->clipRect(contentRect); |
| 381 | canvas->translate(contentRect.fLeft, contentRect.fTop); |
| 382 | } |
| 383 | |
jvanverth | 3d6ed3a | 2016-04-07 11:09:51 -0700 | [diff] [blame] | 384 | canvas->clipRect(rect); |
| 385 | paint.setColor(SK_ColorBLACK); |
| 386 | canvas->drawRect(rect, paint); |
| 387 | // draw the 16ms line |
| 388 | paint.setColor(SK_ColorLTGRAY); |
| 389 | canvas->drawLine(rect.fLeft, rect.fBottom - kBaseMS*kPixelPerMS, |
| 390 | rect.fRight, rect.fBottom - kBaseMS*kPixelPerMS, paint); |
| 391 | paint.setColor(SK_ColorRED); |
| 392 | paint.setStyle(SkPaint::kStroke_Style); |
| 393 | canvas->drawRect(rect, paint); |
| 394 | |
| 395 | int x = SkScalarTruncToInt(rect.fLeft) + kGraphPadding; |
| 396 | const int xStep = 2; |
| 397 | const int startY = SkScalarTruncToInt(rect.fBottom); |
| 398 | int i = fCurrentMeasurement; |
| 399 | do { |
| 400 | int endY = startY - (int)(fMeasurements[i] * kPixelPerMS + 0.5); // round to nearest value |
| 401 | canvas->drawLine(SkIntToScalar(x), SkIntToScalar(startY), |
| 402 | SkIntToScalar(x), SkIntToScalar(endY), paint); |
| 403 | i++; |
| 404 | i &= (kMeasurementCount - 1); // fast mod |
| 405 | x += xStep; |
| 406 | } while (i != fCurrentMeasurement); |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 407 | |
| 408 | canvas->restore(); |
| 409 | } |
| 410 | |
jvanverth | 3452426 | 2016-05-04 13:49:13 -0700 | [diff] [blame] | 411 | void Viewer::onIdle(double ms) { |
jvanverth | 3d6ed3a | 2016-04-07 11:09:51 -0700 | [diff] [blame] | 412 | // Record measurements |
| 413 | fMeasurements[fCurrentMeasurement++] = ms; |
| 414 | fCurrentMeasurement &= (kMeasurementCount - 1); // fast mod |
| 415 | SkASSERT(fCurrentMeasurement < kMeasurementCount); |
| 416 | |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 417 | fAnimTimer.updateTime(); |
jvanverth | 9d5e47f | 2016-04-26 08:01:33 -0700 | [diff] [blame] | 418 | if (fSlides[fCurrentSlide]->animate(fAnimTimer) || fDisplayStats) { |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 419 | fWindow->inval(); |
liyuqian | 1f508fd | 2016-06-07 06:57:40 -0700 | [diff] [blame] | 420 | updateUIState(); // Update the FPS |
jvanverth | c265a92 | 2016-04-08 12:51:45 -0700 | [diff] [blame] | 421 | } |
jvanverth | 9f37246 | 2016-04-06 06:08:59 -0700 | [diff] [blame] | 422 | } |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 423 | |
| 424 | void Viewer::updateUIState() { |
liyuqian | b73c24b | 2016-06-03 08:47:23 -0700 | [diff] [blame] | 425 | // Slide state |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 426 | Json::Value slideState(Json::objectValue); |
| 427 | slideState[kName] = kSlideStateName; |
| 428 | slideState[kValue] = fSlides[fCurrentSlide]->getName().c_str(); |
liyuqian | 1f508fd | 2016-06-07 06:57:40 -0700 | [diff] [blame] | 429 | if (fAllSlideNames.size() == 0) { |
| 430 | for(auto slide : fSlides) { |
| 431 | fAllSlideNames.append(Json::Value(slide->getName().c_str())); |
| 432 | } |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 433 | } |
liyuqian | 1f508fd | 2016-06-07 06:57:40 -0700 | [diff] [blame] | 434 | slideState[kOptions] = fAllSlideNames; |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 435 | |
liyuqian | b73c24b | 2016-06-03 08:47:23 -0700 | [diff] [blame] | 436 | // Backend state |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 437 | Json::Value backendState(Json::objectValue); |
| 438 | backendState[kName] = kBackendStateName; |
liyuqian | 6cb7025 | 2016-06-02 12:16:25 -0700 | [diff] [blame] | 439 | backendState[kValue] = kBackendTypeStrings[fBackendType]; |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 440 | backendState[kOptions] = Json::Value(Json::arrayValue); |
liyuqian | b73c24b | 2016-06-03 08:47:23 -0700 | [diff] [blame] | 441 | for (auto str : kBackendTypeStrings) { |
liyuqian | 6cb7025 | 2016-06-02 12:16:25 -0700 | [diff] [blame] | 442 | backendState[kOptions].append(Json::Value(str)); |
| 443 | } |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 444 | |
liyuqian | b73c24b | 2016-06-03 08:47:23 -0700 | [diff] [blame] | 445 | // Softkey state |
| 446 | Json::Value softkeyState(Json::objectValue); |
| 447 | softkeyState[kName] = kSoftkeyStateName; |
| 448 | softkeyState[kValue] = kSoftkeyHint; |
| 449 | softkeyState[kOptions] = Json::Value(Json::arrayValue); |
| 450 | softkeyState[kOptions].append(kSoftkeyHint); |
| 451 | for (const auto& softkey : fCommands.getCommandsAsSoftkeys()) { |
| 452 | softkeyState[kOptions].append(Json::Value(softkey.c_str())); |
| 453 | } |
| 454 | |
liyuqian | 1f508fd | 2016-06-07 06:57:40 -0700 | [diff] [blame] | 455 | // FPS state |
| 456 | Json::Value fpsState(Json::objectValue); |
| 457 | fpsState[kName] = kFpsStateName; |
| 458 | double measurement = fMeasurements[ |
| 459 | (fCurrentMeasurement + (kMeasurementCount-1)) % kMeasurementCount |
| 460 | ]; |
| 461 | fpsState[kValue] = SkStringPrintf("%8.3lf ms", measurement).c_str(); |
| 462 | fpsState[kOptions] = Json::Value(Json::arrayValue); |
| 463 | |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 464 | Json::Value state(Json::arrayValue); |
| 465 | state.append(slideState); |
| 466 | state.append(backendState); |
liyuqian | b73c24b | 2016-06-03 08:47:23 -0700 | [diff] [blame] | 467 | state.append(softkeyState); |
liyuqian | 1f508fd | 2016-06-07 06:57:40 -0700 | [diff] [blame] | 468 | state.append(fpsState); |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 469 | |
| 470 | fWindow->setUIState(state); |
| 471 | } |
| 472 | |
| 473 | void Viewer::onUIStateChanged(const SkString& stateName, const SkString& stateValue) { |
liyuqian | 6cb7025 | 2016-06-02 12:16:25 -0700 | [diff] [blame] | 474 | // For those who will add more features to handle the state change in this function: |
| 475 | // After the change, please call updateUIState no notify the frontend (e.g., Android app). |
| 476 | // For example, after slide change, updateUIState is called inside setupCurrentSlide; |
| 477 | // after backend change, updateUIState is called in this function. |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 478 | if (stateName.equals(kSlideStateName)) { |
| 479 | int previousSlide = fCurrentSlide; |
| 480 | fCurrentSlide = 0; |
| 481 | for(auto slide : fSlides) { |
| 482 | if (slide->getName().equals(stateValue)) { |
| 483 | setupCurrentSlide(previousSlide); |
| 484 | break; |
| 485 | } |
| 486 | fCurrentSlide++; |
| 487 | } |
| 488 | if (fCurrentSlide >= fSlides.count()) { |
| 489 | fCurrentSlide = previousSlide; |
| 490 | SkDebugf("Slide not found: %s", stateValue.c_str()); |
| 491 | } |
liyuqian | 6cb7025 | 2016-06-02 12:16:25 -0700 | [diff] [blame] | 492 | } else if (stateName.equals(kBackendStateName)) { |
| 493 | for (int i = 0; i < sk_app::Window::kBackendTypeCount; i++) { |
| 494 | if (stateValue.equals(kBackendTypeStrings[i])) { |
| 495 | if (fBackendType != i) { |
| 496 | fBackendType = (sk_app::Window::BackendType)i; |
| 497 | fWindow->detach(); |
| 498 | fWindow->attach(fBackendType, DisplayParams()); |
| 499 | fWindow->inval(); |
| 500 | updateTitle(); |
| 501 | updateUIState(); |
| 502 | } |
| 503 | break; |
| 504 | } |
| 505 | } |
liyuqian | b73c24b | 2016-06-03 08:47:23 -0700 | [diff] [blame] | 506 | } else if (stateName.equals(kSoftkeyStateName)) { |
| 507 | if (!stateValue.equals(kSoftkeyHint)) { |
| 508 | fCommands.onSoftkey(stateValue); |
| 509 | updateUIState(); // This is still needed to reset the value to kSoftkeyHint |
| 510 | } |
liyuqian | e5a6cd9 | 2016-05-27 08:52:52 -0700 | [diff] [blame] | 511 | } else { |
| 512 | SkDebugf("Unknown stateName: %s", stateName.c_str()); |
| 513 | } |
| 514 | } |