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