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