blob: 60ef7e97173230aec457729439179a2b969842d1 [file] [log] [blame]
chudy@google.com902ebe52012-06-29 14:21:22 +00001/*
2 * Copyright 2012 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
chudy@google.com902ebe52012-06-29 14:21:22 +00008#include "SkDebuggerGUI.h"
chudy@google.combbad34d2012-08-13 14:26:36 +00009#include "SkGraphics.h"
scroggo@google.comb4467e62012-11-06 23:10:09 +000010#include "SkImageDecoder.h"
chudy@google.com902ebe52012-06-29 14:21:22 +000011#include <QListWidgetItem>
robertphillips@google.com2bde91d2012-11-15 14:57:57 +000012#include "PictureRenderer.h"
robertphillips@google.com2bde91d2012-11-15 14:57:57 +000013#include "SkPictureRecord.h"
14#include "SkPicturePlayback.h"
robertphillips@google.come174a8b2012-11-27 16:04:42 +000015
16#if defined(SK_BUILD_FOR_WIN32)
17 #include "BenchSysTimer_windows.h"
18#elif defined(SK_BUILD_FOR_MAC)
19 #include "BenchSysTimer_mach.h"
20#elif defined(SK_BUILD_FOR_UNIX) || defined(SK_BUILD_FOR_ANDROID)
21 #include "BenchSysTimer_posix.h"
22#else
23 #include "BenchSysTimer_c.h"
24#endif
25
chudy@google.com902ebe52012-06-29 14:21:22 +000026
27SkDebuggerGUI::SkDebuggerGUI(QWidget *parent) :
chudy@google.comc432f002012-07-10 13:19:25 +000028 QMainWindow(parent)
chudy@google.com2d537a12012-07-31 12:49:52 +000029 , fCentralWidget(this)
30 , fStatusBar(this)
31 , fToolBar(this)
chudy@google.comc432f002012-07-10 13:19:25 +000032 , fActionOpen(this)
33 , fActionBreakpoint(this)
robertphillips@google.comd26c7062012-11-12 20:42:12 +000034 , fActionProfile(this)
chudy@google.comc432f002012-07-10 13:19:25 +000035 , fActionCancel(this)
chudy@google.com7e4cfbf2012-07-17 15:40:51 +000036 , fActionClearBreakpoints(this)
chudy@google.come504de02012-07-16 18:35:23 +000037 , fActionClearDeletes(this)
chudy@google.comc432f002012-07-10 13:19:25 +000038 , fActionClose(this)
chudy@google.come504de02012-07-16 18:35:23 +000039 , fActionCreateBreakpoint(this)
chudy@google.comc432f002012-07-10 13:19:25 +000040 , fActionDelete(this)
41 , fActionDirectory(this)
42 , fActionGoToLine(this)
43 , fActionInspector(this)
44 , fActionPlay(this)
chudy@google.come504de02012-07-16 18:35:23 +000045 , fActionPause(this)
chudy@google.comc432f002012-07-10 13:19:25 +000046 , fActionRewind(this)
chudy@google.com0ab03392012-07-28 20:16:11 +000047 , fActionSave(this)
48 , fActionSaveAs(this)
chudy@google.com7e4cfbf2012-07-17 15:40:51 +000049 , fActionShowDeletes(this)
chudy@google.comc432f002012-07-10 13:19:25 +000050 , fActionStepBack(this)
51 , fActionStepForward(this)
chudy@google.coma1226312012-07-26 20:26:44 +000052 , fActionZoomIn(this)
53 , fActionZoomOut(this)
54 , fMapper(this)
chudy@google.comc432f002012-07-10 13:19:25 +000055 , fListWidget(&fCentralWidget)
56 , fDirectoryWidget(&fCentralWidget)
chudy@google.com607357f2012-08-07 16:12:23 +000057 , fCanvasWidget(this, &fDebugger)
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +000058 , fImageWidget(&fDebugger)
chudy@google.comc432f002012-07-10 13:19:25 +000059 , fMenuBar(this)
60 , fMenuFile(this)
61 , fMenuNavigate(this)
62 , fMenuView(this)
chudy@google.com7e4cfbf2012-07-17 15:40:51 +000063 , fBreakpointsActivated(false)
64 , fDeletesActivated(false)
65 , fPause(false)
chudy@google.comd3058f52012-07-19 13:41:27 +000066 , fLoading(false)
chudy@google.comc432f002012-07-10 13:19:25 +000067{
chudy@google.com902ebe52012-06-29 14:21:22 +000068 setupUi(this);
robertphillips@google.comdd4b7452013-01-22 19:38:46 +000069 fListWidget.setSelectionMode(QAbstractItemView::ExtendedSelection);
chudy@google.comea5488b2012-07-26 19:38:22 +000070 connect(&fListWidget, SIGNAL(currentItemChanged(QListWidgetItem*, QListWidgetItem*)), this, SLOT(registerListClick(QListWidgetItem *)));
chudy@google.comc432f002012-07-10 13:19:25 +000071 connect(&fActionOpen, SIGNAL(triggered()), this, SLOT(openFile()));
chudy@google.comea5488b2012-07-26 19:38:22 +000072 connect(&fActionDirectory, SIGNAL(triggered()), this, SLOT(toggleDirectory()));
73 connect(&fDirectoryWidget, SIGNAL(currentItemChanged(QListWidgetItem*, QListWidgetItem*)), this, SLOT(loadFile(QListWidgetItem *)));
chudy@google.comc432f002012-07-10 13:19:25 +000074 connect(&fActionDelete, SIGNAL(triggered()), this, SLOT(actionDelete()));
chudy@google.comea5488b2012-07-26 19:38:22 +000075 connect(&fListWidget, SIGNAL(itemDoubleClicked(QListWidgetItem*)), this, SLOT(toggleBreakpoint()));
chudy@google.comc432f002012-07-10 13:19:25 +000076 connect(&fActionRewind, SIGNAL(triggered()), this, SLOT(actionRewind()));
77 connect(&fActionPlay, SIGNAL(triggered()), this, SLOT(actionPlay()));
78 connect(&fActionStepBack, SIGNAL(triggered()), this, SLOT(actionStepBack()));
chudy@google.comea5488b2012-07-26 19:38:22 +000079 connect(&fActionStepForward, SIGNAL(triggered()), this, SLOT(actionStepForward()));
80 connect(&fActionBreakpoint, SIGNAL(triggered()), this, SLOT(actionBreakpoints()));
81 connect(&fActionInspector, SIGNAL(triggered()), this, SLOT(actionInspector()));
82 connect(&fActionInspector, SIGNAL(triggered()), this, SLOT(actionSettings()));
83 connect(&fFilter, SIGNAL(activated(QString)), this, SLOT(toggleFilter(QString)));
robertphillips@google.comd26c7062012-11-12 20:42:12 +000084 connect(&fActionProfile, SIGNAL(triggered()), this, SLOT(actionProfile()));
chudy@google.comc432f002012-07-10 13:19:25 +000085 connect(&fActionCancel, SIGNAL(triggered()), this, SLOT(actionCancel()));
chudy@google.com7e4cfbf2012-07-17 15:40:51 +000086 connect(&fActionClearBreakpoints, SIGNAL(triggered()), this, SLOT(actionClearBreakpoints()));
87 connect(&fActionClearDeletes, SIGNAL(triggered()), this, SLOT(actionClearDeletes()));
chudy@google.comc432f002012-07-10 13:19:25 +000088 connect(&fActionClose, SIGNAL(triggered()), this, SLOT(actionClose()));
chudy@google.comea5488b2012-07-26 19:38:22 +000089 connect(fSettingsWidget.getVisibilityButton(), SIGNAL(toggled(bool)), this, SLOT(actionCommandFilter()));
robertphillips@google.come8fe4bc2013-02-13 13:26:13 +000090#if SK_SUPPORT_GPU
chudy@google.comea5488b2012-07-26 19:38:22 +000091 connect(fSettingsWidget.getGLCheckBox(), SIGNAL(toggled(bool)), this, SLOT(actionGLWidget(bool)));
robertphillips@google.come8fe4bc2013-02-13 13:26:13 +000092#endif
chudy@google.comea5488b2012-07-26 19:38:22 +000093 connect(fSettingsWidget.getRasterCheckBox(), SIGNAL(toggled(bool)), this, SLOT(actionRasterWidget(bool)));
robertphillips@google.comf4741c12013-02-06 20:13:54 +000094 connect(fSettingsWidget.getOverdrawVizCheckBox(), SIGNAL(toggled(bool)), this, SLOT(actionOverdrawVizWidget(bool)));
chudy@google.comea5488b2012-07-26 19:38:22 +000095 connect(&fActionPause, SIGNAL(toggled(bool)), this, SLOT(pauseDrawing(bool)));
chudy@google.come504de02012-07-16 18:35:23 +000096 connect(&fActionCreateBreakpoint, SIGNAL(activated()), this, SLOT(toggleBreakpoint()));
chudy@google.com7e4cfbf2012-07-17 15:40:51 +000097 connect(&fActionShowDeletes, SIGNAL(triggered()), this, SLOT(showDeletes()));
chudy@google.comea5488b2012-07-26 19:38:22 +000098 connect(&fCanvasWidget, SIGNAL(hitChanged(int)), this, SLOT(selectCommand(int)));
99 connect(&fCanvasWidget, SIGNAL(hitChanged(int)), &fSettingsWidget, SLOT(updateHit(int)));
100 connect(&fCanvasWidget, SIGNAL(scaleFactorChanged(float)), this, SLOT(actionScale(float)));
101 connect(&fCanvasWidget, SIGNAL(commandChanged(int)), &fSettingsWidget, SLOT(updateCommand(int)));
chudy@google.com0ab03392012-07-28 20:16:11 +0000102 connect(&fActionSaveAs, SIGNAL(triggered()), this, SLOT(actionSaveAs()));
103 connect(&fActionSave, SIGNAL(triggered()), this, SLOT(actionSave()));
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000104
bungeman@google.come8cc6e82013-01-17 16:30:56 +0000105 fMapper.setMapping(&fActionZoomIn, SkCanvasWidget::kIn_ZoomCommand);
106 fMapper.setMapping(&fActionZoomOut, SkCanvasWidget::kOut_ZoomCommand);
chudy@google.coma1226312012-07-26 20:26:44 +0000107
108 connect(&fActionZoomIn, SIGNAL(triggered()), &fMapper, SLOT(map()));
109 connect(&fActionZoomOut, SIGNAL(triggered()), &fMapper, SLOT(map()));
bungeman@google.come8cc6e82013-01-17 16:30:56 +0000110 connect(&fMapper, SIGNAL(mapped(int)), &fCanvasWidget, SLOT(zoom(int)));
chudy@google.coma1226312012-07-26 20:26:44 +0000111
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000112 fInspectorWidget.setDisabled(true);
chudy@google.comd3058f52012-07-19 13:41:27 +0000113 fMenuEdit.setDisabled(true);
114 fMenuNavigate.setDisabled(true);
115 fMenuView.setDisabled(true);
chudy@google.combbad34d2012-08-13 14:26:36 +0000116
117 SkGraphics::Init();
chudy@google.com902ebe52012-06-29 14:21:22 +0000118}
119
chudy@google.combbad34d2012-08-13 14:26:36 +0000120SkDebuggerGUI::~SkDebuggerGUI() {
121 SkGraphics::Term();
122}
chudy@google.com902ebe52012-06-29 14:21:22 +0000123
124void SkDebuggerGUI::actionBreakpoints() {
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000125 fBreakpointsActivated = !fBreakpointsActivated;
chudy@google.comc432f002012-07-10 13:19:25 +0000126 for (int row = 0; row < fListWidget.count(); row++) {
127 QListWidgetItem *item = fListWidget.item(row);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000128 item->setHidden(item->checkState() == Qt::Unchecked && fBreakpointsActivated);
129 }
130}
chudy@google.com902ebe52012-06-29 14:21:22 +0000131
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000132void SkDebuggerGUI::showDeletes() {
133 fDeletesActivated = !fDeletesActivated;
134 for (int row = 0; row < fListWidget.count(); row++) {
135 QListWidgetItem *item = fListWidget.item(row);
chudy@google.com607357f2012-08-07 16:12:23 +0000136 item->setHidden(fDebugger.isCommandVisible(row)
137 && fDeletesActivated);
chudy@google.com902ebe52012-06-29 14:21:22 +0000138 }
139}
140
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000141// The timed picture playback uses the SkPicturePlayback's profiling stubs
142// to time individual commands. The offsets are needed to map SkPicture
143// offsets to individual commands.
144class SkTimedPicturePlayback : public SkPicturePlayback {
145public:
scroggo@google.com12d588a2013-02-25 16:05:00 +0000146 SkTimedPicturePlayback(SkStream* stream, const SkPictInfo& info,
skia.committer@gmail.com3e2345a2013-05-24 07:01:26 +0000147 SkPicture::InstallPixelRefProc proc,
robertphillips@google.com5f971142012-12-07 20:48:56 +0000148 const SkTDArray<bool>& deletedCommands)
scroggo@google.com12d588a2013-02-25 16:05:00 +0000149 : INHERITED(stream, info, proc)
bungeman@google.come8cc6e82013-01-17 16:30:56 +0000150 , fSkipCommands(deletedCommands)
151 , fTot(0.0)
152 , fCurCommand(0) {
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000153 fTimes.setCount(deletedCommands.count());
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000154 fTypeTimes.setCount(LAST_DRAWTYPE_ENUM+1);
155 this->resetTimes();
156 }
157
158 void resetTimes() {
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000159 for (int i = 0; i < fTimes.count(); ++i) {
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000160 fTimes[i] = 0.0;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000161 }
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000162 for (int i = 0; i < fTypeTimes.count(); ++i) {
163 fTypeTimes[i] = 0.0f;
164 }
165 fTot = 0.0;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000166 }
167
168 int count() const { return fTimes.count(); }
169
170 double time(int index) const { return fTimes[index] / fTot; }
171
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000172 const SkTDArray<double>* typeTimes() const { return &fTypeTimes; }
173
174 double totTime() const { return fTot; }
175
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000176protected:
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000177 BenchSysTimer fTimer;
robertphillips@google.com5f971142012-12-07 20:48:56 +0000178 SkTDArray<bool> fSkipCommands; // has the command been deleted in the GUI?
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000179 SkTDArray<double> fTimes; // sum of time consumed for each command
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000180 SkTDArray<double> fTypeTimes; // sum of time consumed for each type of command (e.g., drawPath)
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000181 double fTot; // total of all times in 'fTimes'
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000182 int fCurType;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000183 int fCurCommand; // the current command being executed/timed
184
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000185#ifdef SK_DEVELOPER
186 virtual bool preDraw(int opIndex, int type) SK_OVERRIDE {
187 fCurCommand = opIndex;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000188
robertphillips@google.com5f971142012-12-07 20:48:56 +0000189 if (fSkipCommands[fCurCommand]) {
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000190 return true;
robertphillips@google.com5f971142012-12-07 20:48:56 +0000191 }
192
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000193 fCurType = type;
194 // The SkDebugCanvas doesn't recognize these types. This class needs to
195 // convert or else we'll wind up with a mismatch between the type counts
196 // the debugger displays and the profile times.
197 if (DRAW_POS_TEXT_TOP_BOTTOM == type) {
198 fCurType = DRAW_POS_TEXT;
199 } else if (DRAW_POS_TEXT_H_TOP_BOTTOM == type) {
200 fCurType = DRAW_POS_TEXT_H;
201 }
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000202
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000203#if defined(SK_BUILD_FOR_WIN32)
204 // CPU timer doesn't work well on Windows
205 fTimer.startWall();
206#else
207 fTimer.startCpu();
208#endif
robertphillips@google.com5f971142012-12-07 20:48:56 +0000209
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000210 return false;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000211 }
212
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000213 virtual void postDraw(int opIndex) SK_OVERRIDE {
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000214#if defined(SK_BUILD_FOR_WIN32)
215 // CPU timer doesn't work well on Windows
216 double time = fTimer.endWall();
217#else
218 double time = fTimer.endCpu();
219#endif
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000220
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000221 SkASSERT(opIndex == fCurCommand);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000222 SkASSERT(fCurType <= LAST_DRAWTYPE_ENUM);
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000223
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000224 fTimes[fCurCommand] += time;
225 fTypeTimes[fCurType] += time;
226 fTot += time;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000227 }
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000228#endif
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000229
230private:
231 typedef SkPicturePlayback INHERITED;
232};
233
234// Wrap SkPicture to allow installation of an SkTimedPicturePlayback object
235class SkTimedPicture : public SkPicture {
236public:
scroggo@google.com9f123162013-02-22 22:37:31 +0000237 explicit SkTimedPicture(SkStream* stream, bool* success, SkPicture::InstallPixelRefProc proc,
robertphillips@google.com5f971142012-12-07 20:48:56 +0000238 const SkTDArray<bool>& deletedCommands) {
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000239 if (success) {
240 *success = false;
241 }
242 fRecord = NULL;
243 fPlayback = NULL;
244 fWidth = fHeight = 0;
245
246 SkPictInfo info;
247
248 if (!stream->read(&info, sizeof(info))) {
249 return;
250 }
251 if (SkPicture::PICTURE_VERSION != info.fVersion) {
252 return;
253 }
254
255 if (stream->readBool()) {
skia.committer@gmail.com884e60b2012-11-16 02:01:17 +0000256 fPlayback = SkNEW_ARGS(SkTimedPicturePlayback,
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000257 (stream, info, proc, deletedCommands));
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000258 }
259
260 // do this at the end, so that they will be zero if we hit an error.
261 fWidth = info.fWidth;
262 fHeight = info.fHeight;
263 if (success) {
264 *success = true;
265 }
266 }
267
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000268 void resetTimes() { ((SkTimedPicturePlayback*) fPlayback)->resetTimes(); }
269
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000270 int count() const { return ((SkTimedPicturePlayback*) fPlayback)->count(); }
271
272 // return the fraction of the total time this command consumed
273 double time(int index) const { return ((SkTimedPicturePlayback*) fPlayback)->time(index); }
274
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000275 const SkTDArray<double>* typeTimes() const { return ((SkTimedPicturePlayback*) fPlayback)->typeTimes(); }
276
277 double totTime() const { return ((SkTimedPicturePlayback*) fPlayback)->totTime(); }
278
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000279private:
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000280 // disallow default ctor b.c. we don't have a good way to setup the fPlayback ptr
281 SkTimedPicture();
282 // disallow the copy ctor - enabling would require copying code from SkPicture
283 SkTimedPicture(const SkTimedPicture& src);
284
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000285 typedef SkPicture INHERITED;
286};
287
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000288// This is a simplification of PictureBenchmark's run with the addition of
289// clearing of the times after the first pass (in resetTimes)
skia.committer@gmail.com34587162012-11-20 02:01:23 +0000290void SkDebuggerGUI::run(SkTimedPicture* pict,
291 sk_tools::PictureRenderer* renderer,
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000292 int repeats) {
293 SkASSERT(pict);
294 if (NULL == pict) {
295 return;
296 }
297
298 SkASSERT(renderer != NULL);
299 if (NULL == renderer) {
300 return;
301 }
302
303 renderer->init(pict);
304
305 renderer->setup();
306 renderer->render(NULL);
jvanverth@google.comade32662013-01-28 21:09:05 +0000307 renderer->resetState(true);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000308
309 // We throw this away the first batch of times to remove first time effects (such as paging in this program)
310 pict->resetTimes();
311
312 for (int i = 0; i < repeats; ++i) {
313 renderer->setup();
314 renderer->render(NULL);
jvanverth@google.comade32662013-01-28 21:09:05 +0000315 renderer->resetState(true);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000316 }
317
318 renderer->end();
319}
320
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000321void SkDebuggerGUI::actionProfile() {
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000322 // In order to profile we pass the command offsets (that were read-in
323 // in loadPicture by the SkOffsetPicture) to an SkTimedPlaybackPicture.
skia.committer@gmail.com884e60b2012-11-16 02:01:17 +0000324 // The SkTimedPlaybackPicture in turn passes the offsets to an
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000325 // SkTimedPicturePlayback object which uses them to track the performance
326 // of individual commands.
327 if (fFileName.isEmpty()) {
328 return;
329 }
330
331 SkFILEStream inputStream;
332
333 inputStream.setPath(fFileName.c_str());
334 if (!inputStream.isValid()) {
335 return;
336 }
337
338 bool success = false;
scroggo@google.comf8d7d272013-02-22 21:38:35 +0000339 SkTimedPicture picture(&inputStream, &success, &SkImageDecoder::DecodeMemory,
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000340 fSkipCommands);
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000341 if (!success) {
342 return;
343 }
344
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000345 // For now this #if allows switching between tiled and simple rendering
346 // modes. Eventually this will be accomplished via the GUI
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000347#if 0
348 // With the current batch of SysTimers, profiling in tiled mode
349 // gets swamped by the timing overhead:
350 //
351 // tile mode simple mode
352 // debugger 64.2ms 12.8ms
353 // bench_pictures 16.9ms 12.4ms
354 //
355 // This is b.c. in tiled mode each command is called many more times
356 // but typically does less work on each invocation (due to clipping)
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000357 sk_tools::TiledPictureRenderer* renderer = NULL;
358
359 renderer = SkNEW(sk_tools::TiledPictureRenderer);
360 renderer->setTileWidth(256);
361 renderer->setTileHeight(256);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000362#else
363 sk_tools::SimplePictureRenderer* renderer = NULL;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000364
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000365 renderer = SkNEW(sk_tools::SimplePictureRenderer);
robertphillips@google.com1447aa32013-01-30 21:09:09 +0000366
367#if SK_SUPPORT_GPU
368 if (Qt::Checked == fSettingsWidget.getGLCheckBox()->checkState()) {
369 renderer->setDeviceType(sk_tools::PictureRenderer::kGPU_DeviceType);
370 }
371#endif
372
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000373#endif
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000374
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000375 static const int kNumRepeats = 10;
376
377 run(&picture, renderer, kNumRepeats);
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000378
379 SkASSERT(picture.count() == fListWidget.count());
380
381 // extract the individual command times from the SkTimedPlaybackPicture
382 for (int i = 0; i < picture.count(); ++i) {
383 double temp = picture.time(i);
384
385 QListWidgetItem* item = fListWidget.item(i);
386
387 item->setData(Qt::UserRole + 4, 100.0*temp);
388 }
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000389
robertphillips@google.come428f9b2013-03-12 15:33:40 +0000390 setupOverviewText(picture.typeTimes(), picture.totTime(), kNumRepeats);
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000391}
392
chudy@google.com902ebe52012-06-29 14:21:22 +0000393void SkDebuggerGUI::actionCancel() {
chudy@google.comc432f002012-07-10 13:19:25 +0000394 for (int row = 0; row < fListWidget.count(); row++) {
395 fListWidget.item(row)->setHidden(false);
chudy@google.com902ebe52012-06-29 14:21:22 +0000396 }
397}
398
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000399void SkDebuggerGUI::actionClearBreakpoints() {
400 for (int row = 0; row < fListWidget.count(); row++) {
401 QListWidgetItem* item = fListWidget.item(row);
402 item->setCheckState(Qt::Unchecked);
403 item->setData(Qt::DecorationRole,
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000404 QPixmap(":/blank.png"));
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000405 }
406}
407
408void SkDebuggerGUI::actionClearDeletes() {
409 for (int row = 0; row < fListWidget.count(); row++) {
410 QListWidgetItem* item = fListWidget.item(row);
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000411 item->setData(Qt::UserRole + 2, QPixmap(":/blank.png"));
chudy@google.com607357f2012-08-07 16:12:23 +0000412 fDebugger.setCommandVisible(row, true);
robertphillips@google.com5f971142012-12-07 20:48:56 +0000413 fSkipCommands[row] = false;
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000414 }
415 if (fPause) {
416 fCanvasWidget.drawTo(fPausedRow);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000417 fImageWidget.draw();
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000418 } else {
419 fCanvasWidget.drawTo(fListWidget.currentRow());
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000420 fImageWidget.draw();
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000421 }
422}
423
chudy@google.com902ebe52012-06-29 14:21:22 +0000424void SkDebuggerGUI::actionCommandFilter() {
chudy@google.com607357f2012-08-07 16:12:23 +0000425 fDebugger.highlightCurrentCommand(
chudy@google.comc432f002012-07-10 13:19:25 +0000426 fSettingsWidget.getVisibilityButton()->isChecked());
427 fCanvasWidget.drawTo(fListWidget.currentRow());
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000428 fImageWidget.draw();
chudy@google.com902ebe52012-06-29 14:21:22 +0000429}
430
431void SkDebuggerGUI::actionClose() {
432 this->close();
433}
434
435void SkDebuggerGUI::actionDelete() {
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000436
robertphillips@google.comdd4b7452013-01-22 19:38:46 +0000437 for (int row = 0; row < fListWidget.count(); ++row) {
438 QListWidgetItem* item = fListWidget.item(row);
439
440 if (!item->isSelected()) {
441 continue;
442 }
443
444 if (fDebugger.isCommandVisible(row)) {
445 item->setData(Qt::UserRole + 2, QPixmap(":/delete.png"));
446 fDebugger.setCommandVisible(row, false);
447 fSkipCommands[row] = true;
448 } else {
449 item->setData(Qt::UserRole + 2, QPixmap(":/blank.png"));
450 fDebugger.setCommandVisible(row, true);
451 fSkipCommands[row] = false;
452 }
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000453 }
454
robertphillips@google.comdd4b7452013-01-22 19:38:46 +0000455 int currentRow = fListWidget.currentRow();
456
chudy@google.come504de02012-07-16 18:35:23 +0000457 if (fPause) {
458 fCanvasWidget.drawTo(fPausedRow);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000459 fImageWidget.draw();
chudy@google.come504de02012-07-16 18:35:23 +0000460 } else {
461 fCanvasWidget.drawTo(currentRow);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000462 fImageWidget.draw();
chudy@google.come504de02012-07-16 18:35:23 +0000463 }
chudy@google.com902ebe52012-06-29 14:21:22 +0000464}
465
robertphillips@google.come8fe4bc2013-02-13 13:26:13 +0000466#if SK_SUPPORT_GPU
chudy@google.comea5488b2012-07-26 19:38:22 +0000467void SkDebuggerGUI::actionGLWidget(bool isToggled) {
468 fCanvasWidget.setWidgetVisibility(SkCanvasWidget::kGPU_WidgetType, !isToggled);
469}
robertphillips@google.come8fe4bc2013-02-13 13:26:13 +0000470#endif
chudy@google.comea5488b2012-07-26 19:38:22 +0000471
chudy@google.com902ebe52012-06-29 14:21:22 +0000472void SkDebuggerGUI::actionInspector() {
chudy@google.comc432f002012-07-10 13:19:25 +0000473 if (fInspectorWidget.isHidden()) {
474 fInspectorWidget.setHidden(false);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000475 fImageWidget.setHidden(false);
chudy@google.com902ebe52012-06-29 14:21:22 +0000476 } else {
chudy@google.comc432f002012-07-10 13:19:25 +0000477 fInspectorWidget.setHidden(true);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000478 fImageWidget.setHidden(true);
chudy@google.com902ebe52012-06-29 14:21:22 +0000479 }
480}
481
482void SkDebuggerGUI::actionPlay() {
chudy@google.comc432f002012-07-10 13:19:25 +0000483 for (int row = fListWidget.currentRow() + 1; row < fListWidget.count();
chudy@google.com7dcae672012-07-09 20:26:53 +0000484 row++) {
chudy@google.comc432f002012-07-10 13:19:25 +0000485 QListWidgetItem *item = fListWidget.item(row);
chudy@google.com902ebe52012-06-29 14:21:22 +0000486 if (item->checkState() == Qt::Checked) {
chudy@google.comc432f002012-07-10 13:19:25 +0000487 fListWidget.setCurrentItem(item);
chudy@google.com902ebe52012-06-29 14:21:22 +0000488 return;
489 }
490 }
chudy@google.comc432f002012-07-10 13:19:25 +0000491 fListWidget.setCurrentRow(fListWidget.count() - 1);
chudy@google.com902ebe52012-06-29 14:21:22 +0000492}
493
chudy@google.comea5488b2012-07-26 19:38:22 +0000494void SkDebuggerGUI::actionRasterWidget(bool isToggled) {
495 fCanvasWidget.setWidgetVisibility(SkCanvasWidget::kRaster_8888_WidgetType, !isToggled);
496}
497
robertphillips@google.comf4741c12013-02-06 20:13:54 +0000498void SkDebuggerGUI::actionOverdrawVizWidget(bool isToggled) {
499 fDebugger.setOverdrawViz(isToggled);
500 fCanvasWidget.update();
501}
502
chudy@google.com902ebe52012-06-29 14:21:22 +0000503void SkDebuggerGUI::actionRewind() {
chudy@google.come504de02012-07-16 18:35:23 +0000504 fListWidget.setCurrentRow(0);
chudy@google.com902ebe52012-06-29 14:21:22 +0000505}
506
chudy@google.com0ab03392012-07-28 20:16:11 +0000507void SkDebuggerGUI::actionSave() {
robertphillips@google.come219baf2013-01-28 19:25:43 +0000508 fFileName = fPath.toAscii().data();
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000509 fFileName.append("/");
robertphillips@google.come219baf2013-01-28 19:25:43 +0000510 fFileName.append(fDirectoryWidget.currentItem()->text().toAscii().data());
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000511 saveToFile(fFileName);
chudy@google.com0ab03392012-07-28 20:16:11 +0000512}
513
514void SkDebuggerGUI::actionSaveAs() {
515 QString filename = QFileDialog::getSaveFileName(this, "Save File", "",
516 "Skia Picture (*skp)");
chudy@google.com38b08ce2012-07-28 23:26:10 +0000517 if (!filename.endsWith(".skp", Qt::CaseInsensitive)) {
chudy@google.com0ab03392012-07-28 20:16:11 +0000518 filename.append(".skp");
519 }
djsollen@google.comc3c82162012-11-13 18:35:10 +0000520 saveToFile(SkString(filename.toAscii().data()));
chudy@google.com0ab03392012-07-28 20:16:11 +0000521}
522
chudy@google.com7dcae672012-07-09 20:26:53 +0000523void SkDebuggerGUI::actionScale(float scaleFactor) {
chudy@google.comc432f002012-07-10 13:19:25 +0000524 fSettingsWidget.setZoomText(scaleFactor);
chudy@google.com7dcae672012-07-09 20:26:53 +0000525}
526
chudy@google.com902ebe52012-06-29 14:21:22 +0000527void SkDebuggerGUI::actionSettings() {
chudy@google.comc432f002012-07-10 13:19:25 +0000528 if (fSettingsWidget.isHidden()) {
529 fSettingsWidget.setHidden(false);
chudy@google.com902ebe52012-06-29 14:21:22 +0000530 } else {
chudy@google.comc432f002012-07-10 13:19:25 +0000531 fSettingsWidget.setHidden(true);
chudy@google.com902ebe52012-06-29 14:21:22 +0000532 }
533}
534
535void SkDebuggerGUI::actionStepBack() {
chudy@google.comc432f002012-07-10 13:19:25 +0000536 int currentRow = fListWidget.currentRow();
chudy@google.com902ebe52012-06-29 14:21:22 +0000537 if (currentRow != 0) {
chudy@google.comc432f002012-07-10 13:19:25 +0000538 fListWidget.setCurrentRow(currentRow - 1);
chudy@google.com902ebe52012-06-29 14:21:22 +0000539 }
540}
541
542void SkDebuggerGUI::actionStepForward() {
chudy@google.comc432f002012-07-10 13:19:25 +0000543 int currentRow = fListWidget.currentRow();
chudy@google.com902ebe52012-06-29 14:21:22 +0000544 QString curRow = QString::number(currentRow);
chudy@google.comc432f002012-07-10 13:19:25 +0000545 QString curCount = QString::number(fListWidget.count());
546 if (currentRow < fListWidget.count() - 1) {
547 fListWidget.setCurrentRow(currentRow + 1);
chudy@google.com902ebe52012-06-29 14:21:22 +0000548 }
549}
550
chudy@google.coma9e937c2012-08-03 17:32:05 +0000551void SkDebuggerGUI::drawComplete() {
chudy@google.com607357f2012-08-07 16:12:23 +0000552 fInspectorWidget.setMatrix(fDebugger.getCurrentMatrix());
553 fInspectorWidget.setClip(fDebugger.getCurrentClip());
chudy@google.coma9e937c2012-08-03 17:32:05 +0000554}
555
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000556void SkDebuggerGUI::saveToFile(const SkString& filename) {
557 SkFILEWStream file(filename.c_str());
robertphillips@google.com25bc2f82013-01-22 18:03:56 +0000558 SkAutoTUnref<SkPicture> copy(fDebugger.copyPicture());
559
560 copy->serialize(&file);
chudy@google.com0ab03392012-07-28 20:16:11 +0000561}
562
chudy@google.com902ebe52012-06-29 14:21:22 +0000563void SkDebuggerGUI::loadFile(QListWidgetItem *item) {
564 if (fDirectoryWidgetActive) {
robertphillips@google.come219baf2013-01-28 19:25:43 +0000565 fFileName = fPath.toAscii().data();
jvanverth@google.com0ac6f162013-02-05 19:44:07 +0000566 // don't add a '/' to files in the local directory
567 if (fFileName.size() > 0) {
568 fFileName.append("/");
569 }
robertphillips@google.come219baf2013-01-28 19:25:43 +0000570 fFileName.append(item->text().toAscii().data());
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000571 loadPicture(fFileName);
chudy@google.com902ebe52012-06-29 14:21:22 +0000572 }
573}
574
575void SkDebuggerGUI::openFile() {
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000576 QString temp = QFileDialog::getOpenFileName(this, tr("Open File"), "",
chudy@google.com7dcae672012-07-09 20:26:53 +0000577 tr("Files (*.*)"));
robertphillips@google.comff6e6ba2013-01-28 17:43:26 +0000578 openFile(temp);
579}
580
581void SkDebuggerGUI::openFile(const QString &filename) {
chudy@google.com902ebe52012-06-29 14:21:22 +0000582 fDirectoryWidgetActive = false;
robertphillips@google.comff6e6ba2013-01-28 17:43:26 +0000583 if (!filename.isEmpty()) {
584 QFileInfo pathInfo(filename);
585 loadPicture(SkString(filename.toAscii().data()));
586 setupDirectoryWidget(pathInfo.path());
chudy@google.com902ebe52012-06-29 14:21:22 +0000587 }
chudy@google.com902ebe52012-06-29 14:21:22 +0000588 fDirectoryWidgetActive = true;
589}
590
chudy@google.comc432f002012-07-10 13:19:25 +0000591void SkDebuggerGUI::pauseDrawing(bool isPaused) {
chudy@google.com607357f2012-08-07 16:12:23 +0000592 fPause = isPaused;
593 fPausedRow = fListWidget.currentRow();
594 fCanvasWidget.drawTo(fPausedRow);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000595 fImageWidget.draw();
chudy@google.com7dcae672012-07-09 20:26:53 +0000596}
597
chudy@google.com902ebe52012-06-29 14:21:22 +0000598void SkDebuggerGUI::registerListClick(QListWidgetItem *item) {
chudy@google.comd3058f52012-07-19 13:41:27 +0000599 if(!fLoading) {
600 int currentRow = fListWidget.currentRow();
chudy@google.comd3058f52012-07-19 13:41:27 +0000601
chudy@google.comea5488b2012-07-26 19:38:22 +0000602 if (currentRow != -1) {
603 if (!fPause) {
604 fCanvasWidget.drawTo(currentRow);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000605 fImageWidget.draw();
chudy@google.comd3058f52012-07-19 13:41:27 +0000606 }
chudy@google.com97cee972012-08-07 20:41:37 +0000607 SkTDArray<SkString*> *currInfo = fDebugger.getCommandInfo(
chudy@google.comea5488b2012-07-26 19:38:22 +0000608 currentRow);
609
610 /* TODO(chudy): Add command type before parameters. Rename v
611 * to something more informative. */
chudy@google.com97cee972012-08-07 20:41:37 +0000612 if (currInfo) {
chudy@google.comea5488b2012-07-26 19:38:22 +0000613 QString info;
614 info.append("<b>Parameters: </b><br/>");
chudy@google.com97cee972012-08-07 20:41:37 +0000615 for (int i = 0; i < currInfo->count(); i++) {
616
617 info.append(QString((*currInfo)[i]->c_str()));
chudy@google.comea5488b2012-07-26 19:38:22 +0000618 info.append("<br/>");
619 }
chudy@google.com6bd109a2012-08-14 19:34:13 +0000620 fInspectorWidget.setText(info, SkInspectorWidget::kDetail_TabType);
chudy@google.comea5488b2012-07-26 19:38:22 +0000621 fInspectorWidget.setDisabled(false);
chudy@google.comea5488b2012-07-26 19:38:22 +0000622 }
chudy@google.comd3058f52012-07-19 13:41:27 +0000623 }
chudy@google.comea5488b2012-07-26 19:38:22 +0000624
chudy@google.com902ebe52012-06-29 14:21:22 +0000625 }
626}
627
chudy@google.com9ca9bfe2012-07-12 21:58:14 +0000628void SkDebuggerGUI::selectCommand(int command) {
629 if (fPause) {
630 fListWidget.setCurrentRow(command);
631 }
632}
633
chudy@google.com902ebe52012-06-29 14:21:22 +0000634void SkDebuggerGUI::toggleBreakpoint() {
chudy@google.comc432f002012-07-10 13:19:25 +0000635 QListWidgetItem* item = fListWidget.currentItem();
chudy@google.com902ebe52012-06-29 14:21:22 +0000636 if (item->checkState() == Qt::Unchecked) {
637 item->setCheckState(Qt::Checked);
chudy@google.come565de42012-07-12 14:15:54 +0000638 item->setData(Qt::DecorationRole,
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000639 QPixmap(":/breakpoint_16x16.png"));
chudy@google.com902ebe52012-06-29 14:21:22 +0000640 } else {
chudy@google.com902ebe52012-06-29 14:21:22 +0000641 item->setCheckState(Qt::Unchecked);
chudy@google.come565de42012-07-12 14:15:54 +0000642 item->setData(Qt::DecorationRole,
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000643 QPixmap(":/blank.png"));
chudy@google.com902ebe52012-06-29 14:21:22 +0000644 }
645}
646
647void SkDebuggerGUI::toggleDirectory() {
chudy@google.com607357f2012-08-07 16:12:23 +0000648 fDirectoryWidget.setHidden(!fDirectoryWidget.isHidden());
chudy@google.com902ebe52012-06-29 14:21:22 +0000649}
650
651void SkDebuggerGUI::toggleFilter(QString string) {
chudy@google.comc432f002012-07-10 13:19:25 +0000652 for (int row = 0; row < fListWidget.count(); row++) {
653 QListWidgetItem *item = fListWidget.item(row);
chudy@google.com607357f2012-08-07 16:12:23 +0000654 item->setHidden(item->text() != string);
chudy@google.com902ebe52012-06-29 14:21:22 +0000655 }
656}
657
658void SkDebuggerGUI::setupUi(QMainWindow *SkDebuggerGUI) {
659 QIcon windowIcon;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000660 windowIcon.addFile(QString::fromUtf8(":/skia.png"), QSize(),
chudy@google.com7dcae672012-07-09 20:26:53 +0000661 QIcon::Normal, QIcon::Off);
chudy@google.com902ebe52012-06-29 14:21:22 +0000662 SkDebuggerGUI->setObjectName(QString::fromUtf8("SkDebuggerGUI"));
663 SkDebuggerGUI->resize(1200, 1000);
664 SkDebuggerGUI->setWindowIcon(windowIcon);
chudy@google.comc432f002012-07-10 13:19:25 +0000665 SkDebuggerGUI->setWindowTitle("Skia Debugger");
chudy@google.com902ebe52012-06-29 14:21:22 +0000666
chudy@google.come504de02012-07-16 18:35:23 +0000667 fActionOpen.setShortcuts(QKeySequence::Open);
chudy@google.comc432f002012-07-10 13:19:25 +0000668 fActionOpen.setText("Open");
chudy@google.com902ebe52012-06-29 14:21:22 +0000669
670 QIcon breakpoint;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000671 breakpoint.addFile(QString::fromUtf8(":/breakpoint.png"),
chudy@google.com7dcae672012-07-09 20:26:53 +0000672 QSize(), QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000673 fActionBreakpoint.setShortcut(QKeySequence(tr("Ctrl+B")));
chudy@google.comc432f002012-07-10 13:19:25 +0000674 fActionBreakpoint.setIcon(breakpoint);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000675 fActionBreakpoint.setText("Breakpoints");
chudy@google.com902ebe52012-06-29 14:21:22 +0000676
677 QIcon cancel;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000678 cancel.addFile(QString::fromUtf8(":/reload.png"), QSize(),
chudy@google.com7dcae672012-07-09 20:26:53 +0000679 QIcon::Normal, QIcon::Off);
chudy@google.comc432f002012-07-10 13:19:25 +0000680 fActionCancel.setIcon(cancel);
681 fActionCancel.setText("Clear Filter");
chudy@google.com902ebe52012-06-29 14:21:22 +0000682
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000683 fActionClearBreakpoints.setShortcut(QKeySequence(tr("Alt+B")));
684 fActionClearBreakpoints.setText("Clear Breakpoints");
685
686 fActionClearDeletes.setShortcut(QKeySequence(tr("Alt+X")));
687 fActionClearDeletes.setText("Clear Deletes");
688
chudy@google.come504de02012-07-16 18:35:23 +0000689 fActionClose.setShortcuts(QKeySequence::Quit);
chudy@google.comc432f002012-07-10 13:19:25 +0000690 fActionClose.setText("Exit");
chudy@google.com902ebe52012-06-29 14:21:22 +0000691
chudy@google.come504de02012-07-16 18:35:23 +0000692 fActionCreateBreakpoint.setShortcut(QKeySequence(tr("B")));
693 fActionCreateBreakpoint.setText("Set Breakpoint");
694
695 fActionDelete.setShortcut(QKeySequence(tr("X")));
chudy@google.comc432f002012-07-10 13:19:25 +0000696 fActionDelete.setText("Delete Command");
chudy@google.com902ebe52012-06-29 14:21:22 +0000697
chudy@google.come504de02012-07-16 18:35:23 +0000698 fActionDirectory.setShortcut(QKeySequence(tr("Ctrl+D")));
699 fActionDirectory.setText("Directory");
chudy@google.com902ebe52012-06-29 14:21:22 +0000700
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000701 QIcon profile;
robertphillips@google.comd1636362012-11-19 18:25:09 +0000702 profile.addFile(QString::fromUtf8(":/profile.png"), QSize(),
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000703 QIcon::Normal, QIcon::Off);
704 fActionProfile.setIcon(profile);
705 fActionProfile.setText("Profile");
robertphillips@google.come099bc42012-11-19 16:26:40 +0000706 fActionProfile.setDisabled(true);
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000707
chudy@google.comc432f002012-07-10 13:19:25 +0000708 QIcon inspector;
robertphillips@google.comd1636362012-11-19 18:25:09 +0000709 inspector.addFile(QString::fromUtf8(":/inspector.png"),
chudy@google.comc432f002012-07-10 13:19:25 +0000710 QSize(), QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000711 fActionInspector.setShortcut(QKeySequence(tr("Ctrl+I")));
chudy@google.comc432f002012-07-10 13:19:25 +0000712 fActionInspector.setIcon(inspector);
chudy@google.come504de02012-07-16 18:35:23 +0000713 fActionInspector.setText("Inspector");
chudy@google.com902ebe52012-06-29 14:21:22 +0000714
chudy@google.comc432f002012-07-10 13:19:25 +0000715 QIcon play;
robertphillips@google.comd1636362012-11-19 18:25:09 +0000716 play.addFile(QString::fromUtf8(":/play.png"), QSize(),
chudy@google.comc432f002012-07-10 13:19:25 +0000717 QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000718 fActionPlay.setShortcut(QKeySequence(tr("Ctrl+P")));
chudy@google.comc432f002012-07-10 13:19:25 +0000719 fActionPlay.setIcon(play);
720 fActionPlay.setText("Play");
chudy@google.com902ebe52012-06-29 14:21:22 +0000721
chudy@google.come504de02012-07-16 18:35:23 +0000722 QIcon pause;
robertphillips@google.comd1636362012-11-19 18:25:09 +0000723 pause.addFile(QString::fromUtf8(":/pause.png"), QSize(),
chudy@google.comc432f002012-07-10 13:19:25 +0000724 QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000725 fActionPause.setShortcut(QKeySequence(tr("Space")));
726 fActionPause.setCheckable(true);
727 fActionPause.setIcon(pause);
728 fActionPause.setText("Pause");
729
chudy@google.comc432f002012-07-10 13:19:25 +0000730 QIcon rewind;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000731 rewind.addFile(QString::fromUtf8(":/rewind.png"), QSize(),
chudy@google.comc432f002012-07-10 13:19:25 +0000732 QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000733 fActionRewind.setShortcut(QKeySequence(tr("Ctrl+R")));
chudy@google.comc432f002012-07-10 13:19:25 +0000734 fActionRewind.setIcon(rewind);
735 fActionRewind.setText("Rewind");
chudy@google.com902ebe52012-06-29 14:21:22 +0000736
chudy@google.com0ab03392012-07-28 20:16:11 +0000737 fActionSave.setShortcut(QKeySequence::Save);
738 fActionSave.setText("Save");
739 fActionSave.setDisabled(true);
740 fActionSaveAs.setShortcut(QKeySequence::SaveAs);
741 fActionSaveAs.setText("Save As");
742 fActionSaveAs.setDisabled(true);
743
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000744 fActionShowDeletes.setShortcut(QKeySequence(tr("Ctrl+X")));
745 fActionShowDeletes.setText("Deleted Commands");
746
chudy@google.comc432f002012-07-10 13:19:25 +0000747 QIcon stepBack;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000748 stepBack.addFile(QString::fromUtf8(":/previous.png"), QSize(),
chudy@google.comc432f002012-07-10 13:19:25 +0000749 QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000750 fActionStepBack.setShortcut(QKeySequence(tr("[")));
chudy@google.comc432f002012-07-10 13:19:25 +0000751 fActionStepBack.setIcon(stepBack);
752 fActionStepBack.setText("Step Back");
chudy@google.com902ebe52012-06-29 14:21:22 +0000753
chudy@google.comc432f002012-07-10 13:19:25 +0000754 QIcon stepForward;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000755 stepForward.addFile(QString::fromUtf8(":/next.png"),
chudy@google.comc432f002012-07-10 13:19:25 +0000756 QSize(), QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000757 fActionStepForward.setShortcut(QKeySequence(tr("]")));
chudy@google.comc432f002012-07-10 13:19:25 +0000758 fActionStepForward.setIcon(stepForward);
759 fActionStepForward.setText("Step Forward");
760
chudy@google.coma1226312012-07-26 20:26:44 +0000761 fActionZoomIn.setShortcut(QKeySequence(tr("Ctrl+=")));
762 fActionZoomIn.setText("Zoom In");
763 fActionZoomOut.setShortcut(QKeySequence(tr("Ctrl+-")));
764 fActionZoomOut.setText("Zoom Out");
765
chudy@google.comc432f002012-07-10 13:19:25 +0000766 fListWidget.setItemDelegate(new SkListWidget(&fListWidget));
767 fListWidget.setObjectName(QString::fromUtf8("listWidget"));
768 fListWidget.setMaximumWidth(250);
769
770 fFilter.addItem("--Filter By Available Commands--");
771
772 fDirectoryWidget.setMaximumWidth(250);
773 fDirectoryWidget.setStyleSheet("QListWidget::Item {padding: 5px;}");
774
775 fCanvasWidget.setSizePolicy(QSizePolicy::Expanding,
chudy@google.com7dcae672012-07-09 20:26:53 +0000776 QSizePolicy::Expanding);
chudy@google.com902ebe52012-06-29 14:21:22 +0000777
skia.committer@gmail.com1c9c0d32012-11-22 02:02:41 +0000778 fImageWidget.setFixedSize(SkImageWidget::kImageWidgetWidth,
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000779 SkImageWidget::kImageWidgetHeight);
780
chudy@google.comc432f002012-07-10 13:19:25 +0000781 fInspectorWidget.setSizePolicy(QSizePolicy::Expanding,
chudy@google.com7dcae672012-07-09 20:26:53 +0000782 QSizePolicy::Expanding);
chudy@google.comc432f002012-07-10 13:19:25 +0000783 fInspectorWidget.setMaximumHeight(300);
chudy@google.com902ebe52012-06-29 14:21:22 +0000784
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000785 fSettingsAndImageLayout.setSpacing(6);
786 fSettingsAndImageLayout.addWidget(&fSettingsWidget);
787 fSettingsAndImageLayout.addWidget(&fImageWidget);
788
chudy@google.comc432f002012-07-10 13:19:25 +0000789 fSettingsWidget.setSizePolicy(QSizePolicy::Expanding,
790 QSizePolicy::Expanding);
791 fSettingsWidget.setMaximumWidth(250);
chudy@google.com902ebe52012-06-29 14:21:22 +0000792
chudy@google.comc432f002012-07-10 13:19:25 +0000793 fLeftColumnLayout.setSpacing(6);
794 fLeftColumnLayout.addWidget(&fListWidget);
795 fLeftColumnLayout.addWidget(&fDirectoryWidget);
chudy@google.com902ebe52012-06-29 14:21:22 +0000796
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000797 fCanvasSettingsAndImageLayout.setSpacing(6);
798 fCanvasSettingsAndImageLayout.addWidget(&fCanvasWidget);
799 fCanvasSettingsAndImageLayout.addLayout(&fSettingsAndImageLayout);
800
chudy@google.com902ebe52012-06-29 14:21:22 +0000801
chudy@google.comc432f002012-07-10 13:19:25 +0000802 fMainAndRightColumnLayout.setSpacing(6);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000803 fMainAndRightColumnLayout.addLayout(&fCanvasSettingsAndImageLayout);
chudy@google.comc432f002012-07-10 13:19:25 +0000804 fMainAndRightColumnLayout.addWidget(&fInspectorWidget);
chudy@google.com902ebe52012-06-29 14:21:22 +0000805
chudy@google.com2d537a12012-07-31 12:49:52 +0000806 fCentralWidget.setLayout(&fContainerLayout);
chudy@google.comc432f002012-07-10 13:19:25 +0000807 fContainerLayout.setSpacing(6);
808 fContainerLayout.setContentsMargins(11, 11, 11, 11);
809 fContainerLayout.addLayout(&fLeftColumnLayout);
810 fContainerLayout.addLayout(&fMainAndRightColumnLayout);
811
812 SkDebuggerGUI->setCentralWidget(&fCentralWidget);
813 SkDebuggerGUI->setStatusBar(&fStatusBar);
814
chudy@google.come504de02012-07-16 18:35:23 +0000815 fToolBar.setIconSize(QSize(32, 32));
chudy@google.comc432f002012-07-10 13:19:25 +0000816 fToolBar.setToolButtonStyle(Qt::ToolButtonTextUnderIcon);
817 SkDebuggerGUI->addToolBar(Qt::TopToolBarArea, &fToolBar);
chudy@google.com902ebe52012-06-29 14:21:22 +0000818
chudy@google.com0ab03392012-07-28 20:16:11 +0000819 fSpacer.setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
chudy@google.com902ebe52012-06-29 14:21:22 +0000820
chudy@google.comc432f002012-07-10 13:19:25 +0000821 fToolBar.addAction(&fActionRewind);
822 fToolBar.addAction(&fActionStepBack);
chudy@google.come504de02012-07-16 18:35:23 +0000823 fToolBar.addAction(&fActionPause);
chudy@google.comc432f002012-07-10 13:19:25 +0000824 fToolBar.addAction(&fActionStepForward);
825 fToolBar.addAction(&fActionPlay);
826 fToolBar.addSeparator();
chudy@google.come504de02012-07-16 18:35:23 +0000827 fToolBar.addAction(&fActionInspector);
chudy@google.comc432f002012-07-10 13:19:25 +0000828 fToolBar.addSeparator();
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000829 fToolBar.addAction(&fActionProfile);
830
831 fToolBar.addSeparator();
chudy@google.com0ab03392012-07-28 20:16:11 +0000832 fToolBar.addWidget(&fSpacer);
chudy@google.comc432f002012-07-10 13:19:25 +0000833 fToolBar.addWidget(&fFilter);
834 fToolBar.addAction(&fActionCancel);
chudy@google.com902ebe52012-06-29 14:21:22 +0000835
836 // TODO(chudy): Remove static call.
837 fDirectoryWidgetActive = false;
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000838 fFileName = "";
robertphillips@google.comff6e6ba2013-01-28 17:43:26 +0000839 setupDirectoryWidget("");
chudy@google.com902ebe52012-06-29 14:21:22 +0000840 fDirectoryWidgetActive = true;
841
chudy@google.com902ebe52012-06-29 14:21:22 +0000842 // Menu Bar
chudy@google.comc432f002012-07-10 13:19:25 +0000843 fMenuFile.setTitle("File");
844 fMenuFile.addAction(&fActionOpen);
chudy@google.com0ab03392012-07-28 20:16:11 +0000845 fMenuFile.addAction(&fActionSave);
846 fMenuFile.addAction(&fActionSaveAs);
chudy@google.comc432f002012-07-10 13:19:25 +0000847 fMenuFile.addAction(&fActionClose);
chudy@google.come504de02012-07-16 18:35:23 +0000848
849 fMenuEdit.setTitle("Edit");
850 fMenuEdit.addAction(&fActionDelete);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000851 fMenuEdit.addAction(&fActionClearDeletes);
852 fMenuEdit.addSeparator();
chudy@google.come504de02012-07-16 18:35:23 +0000853 fMenuEdit.addAction(&fActionCreateBreakpoint);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000854 fMenuEdit.addAction(&fActionClearBreakpoints);
chudy@google.come504de02012-07-16 18:35:23 +0000855
chudy@google.comc432f002012-07-10 13:19:25 +0000856 fMenuNavigate.setTitle("Navigate");
chudy@google.come504de02012-07-16 18:35:23 +0000857 fMenuNavigate.addAction(&fActionRewind);
858 fMenuNavigate.addAction(&fActionStepBack);
859 fMenuNavigate.addAction(&fActionStepForward);
860 fMenuNavigate.addAction(&fActionPlay);
861 fMenuNavigate.addAction(&fActionPause);
chudy@google.comc432f002012-07-10 13:19:25 +0000862 fMenuNavigate.addAction(&fActionGoToLine);
chudy@google.come504de02012-07-16 18:35:23 +0000863
chudy@google.comc432f002012-07-10 13:19:25 +0000864 fMenuView.setTitle("View");
chudy@google.come504de02012-07-16 18:35:23 +0000865 fMenuView.addAction(&fActionBreakpoint);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000866 fMenuView.addAction(&fActionShowDeletes);
chudy@google.coma1226312012-07-26 20:26:44 +0000867 fMenuView.addAction(&fActionZoomIn);
868 fMenuView.addAction(&fActionZoomOut);
chudy@google.come504de02012-07-16 18:35:23 +0000869
870 fMenuWindows.setTitle("Window");
871 fMenuWindows.addAction(&fActionInspector);
872 fMenuWindows.addAction(&fActionDirectory);
chudy@google.comc432f002012-07-10 13:19:25 +0000873
874 fActionGoToLine.setText("Go to Line...");
875 fActionGoToLine.setDisabled(true);
876 fMenuBar.addAction(fMenuFile.menuAction());
chudy@google.come504de02012-07-16 18:35:23 +0000877 fMenuBar.addAction(fMenuEdit.menuAction());
chudy@google.comc432f002012-07-10 13:19:25 +0000878 fMenuBar.addAction(fMenuView.menuAction());
879 fMenuBar.addAction(fMenuNavigate.menuAction());
chudy@google.come504de02012-07-16 18:35:23 +0000880 fMenuBar.addAction(fMenuWindows.menuAction());
chudy@google.com902ebe52012-06-29 14:21:22 +0000881
chudy@google.com7dcae672012-07-09 20:26:53 +0000882 fPause = false;
883
chudy@google.comc432f002012-07-10 13:19:25 +0000884 SkDebuggerGUI->setMenuBar(&fMenuBar);
chudy@google.com902ebe52012-06-29 14:21:22 +0000885 QMetaObject::connectSlotsByName(SkDebuggerGUI);
886}
887
robertphillips@google.comff6e6ba2013-01-28 17:43:26 +0000888void SkDebuggerGUI::setupDirectoryWidget(const QString& path) {
889 fPath = path;
890 QDir dir(path);
chudy@google.com902ebe52012-06-29 14:21:22 +0000891 QRegExp r(".skp");
chudy@google.comc432f002012-07-10 13:19:25 +0000892 fDirectoryWidget.clear();
893 const QStringList files = dir.entryList();
chudy@google.com902ebe52012-06-29 14:21:22 +0000894 foreach (QString f, files) {
chudy@google.com7dcae672012-07-09 20:26:53 +0000895 if (f.contains(r))
chudy@google.comc432f002012-07-10 13:19:25 +0000896 fDirectoryWidget.addItem(f);
chudy@google.com902ebe52012-06-29 14:21:22 +0000897 }
898}
899
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000900void SkDebuggerGUI::loadPicture(const SkString& fileName) {
901 fFileName = fileName;
chudy@google.comd3058f52012-07-19 13:41:27 +0000902 fLoading = true;
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000903 SkStream* stream = SkNEW_ARGS(SkFILEStream, (fileName.c_str()));
robertphillips@google.com2d40ec42013-02-07 20:39:40 +0000904
905 bool success = false;
906
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000907 SkPicture* picture = SkNEW_ARGS(SkPicture,
908 (stream, &success, &SkImageDecoder::DecodeMemory));
robertphillips@google.com2d40ec42013-02-07 20:39:40 +0000909
910 if (!success) {
911 QMessageBox::critical(this, "Error loading file", "Couldn't read file, sorry.");
912 SkSafeUnref(stream);
913 return;
914 }
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000915
chudy@google.com686e6802012-08-14 16:00:32 +0000916 fCanvasWidget.resetWidgetTransform();
chudy@google.com607357f2012-08-07 16:12:23 +0000917 fDebugger.loadPicture(picture);
chudy@google.com4c7962e2012-08-14 19:38:31 +0000918
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000919 fSkipCommands.setCount(fDebugger.getSize());
920 for (int i = 0; i < fSkipCommands.count(); ++i) {
robertphillips@google.com5f971142012-12-07 20:48:56 +0000921 fSkipCommands[i] = false;
922 }
923
chudy@google.com607357f2012-08-07 16:12:23 +0000924 SkSafeUnref(stream);
925 SkSafeUnref(picture);
926
chudy@google.com97cee972012-08-07 20:41:37 +0000927 // Will this automatically clear out due to nature of refcnt?
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000928 SkTArray<SkString>* commands = fDebugger.getDrawCommandsAsStrings();
chudy@google.com607357f2012-08-07 16:12:23 +0000929
robertphillips@google.com6d9c92b2013-05-23 13:21:18 +0000930 fActionProfile.setDisabled(false);
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000931
chudy@google.com7dcae672012-07-09 20:26:53 +0000932 /* fDebugCanvas is reinitialized every load picture. Need it to retain value
chudy@google.com607357f2012-08-07 16:12:23 +0000933 * of the visibility filter.
934 * TODO(chudy): This should be deprecated since fDebugger is not
935 * recreated.
936 * */
937 fDebugger.highlightCurrentCommand(fSettingsWidget.getVisibilityButton()->isChecked());
938
chudy@google.com97cee972012-08-07 20:41:37 +0000939 setupListWidget(commands);
940 setupComboBox(commands);
robertphillips@google.come428f9b2013-03-12 15:33:40 +0000941 setupOverviewText(NULL, 0.0, 1);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000942 fInspectorWidget.setDisabled(false);
chudy@google.come606d6e2012-07-12 14:31:25 +0000943 fSettingsWidget.setDisabled(false);
chudy@google.comd3058f52012-07-19 13:41:27 +0000944 fMenuEdit.setDisabled(false);
945 fMenuNavigate.setDisabled(false);
946 fMenuView.setDisabled(false);
chudy@google.com0ab03392012-07-28 20:16:11 +0000947 fActionSave.setDisabled(false);
948 fActionSaveAs.setDisabled(false);
chudy@google.comd3058f52012-07-19 13:41:27 +0000949 fLoading = false;
950 actionPlay();
chudy@google.com902ebe52012-06-29 14:21:22 +0000951}
952
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000953void SkDebuggerGUI::setupListWidget(SkTArray<SkString>* command) {
chudy@google.comc432f002012-07-10 13:19:25 +0000954 fListWidget.clear();
chudy@google.com902ebe52012-06-29 14:21:22 +0000955 int counter = 0;
robertphillips@google.com30d35f22012-11-06 16:45:36 +0000956 int indent = 0;
chudy@google.com97cee972012-08-07 20:41:37 +0000957 for (int i = 0; i < command->count(); i++) {
chudy@google.com902ebe52012-06-29 14:21:22 +0000958 QListWidgetItem *item = new QListWidgetItem();
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000959 item->setData(Qt::DisplayRole, (*command)[i].c_str());
chudy@google.com902ebe52012-06-29 14:21:22 +0000960 item->setData(Qt::UserRole + 1, counter++);
robertphillips@google.com30d35f22012-11-06 16:45:36 +0000961
robertphillips@google.com0a4805e2013-05-29 13:24:23 +0000962 if (0 == strcmp("Restore", (*command)[i].c_str()) ||
963 0 == strcmp("EndCommentGroup", (*command)[i].c_str())) {
robertphillips@google.com30d35f22012-11-06 16:45:36 +0000964 indent -= 10;
965 }
966
967 item->setData(Qt::UserRole + 3, indent);
968
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000969 if (0 == strcmp("Save", (*command)[i].c_str()) ||
robertphillips@google.com0a4805e2013-05-29 13:24:23 +0000970 0 == strcmp("Save Layer", (*command)[i].c_str()) ||
971 0 == strcmp("BeginCommentGroup", (*command)[i].c_str())) {
robertphillips@google.com30d35f22012-11-06 16:45:36 +0000972 indent += 10;
973 }
974
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000975 item->setData(Qt::UserRole + 4, -1.0);
976
chudy@google.comc432f002012-07-10 13:19:25 +0000977 fListWidget.addItem(item);
chudy@google.com902ebe52012-06-29 14:21:22 +0000978 }
979}
980
skia.committer@gmail.com91274b92013-03-13 07:01:04 +0000981void SkDebuggerGUI::setupOverviewText(const SkTDArray<double>* typeTimes,
robertphillips@google.come428f9b2013-03-12 15:33:40 +0000982 double totTime,
983 int numRuns) {
borenet@google.com2d9dbd42013-03-12 13:07:40 +0000984 SkString overview;
robertphillips@google.come428f9b2013-03-12 15:33:40 +0000985 fDebugger.getOverviewText(typeTimes, totTime, &overview, numRuns);
borenet@google.com2d9dbd42013-03-12 13:07:40 +0000986 fInspectorWidget.setText(overview.c_str(), SkInspectorWidget::kOverview_TabType);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000987}
988
989void SkDebuggerGUI::setupComboBox(SkTArray<SkString>* command) {
990 fFilter.clear();
991 fFilter.addItem("--Filter By Available Commands--");
992
993 std::map<std::string, int> map;
994 for (int i = 0; i < command->count(); i++) {
995 map[(*command)[i].c_str()]++;
996 }
997
skia.committer@gmail.com34587162012-11-20 02:01:23 +0000998 for (std::map<std::string, int>::iterator it = map.begin(); it != map.end();
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000999 ++it) {
1000 fFilter.addItem((it->first).c_str());
1001 }
chudy@google.com902ebe52012-06-29 14:21:22 +00001002
1003 // NOTE(chudy): Makes first item unselectable.
chudy@google.com7dcae672012-07-09 20:26:53 +00001004 QStandardItemModel* model = qobject_cast<QStandardItemModel*>(
chudy@google.comc432f002012-07-10 13:19:25 +00001005 fFilter.model());
1006 QModelIndex firstIndex = model->index(0, fFilter.modelColumn(),
1007 fFilter.rootModelIndex());
chudy@google.com902ebe52012-06-29 14:21:22 +00001008 QStandardItem* firstItem = model->itemFromIndex(firstIndex);
1009 firstItem->setSelectable(false);
1010}