blob: fba7a1e42da3cdd95cf49fde2cc139c3e36b3540 [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:
146 SkTimedPicturePlayback(SkStream* stream, const SkPictInfo& info, bool* isValid,
147 SkSerializationHelpers::DecodeBitmap decoder,
robertphillips@google.com5f971142012-12-07 20:48:56 +0000148 const SkTDArray<size_t>& offsets,
149 const SkTDArray<bool>& deletedCommands)
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000150 : INHERITED(stream, info, isValid, decoder)
robertphillips@google.com5f971142012-12-07 20:48:56 +0000151 , fOffsets(offsets)
bungeman@google.come8cc6e82013-01-17 16:30:56 +0000152 , fSkipCommands(deletedCommands)
153 , fTot(0.0)
154 , fCurCommand(0) {
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000155 fTimes.setCount(fOffsets.count());
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000156 fTypeTimes.setCount(LAST_DRAWTYPE_ENUM+1);
157 this->resetTimes();
158 }
159
160 void resetTimes() {
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000161 for (int i = 0; i < fOffsets.count(); ++i) {
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000162 fTimes[i] = 0.0;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000163 }
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000164 for (int i = 0; i < fTypeTimes.count(); ++i) {
165 fTypeTimes[i] = 0.0f;
166 }
167 fTot = 0.0;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000168 }
169
170 int count() const { return fTimes.count(); }
171
172 double time(int index) const { return fTimes[index] / fTot; }
173
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000174 const SkTDArray<double>* typeTimes() const { return &fTypeTimes; }
175
176 double totTime() const { return fTot; }
177
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000178protected:
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000179 BenchSysTimer fTimer;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000180 SkTDArray<size_t> fOffsets; // offset in the SkPicture for each command
robertphillips@google.com5f971142012-12-07 20:48:56 +0000181 SkTDArray<bool> fSkipCommands; // has the command been deleted in the GUI?
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000182 SkTDArray<double> fTimes; // sum of time consumed for each command
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000183 SkTDArray<double> fTypeTimes; // sum of time consumed for each type of command (e.g., drawPath)
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000184 double fTot; // total of all times in 'fTimes'
185 size_t fCurOffset;
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000186 int fCurType;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000187 int fCurCommand; // the current command being executed/timed
188
robertphillips@google.comca47aae2012-12-12 15:58:25 +0000189 virtual size_t preDraw(size_t offset, int type) {
skia.committer@gmail.com884e60b2012-11-16 02:01:17 +0000190 // This search isn't as bad as it seems. In normal playback mode, the
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000191 // base class steps through the commands in order and can only skip ahead
skia.committer@gmail.com884e60b2012-11-16 02:01:17 +0000192 // a bit on a clip. This class is only used during profiling so we
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000193 // don't have to worry about forward/backward scrubbing through commands.
194 for (int i = 0; offset != fOffsets[fCurCommand]; ++i) {
195 fCurCommand = (fCurCommand+1) % fOffsets.count();
196 SkASSERT(i <= fOffsets.count()); // should always find the offset in the list
197 }
198
robertphillips@google.com5f971142012-12-07 20:48:56 +0000199 if (fSkipCommands[fCurCommand]) {
200 while (fCurCommand < fSkipCommands.count() && fSkipCommands[fCurCommand]) {
201 ++fCurCommand;
202 }
203 if (fCurCommand == fSkipCommands.count()) {
204 // Signal SkPicturePlayback to stop playing back
205 return SK_MaxU32;
206 }
207 return fOffsets[fCurCommand];
208 }
209
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000210 fCurOffset = offset;
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000211 fCurType = type;
212 // The SkDebugCanvas doesn't recognize these types. This class needs to
213 // convert or else we'll wind up with a mismatch between the type counts
214 // the debugger displays and the profile times.
215 if (DRAW_POS_TEXT_TOP_BOTTOM == type) {
216 fCurType = DRAW_POS_TEXT;
217 } else if (DRAW_POS_TEXT_H_TOP_BOTTOM == type) {
218 fCurType = DRAW_POS_TEXT_H;
219 }
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000220
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000221#if defined(SK_BUILD_FOR_WIN32)
222 // CPU timer doesn't work well on Windows
223 fTimer.startWall();
224#else
225 fTimer.startCpu();
226#endif
robertphillips@google.com5f971142012-12-07 20:48:56 +0000227
228 return 0;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000229 }
230
robertphillips@google.comca47aae2012-12-12 15:58:25 +0000231 virtual void postDraw(size_t offset) {
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000232#if defined(SK_BUILD_FOR_WIN32)
233 // CPU timer doesn't work well on Windows
234 double time = fTimer.endWall();
235#else
236 double time = fTimer.endCpu();
237#endif
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000238
239 SkASSERT(offset == fCurOffset);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000240 SkASSERT(fCurType <= LAST_DRAWTYPE_ENUM);
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000241
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000242 fTimes[fCurCommand] += time;
243 fTypeTimes[fCurType] += time;
244 fTot += time;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000245 }
246
247private:
248 typedef SkPicturePlayback INHERITED;
249};
250
251// Wrap SkPicture to allow installation of an SkTimedPicturePlayback object
252class SkTimedPicture : public SkPicture {
253public:
254 explicit SkTimedPicture(SkStream* stream,
255 bool* success,
256 SkSerializationHelpers::DecodeBitmap decoder,
robertphillips@google.com5f971142012-12-07 20:48:56 +0000257 const SkTDArray<size_t>& offsets,
258 const SkTDArray<bool>& deletedCommands) {
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000259 if (success) {
260 *success = false;
261 }
262 fRecord = NULL;
263 fPlayback = NULL;
264 fWidth = fHeight = 0;
265
266 SkPictInfo info;
267
268 if (!stream->read(&info, sizeof(info))) {
269 return;
270 }
271 if (SkPicture::PICTURE_VERSION != info.fVersion) {
272 return;
273 }
274
275 if (stream->readBool()) {
276 bool isValid = false;
skia.committer@gmail.com884e60b2012-11-16 02:01:17 +0000277 fPlayback = SkNEW_ARGS(SkTimedPicturePlayback,
robertphillips@google.com5f971142012-12-07 20:48:56 +0000278 (stream, info, &isValid, decoder, offsets, deletedCommands));
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000279 if (!isValid) {
280 SkDELETE(fPlayback);
281 fPlayback = NULL;
282 return;
283 }
284 }
285
286 // do this at the end, so that they will be zero if we hit an error.
287 fWidth = info.fWidth;
288 fHeight = info.fHeight;
289 if (success) {
290 *success = true;
291 }
292 }
293
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000294 void resetTimes() { ((SkTimedPicturePlayback*) fPlayback)->resetTimes(); }
295
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000296 int count() const { return ((SkTimedPicturePlayback*) fPlayback)->count(); }
297
298 // return the fraction of the total time this command consumed
299 double time(int index) const { return ((SkTimedPicturePlayback*) fPlayback)->time(index); }
300
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000301 const SkTDArray<double>* typeTimes() const { return ((SkTimedPicturePlayback*) fPlayback)->typeTimes(); }
302
303 double totTime() const { return ((SkTimedPicturePlayback*) fPlayback)->totTime(); }
304
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000305private:
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000306 // disallow default ctor b.c. we don't have a good way to setup the fPlayback ptr
307 SkTimedPicture();
308 // disallow the copy ctor - enabling would require copying code from SkPicture
309 SkTimedPicture(const SkTimedPicture& src);
310
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000311 typedef SkPicture INHERITED;
312};
313
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000314// This is a simplification of PictureBenchmark's run with the addition of
315// clearing of the times after the first pass (in resetTimes)
skia.committer@gmail.com34587162012-11-20 02:01:23 +0000316void SkDebuggerGUI::run(SkTimedPicture* pict,
317 sk_tools::PictureRenderer* renderer,
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000318 int repeats) {
319 SkASSERT(pict);
320 if (NULL == pict) {
321 return;
322 }
323
324 SkASSERT(renderer != NULL);
325 if (NULL == renderer) {
326 return;
327 }
328
329 renderer->init(pict);
330
331 renderer->setup();
332 renderer->render(NULL);
jvanverth@google.comade32662013-01-28 21:09:05 +0000333 renderer->resetState(true);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000334
335 // We throw this away the first batch of times to remove first time effects (such as paging in this program)
336 pict->resetTimes();
337
338 for (int i = 0; i < repeats; ++i) {
339 renderer->setup();
340 renderer->render(NULL);
jvanverth@google.comade32662013-01-28 21:09:05 +0000341 renderer->resetState(true);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000342 }
343
344 renderer->end();
345}
346
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000347void SkDebuggerGUI::actionProfile() {
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000348 // In order to profile we pass the command offsets (that were read-in
349 // in loadPicture by the SkOffsetPicture) to an SkTimedPlaybackPicture.
skia.committer@gmail.com884e60b2012-11-16 02:01:17 +0000350 // The SkTimedPlaybackPicture in turn passes the offsets to an
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000351 // SkTimedPicturePlayback object which uses them to track the performance
352 // of individual commands.
353 if (fFileName.isEmpty()) {
354 return;
355 }
356
357 SkFILEStream inputStream;
358
359 inputStream.setPath(fFileName.c_str());
360 if (!inputStream.isValid()) {
361 return;
362 }
363
364 bool success = false;
skia.committer@gmail.comc1f224a2012-12-08 02:01:38 +0000365 SkTimedPicture picture(&inputStream, &success, &SkImageDecoder::DecodeStream,
robertphillips@google.com5f971142012-12-07 20:48:56 +0000366 fOffsets, fSkipCommands);
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000367 if (!success) {
368 return;
369 }
370
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000371 // For now this #if allows switching between tiled and simple rendering
372 // modes. Eventually this will be accomplished via the GUI
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000373#if 0
374 // With the current batch of SysTimers, profiling in tiled mode
375 // gets swamped by the timing overhead:
376 //
377 // tile mode simple mode
378 // debugger 64.2ms 12.8ms
379 // bench_pictures 16.9ms 12.4ms
380 //
381 // This is b.c. in tiled mode each command is called many more times
382 // but typically does less work on each invocation (due to clipping)
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000383 sk_tools::TiledPictureRenderer* renderer = NULL;
384
385 renderer = SkNEW(sk_tools::TiledPictureRenderer);
386 renderer->setTileWidth(256);
387 renderer->setTileHeight(256);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000388#else
389 sk_tools::SimplePictureRenderer* renderer = NULL;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000390
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000391 renderer = SkNEW(sk_tools::SimplePictureRenderer);
robertphillips@google.com1447aa32013-01-30 21:09:09 +0000392
393#if SK_SUPPORT_GPU
394 if (Qt::Checked == fSettingsWidget.getGLCheckBox()->checkState()) {
395 renderer->setDeviceType(sk_tools::PictureRenderer::kGPU_DeviceType);
396 }
397#endif
398
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000399#endif
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000400
robertphillips@google.come174a8b2012-11-27 16:04:42 +0000401 static const int kNumRepeats = 10;
402
403 run(&picture, renderer, kNumRepeats);
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000404
405 SkASSERT(picture.count() == fListWidget.count());
406
407 // extract the individual command times from the SkTimedPlaybackPicture
408 for (int i = 0; i < picture.count(); ++i) {
409 double temp = picture.time(i);
410
411 QListWidgetItem* item = fListWidget.item(i);
412
413 item->setData(Qt::UserRole + 4, 100.0*temp);
414 }
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000415
416 setupOverviewText(picture.typeTimes(), picture.totTime());
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000417}
418
chudy@google.com902ebe52012-06-29 14:21:22 +0000419void SkDebuggerGUI::actionCancel() {
chudy@google.comc432f002012-07-10 13:19:25 +0000420 for (int row = 0; row < fListWidget.count(); row++) {
421 fListWidget.item(row)->setHidden(false);
chudy@google.com902ebe52012-06-29 14:21:22 +0000422 }
423}
424
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000425void SkDebuggerGUI::actionClearBreakpoints() {
426 for (int row = 0; row < fListWidget.count(); row++) {
427 QListWidgetItem* item = fListWidget.item(row);
428 item->setCheckState(Qt::Unchecked);
429 item->setData(Qt::DecorationRole,
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000430 QPixmap(":/blank.png"));
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000431 }
432}
433
434void SkDebuggerGUI::actionClearDeletes() {
435 for (int row = 0; row < fListWidget.count(); row++) {
436 QListWidgetItem* item = fListWidget.item(row);
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000437 item->setData(Qt::UserRole + 2, QPixmap(":/blank.png"));
chudy@google.com607357f2012-08-07 16:12:23 +0000438 fDebugger.setCommandVisible(row, true);
robertphillips@google.com5f971142012-12-07 20:48:56 +0000439 fSkipCommands[row] = false;
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000440 }
441 if (fPause) {
442 fCanvasWidget.drawTo(fPausedRow);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000443 fImageWidget.draw();
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000444 } else {
445 fCanvasWidget.drawTo(fListWidget.currentRow());
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000446 fImageWidget.draw();
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000447 }
448}
449
chudy@google.com902ebe52012-06-29 14:21:22 +0000450void SkDebuggerGUI::actionCommandFilter() {
chudy@google.com607357f2012-08-07 16:12:23 +0000451 fDebugger.highlightCurrentCommand(
chudy@google.comc432f002012-07-10 13:19:25 +0000452 fSettingsWidget.getVisibilityButton()->isChecked());
453 fCanvasWidget.drawTo(fListWidget.currentRow());
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000454 fImageWidget.draw();
chudy@google.com902ebe52012-06-29 14:21:22 +0000455}
456
457void SkDebuggerGUI::actionClose() {
458 this->close();
459}
460
461void SkDebuggerGUI::actionDelete() {
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000462
robertphillips@google.comdd4b7452013-01-22 19:38:46 +0000463 for (int row = 0; row < fListWidget.count(); ++row) {
464 QListWidgetItem* item = fListWidget.item(row);
465
466 if (!item->isSelected()) {
467 continue;
468 }
469
470 if (fDebugger.isCommandVisible(row)) {
471 item->setData(Qt::UserRole + 2, QPixmap(":/delete.png"));
472 fDebugger.setCommandVisible(row, false);
473 fSkipCommands[row] = true;
474 } else {
475 item->setData(Qt::UserRole + 2, QPixmap(":/blank.png"));
476 fDebugger.setCommandVisible(row, true);
477 fSkipCommands[row] = false;
478 }
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000479 }
480
robertphillips@google.comdd4b7452013-01-22 19:38:46 +0000481 int currentRow = fListWidget.currentRow();
482
chudy@google.come504de02012-07-16 18:35:23 +0000483 if (fPause) {
484 fCanvasWidget.drawTo(fPausedRow);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000485 fImageWidget.draw();
chudy@google.come504de02012-07-16 18:35:23 +0000486 } else {
487 fCanvasWidget.drawTo(currentRow);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000488 fImageWidget.draw();
chudy@google.come504de02012-07-16 18:35:23 +0000489 }
chudy@google.com902ebe52012-06-29 14:21:22 +0000490}
491
robertphillips@google.come8fe4bc2013-02-13 13:26:13 +0000492#if SK_SUPPORT_GPU
chudy@google.comea5488b2012-07-26 19:38:22 +0000493void SkDebuggerGUI::actionGLWidget(bool isToggled) {
494 fCanvasWidget.setWidgetVisibility(SkCanvasWidget::kGPU_WidgetType, !isToggled);
495}
robertphillips@google.come8fe4bc2013-02-13 13:26:13 +0000496#endif
chudy@google.comea5488b2012-07-26 19:38:22 +0000497
chudy@google.com902ebe52012-06-29 14:21:22 +0000498void SkDebuggerGUI::actionInspector() {
chudy@google.comc432f002012-07-10 13:19:25 +0000499 if (fInspectorWidget.isHidden()) {
500 fInspectorWidget.setHidden(false);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000501 fImageWidget.setHidden(false);
chudy@google.com902ebe52012-06-29 14:21:22 +0000502 } else {
chudy@google.comc432f002012-07-10 13:19:25 +0000503 fInspectorWidget.setHidden(true);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000504 fImageWidget.setHidden(true);
chudy@google.com902ebe52012-06-29 14:21:22 +0000505 }
506}
507
508void SkDebuggerGUI::actionPlay() {
chudy@google.comc432f002012-07-10 13:19:25 +0000509 for (int row = fListWidget.currentRow() + 1; row < fListWidget.count();
chudy@google.com7dcae672012-07-09 20:26:53 +0000510 row++) {
chudy@google.comc432f002012-07-10 13:19:25 +0000511 QListWidgetItem *item = fListWidget.item(row);
chudy@google.com902ebe52012-06-29 14:21:22 +0000512 if (item->checkState() == Qt::Checked) {
chudy@google.comc432f002012-07-10 13:19:25 +0000513 fListWidget.setCurrentItem(item);
chudy@google.com902ebe52012-06-29 14:21:22 +0000514 return;
515 }
516 }
chudy@google.comc432f002012-07-10 13:19:25 +0000517 fListWidget.setCurrentRow(fListWidget.count() - 1);
chudy@google.com902ebe52012-06-29 14:21:22 +0000518}
519
chudy@google.comea5488b2012-07-26 19:38:22 +0000520void SkDebuggerGUI::actionRasterWidget(bool isToggled) {
521 fCanvasWidget.setWidgetVisibility(SkCanvasWidget::kRaster_8888_WidgetType, !isToggled);
522}
523
robertphillips@google.comf4741c12013-02-06 20:13:54 +0000524void SkDebuggerGUI::actionOverdrawVizWidget(bool isToggled) {
525 fDebugger.setOverdrawViz(isToggled);
526 fCanvasWidget.update();
527}
528
chudy@google.com902ebe52012-06-29 14:21:22 +0000529void SkDebuggerGUI::actionRewind() {
chudy@google.come504de02012-07-16 18:35:23 +0000530 fListWidget.setCurrentRow(0);
chudy@google.com902ebe52012-06-29 14:21:22 +0000531}
532
chudy@google.com0ab03392012-07-28 20:16:11 +0000533void SkDebuggerGUI::actionSave() {
robertphillips@google.come219baf2013-01-28 19:25:43 +0000534 fFileName = fPath.toAscii().data();
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000535 fFileName.append("/");
robertphillips@google.come219baf2013-01-28 19:25:43 +0000536 fFileName.append(fDirectoryWidget.currentItem()->text().toAscii().data());
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000537 saveToFile(fFileName);
chudy@google.com0ab03392012-07-28 20:16:11 +0000538}
539
540void SkDebuggerGUI::actionSaveAs() {
541 QString filename = QFileDialog::getSaveFileName(this, "Save File", "",
542 "Skia Picture (*skp)");
chudy@google.com38b08ce2012-07-28 23:26:10 +0000543 if (!filename.endsWith(".skp", Qt::CaseInsensitive)) {
chudy@google.com0ab03392012-07-28 20:16:11 +0000544 filename.append(".skp");
545 }
djsollen@google.comc3c82162012-11-13 18:35:10 +0000546 saveToFile(SkString(filename.toAscii().data()));
chudy@google.com0ab03392012-07-28 20:16:11 +0000547}
548
chudy@google.com7dcae672012-07-09 20:26:53 +0000549void SkDebuggerGUI::actionScale(float scaleFactor) {
chudy@google.comc432f002012-07-10 13:19:25 +0000550 fSettingsWidget.setZoomText(scaleFactor);
chudy@google.com7dcae672012-07-09 20:26:53 +0000551}
552
chudy@google.com902ebe52012-06-29 14:21:22 +0000553void SkDebuggerGUI::actionSettings() {
chudy@google.comc432f002012-07-10 13:19:25 +0000554 if (fSettingsWidget.isHidden()) {
555 fSettingsWidget.setHidden(false);
chudy@google.com902ebe52012-06-29 14:21:22 +0000556 } else {
chudy@google.comc432f002012-07-10 13:19:25 +0000557 fSettingsWidget.setHidden(true);
chudy@google.com902ebe52012-06-29 14:21:22 +0000558 }
559}
560
561void SkDebuggerGUI::actionStepBack() {
chudy@google.comc432f002012-07-10 13:19:25 +0000562 int currentRow = fListWidget.currentRow();
chudy@google.com902ebe52012-06-29 14:21:22 +0000563 if (currentRow != 0) {
chudy@google.comc432f002012-07-10 13:19:25 +0000564 fListWidget.setCurrentRow(currentRow - 1);
chudy@google.com902ebe52012-06-29 14:21:22 +0000565 }
566}
567
568void SkDebuggerGUI::actionStepForward() {
chudy@google.comc432f002012-07-10 13:19:25 +0000569 int currentRow = fListWidget.currentRow();
chudy@google.com902ebe52012-06-29 14:21:22 +0000570 QString curRow = QString::number(currentRow);
chudy@google.comc432f002012-07-10 13:19:25 +0000571 QString curCount = QString::number(fListWidget.count());
572 if (currentRow < fListWidget.count() - 1) {
573 fListWidget.setCurrentRow(currentRow + 1);
chudy@google.com902ebe52012-06-29 14:21:22 +0000574 }
575}
576
chudy@google.coma9e937c2012-08-03 17:32:05 +0000577void SkDebuggerGUI::drawComplete() {
chudy@google.com607357f2012-08-07 16:12:23 +0000578 fInspectorWidget.setMatrix(fDebugger.getCurrentMatrix());
579 fInspectorWidget.setClip(fDebugger.getCurrentClip());
chudy@google.coma9e937c2012-08-03 17:32:05 +0000580}
581
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000582void SkDebuggerGUI::saveToFile(const SkString& filename) {
583 SkFILEWStream file(filename.c_str());
robertphillips@google.com25bc2f82013-01-22 18:03:56 +0000584 SkAutoTUnref<SkPicture> copy(fDebugger.copyPicture());
585
586 copy->serialize(&file);
chudy@google.com0ab03392012-07-28 20:16:11 +0000587}
588
chudy@google.com902ebe52012-06-29 14:21:22 +0000589void SkDebuggerGUI::loadFile(QListWidgetItem *item) {
590 if (fDirectoryWidgetActive) {
robertphillips@google.come219baf2013-01-28 19:25:43 +0000591 fFileName = fPath.toAscii().data();
jvanverth@google.com0ac6f162013-02-05 19:44:07 +0000592 // don't add a '/' to files in the local directory
593 if (fFileName.size() > 0) {
594 fFileName.append("/");
595 }
robertphillips@google.come219baf2013-01-28 19:25:43 +0000596 fFileName.append(item->text().toAscii().data());
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000597 loadPicture(fFileName);
chudy@google.com902ebe52012-06-29 14:21:22 +0000598 }
599}
600
601void SkDebuggerGUI::openFile() {
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000602 QString temp = QFileDialog::getOpenFileName(this, tr("Open File"), "",
chudy@google.com7dcae672012-07-09 20:26:53 +0000603 tr("Files (*.*)"));
robertphillips@google.comff6e6ba2013-01-28 17:43:26 +0000604 openFile(temp);
605}
606
607void SkDebuggerGUI::openFile(const QString &filename) {
chudy@google.com902ebe52012-06-29 14:21:22 +0000608 fDirectoryWidgetActive = false;
robertphillips@google.comff6e6ba2013-01-28 17:43:26 +0000609 if (!filename.isEmpty()) {
610 QFileInfo pathInfo(filename);
611 loadPicture(SkString(filename.toAscii().data()));
612 setupDirectoryWidget(pathInfo.path());
chudy@google.com902ebe52012-06-29 14:21:22 +0000613 }
chudy@google.com902ebe52012-06-29 14:21:22 +0000614 fDirectoryWidgetActive = true;
615}
616
chudy@google.comc432f002012-07-10 13:19:25 +0000617void SkDebuggerGUI::pauseDrawing(bool isPaused) {
chudy@google.com607357f2012-08-07 16:12:23 +0000618 fPause = isPaused;
619 fPausedRow = fListWidget.currentRow();
620 fCanvasWidget.drawTo(fPausedRow);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000621 fImageWidget.draw();
chudy@google.com7dcae672012-07-09 20:26:53 +0000622}
623
chudy@google.com902ebe52012-06-29 14:21:22 +0000624void SkDebuggerGUI::registerListClick(QListWidgetItem *item) {
chudy@google.comd3058f52012-07-19 13:41:27 +0000625 if(!fLoading) {
626 int currentRow = fListWidget.currentRow();
chudy@google.comd3058f52012-07-19 13:41:27 +0000627
chudy@google.comea5488b2012-07-26 19:38:22 +0000628 if (currentRow != -1) {
629 if (!fPause) {
630 fCanvasWidget.drawTo(currentRow);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000631 fImageWidget.draw();
chudy@google.comd3058f52012-07-19 13:41:27 +0000632 }
chudy@google.com97cee972012-08-07 20:41:37 +0000633 SkTDArray<SkString*> *currInfo = fDebugger.getCommandInfo(
chudy@google.comea5488b2012-07-26 19:38:22 +0000634 currentRow);
635
636 /* TODO(chudy): Add command type before parameters. Rename v
637 * to something more informative. */
chudy@google.com97cee972012-08-07 20:41:37 +0000638 if (currInfo) {
chudy@google.comea5488b2012-07-26 19:38:22 +0000639 QString info;
640 info.append("<b>Parameters: </b><br/>");
chudy@google.com97cee972012-08-07 20:41:37 +0000641 for (int i = 0; i < currInfo->count(); i++) {
642
643 info.append(QString((*currInfo)[i]->c_str()));
chudy@google.comea5488b2012-07-26 19:38:22 +0000644 info.append("<br/>");
645 }
chudy@google.com6bd109a2012-08-14 19:34:13 +0000646 fInspectorWidget.setText(info, SkInspectorWidget::kDetail_TabType);
chudy@google.comea5488b2012-07-26 19:38:22 +0000647 fInspectorWidget.setDisabled(false);
chudy@google.comea5488b2012-07-26 19:38:22 +0000648 }
chudy@google.comd3058f52012-07-19 13:41:27 +0000649 }
chudy@google.comea5488b2012-07-26 19:38:22 +0000650
chudy@google.com902ebe52012-06-29 14:21:22 +0000651 }
652}
653
chudy@google.com9ca9bfe2012-07-12 21:58:14 +0000654void SkDebuggerGUI::selectCommand(int command) {
655 if (fPause) {
656 fListWidget.setCurrentRow(command);
657 }
658}
659
chudy@google.com902ebe52012-06-29 14:21:22 +0000660void SkDebuggerGUI::toggleBreakpoint() {
chudy@google.comc432f002012-07-10 13:19:25 +0000661 QListWidgetItem* item = fListWidget.currentItem();
chudy@google.com902ebe52012-06-29 14:21:22 +0000662 if (item->checkState() == Qt::Unchecked) {
663 item->setCheckState(Qt::Checked);
chudy@google.come565de42012-07-12 14:15:54 +0000664 item->setData(Qt::DecorationRole,
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000665 QPixmap(":/breakpoint_16x16.png"));
chudy@google.com902ebe52012-06-29 14:21:22 +0000666 } else {
chudy@google.com902ebe52012-06-29 14:21:22 +0000667 item->setCheckState(Qt::Unchecked);
chudy@google.come565de42012-07-12 14:15:54 +0000668 item->setData(Qt::DecorationRole,
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000669 QPixmap(":/blank.png"));
chudy@google.com902ebe52012-06-29 14:21:22 +0000670 }
671}
672
673void SkDebuggerGUI::toggleDirectory() {
chudy@google.com607357f2012-08-07 16:12:23 +0000674 fDirectoryWidget.setHidden(!fDirectoryWidget.isHidden());
chudy@google.com902ebe52012-06-29 14:21:22 +0000675}
676
677void SkDebuggerGUI::toggleFilter(QString string) {
chudy@google.comc432f002012-07-10 13:19:25 +0000678 for (int row = 0; row < fListWidget.count(); row++) {
679 QListWidgetItem *item = fListWidget.item(row);
chudy@google.com607357f2012-08-07 16:12:23 +0000680 item->setHidden(item->text() != string);
chudy@google.com902ebe52012-06-29 14:21:22 +0000681 }
682}
683
684void SkDebuggerGUI::setupUi(QMainWindow *SkDebuggerGUI) {
685 QIcon windowIcon;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000686 windowIcon.addFile(QString::fromUtf8(":/skia.png"), QSize(),
chudy@google.com7dcae672012-07-09 20:26:53 +0000687 QIcon::Normal, QIcon::Off);
chudy@google.com902ebe52012-06-29 14:21:22 +0000688 SkDebuggerGUI->setObjectName(QString::fromUtf8("SkDebuggerGUI"));
689 SkDebuggerGUI->resize(1200, 1000);
690 SkDebuggerGUI->setWindowIcon(windowIcon);
chudy@google.comc432f002012-07-10 13:19:25 +0000691 SkDebuggerGUI->setWindowTitle("Skia Debugger");
chudy@google.com902ebe52012-06-29 14:21:22 +0000692
chudy@google.come504de02012-07-16 18:35:23 +0000693 fActionOpen.setShortcuts(QKeySequence::Open);
chudy@google.comc432f002012-07-10 13:19:25 +0000694 fActionOpen.setText("Open");
chudy@google.com902ebe52012-06-29 14:21:22 +0000695
696 QIcon breakpoint;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000697 breakpoint.addFile(QString::fromUtf8(":/breakpoint.png"),
chudy@google.com7dcae672012-07-09 20:26:53 +0000698 QSize(), QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000699 fActionBreakpoint.setShortcut(QKeySequence(tr("Ctrl+B")));
chudy@google.comc432f002012-07-10 13:19:25 +0000700 fActionBreakpoint.setIcon(breakpoint);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000701 fActionBreakpoint.setText("Breakpoints");
chudy@google.com902ebe52012-06-29 14:21:22 +0000702
703 QIcon cancel;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000704 cancel.addFile(QString::fromUtf8(":/reload.png"), QSize(),
chudy@google.com7dcae672012-07-09 20:26:53 +0000705 QIcon::Normal, QIcon::Off);
chudy@google.comc432f002012-07-10 13:19:25 +0000706 fActionCancel.setIcon(cancel);
707 fActionCancel.setText("Clear Filter");
chudy@google.com902ebe52012-06-29 14:21:22 +0000708
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000709 fActionClearBreakpoints.setShortcut(QKeySequence(tr("Alt+B")));
710 fActionClearBreakpoints.setText("Clear Breakpoints");
711
712 fActionClearDeletes.setShortcut(QKeySequence(tr("Alt+X")));
713 fActionClearDeletes.setText("Clear Deletes");
714
chudy@google.come504de02012-07-16 18:35:23 +0000715 fActionClose.setShortcuts(QKeySequence::Quit);
chudy@google.comc432f002012-07-10 13:19:25 +0000716 fActionClose.setText("Exit");
chudy@google.com902ebe52012-06-29 14:21:22 +0000717
chudy@google.come504de02012-07-16 18:35:23 +0000718 fActionCreateBreakpoint.setShortcut(QKeySequence(tr("B")));
719 fActionCreateBreakpoint.setText("Set Breakpoint");
720
721 fActionDelete.setShortcut(QKeySequence(tr("X")));
chudy@google.comc432f002012-07-10 13:19:25 +0000722 fActionDelete.setText("Delete Command");
chudy@google.com902ebe52012-06-29 14:21:22 +0000723
chudy@google.come504de02012-07-16 18:35:23 +0000724 fActionDirectory.setShortcut(QKeySequence(tr("Ctrl+D")));
725 fActionDirectory.setText("Directory");
chudy@google.com902ebe52012-06-29 14:21:22 +0000726
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000727 QIcon profile;
robertphillips@google.comd1636362012-11-19 18:25:09 +0000728 profile.addFile(QString::fromUtf8(":/profile.png"), QSize(),
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000729 QIcon::Normal, QIcon::Off);
730 fActionProfile.setIcon(profile);
731 fActionProfile.setText("Profile");
robertphillips@google.come099bc42012-11-19 16:26:40 +0000732 fActionProfile.setDisabled(true);
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000733
chudy@google.comc432f002012-07-10 13:19:25 +0000734 QIcon inspector;
robertphillips@google.comd1636362012-11-19 18:25:09 +0000735 inspector.addFile(QString::fromUtf8(":/inspector.png"),
chudy@google.comc432f002012-07-10 13:19:25 +0000736 QSize(), QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000737 fActionInspector.setShortcut(QKeySequence(tr("Ctrl+I")));
chudy@google.comc432f002012-07-10 13:19:25 +0000738 fActionInspector.setIcon(inspector);
chudy@google.come504de02012-07-16 18:35:23 +0000739 fActionInspector.setText("Inspector");
chudy@google.com902ebe52012-06-29 14:21:22 +0000740
chudy@google.comc432f002012-07-10 13:19:25 +0000741 QIcon play;
robertphillips@google.comd1636362012-11-19 18:25:09 +0000742 play.addFile(QString::fromUtf8(":/play.png"), QSize(),
chudy@google.comc432f002012-07-10 13:19:25 +0000743 QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000744 fActionPlay.setShortcut(QKeySequence(tr("Ctrl+P")));
chudy@google.comc432f002012-07-10 13:19:25 +0000745 fActionPlay.setIcon(play);
746 fActionPlay.setText("Play");
chudy@google.com902ebe52012-06-29 14:21:22 +0000747
chudy@google.come504de02012-07-16 18:35:23 +0000748 QIcon pause;
robertphillips@google.comd1636362012-11-19 18:25:09 +0000749 pause.addFile(QString::fromUtf8(":/pause.png"), QSize(),
chudy@google.comc432f002012-07-10 13:19:25 +0000750 QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000751 fActionPause.setShortcut(QKeySequence(tr("Space")));
752 fActionPause.setCheckable(true);
753 fActionPause.setIcon(pause);
754 fActionPause.setText("Pause");
755
chudy@google.comc432f002012-07-10 13:19:25 +0000756 QIcon rewind;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000757 rewind.addFile(QString::fromUtf8(":/rewind.png"), QSize(),
chudy@google.comc432f002012-07-10 13:19:25 +0000758 QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000759 fActionRewind.setShortcut(QKeySequence(tr("Ctrl+R")));
chudy@google.comc432f002012-07-10 13:19:25 +0000760 fActionRewind.setIcon(rewind);
761 fActionRewind.setText("Rewind");
chudy@google.com902ebe52012-06-29 14:21:22 +0000762
chudy@google.com0ab03392012-07-28 20:16:11 +0000763 fActionSave.setShortcut(QKeySequence::Save);
764 fActionSave.setText("Save");
765 fActionSave.setDisabled(true);
766 fActionSaveAs.setShortcut(QKeySequence::SaveAs);
767 fActionSaveAs.setText("Save As");
768 fActionSaveAs.setDisabled(true);
769
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000770 fActionShowDeletes.setShortcut(QKeySequence(tr("Ctrl+X")));
771 fActionShowDeletes.setText("Deleted Commands");
772
chudy@google.comc432f002012-07-10 13:19:25 +0000773 QIcon stepBack;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000774 stepBack.addFile(QString::fromUtf8(":/previous.png"), QSize(),
chudy@google.comc432f002012-07-10 13:19:25 +0000775 QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000776 fActionStepBack.setShortcut(QKeySequence(tr("[")));
chudy@google.comc432f002012-07-10 13:19:25 +0000777 fActionStepBack.setIcon(stepBack);
778 fActionStepBack.setText("Step Back");
chudy@google.com902ebe52012-06-29 14:21:22 +0000779
chudy@google.comc432f002012-07-10 13:19:25 +0000780 QIcon stepForward;
robertphillips@google.com8e41a162012-11-19 17:39:18 +0000781 stepForward.addFile(QString::fromUtf8(":/next.png"),
chudy@google.comc432f002012-07-10 13:19:25 +0000782 QSize(), QIcon::Normal, QIcon::Off);
chudy@google.come504de02012-07-16 18:35:23 +0000783 fActionStepForward.setShortcut(QKeySequence(tr("]")));
chudy@google.comc432f002012-07-10 13:19:25 +0000784 fActionStepForward.setIcon(stepForward);
785 fActionStepForward.setText("Step Forward");
786
chudy@google.coma1226312012-07-26 20:26:44 +0000787 fActionZoomIn.setShortcut(QKeySequence(tr("Ctrl+=")));
788 fActionZoomIn.setText("Zoom In");
789 fActionZoomOut.setShortcut(QKeySequence(tr("Ctrl+-")));
790 fActionZoomOut.setText("Zoom Out");
791
chudy@google.comc432f002012-07-10 13:19:25 +0000792 fListWidget.setItemDelegate(new SkListWidget(&fListWidget));
793 fListWidget.setObjectName(QString::fromUtf8("listWidget"));
794 fListWidget.setMaximumWidth(250);
795
796 fFilter.addItem("--Filter By Available Commands--");
797
798 fDirectoryWidget.setMaximumWidth(250);
799 fDirectoryWidget.setStyleSheet("QListWidget::Item {padding: 5px;}");
800
801 fCanvasWidget.setSizePolicy(QSizePolicy::Expanding,
chudy@google.com7dcae672012-07-09 20:26:53 +0000802 QSizePolicy::Expanding);
chudy@google.com902ebe52012-06-29 14:21:22 +0000803
skia.committer@gmail.com1c9c0d32012-11-22 02:02:41 +0000804 fImageWidget.setFixedSize(SkImageWidget::kImageWidgetWidth,
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000805 SkImageWidget::kImageWidgetHeight);
806
chudy@google.comc432f002012-07-10 13:19:25 +0000807 fInspectorWidget.setSizePolicy(QSizePolicy::Expanding,
chudy@google.com7dcae672012-07-09 20:26:53 +0000808 QSizePolicy::Expanding);
chudy@google.comc432f002012-07-10 13:19:25 +0000809 fInspectorWidget.setMaximumHeight(300);
chudy@google.com902ebe52012-06-29 14:21:22 +0000810
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000811 fSettingsAndImageLayout.setSpacing(6);
812 fSettingsAndImageLayout.addWidget(&fSettingsWidget);
813 fSettingsAndImageLayout.addWidget(&fImageWidget);
814
chudy@google.comc432f002012-07-10 13:19:25 +0000815 fSettingsWidget.setSizePolicy(QSizePolicy::Expanding,
816 QSizePolicy::Expanding);
817 fSettingsWidget.setMaximumWidth(250);
chudy@google.com902ebe52012-06-29 14:21:22 +0000818
chudy@google.comc432f002012-07-10 13:19:25 +0000819 fLeftColumnLayout.setSpacing(6);
820 fLeftColumnLayout.addWidget(&fListWidget);
821 fLeftColumnLayout.addWidget(&fDirectoryWidget);
chudy@google.com902ebe52012-06-29 14:21:22 +0000822
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000823 fCanvasSettingsAndImageLayout.setSpacing(6);
824 fCanvasSettingsAndImageLayout.addWidget(&fCanvasWidget);
825 fCanvasSettingsAndImageLayout.addLayout(&fSettingsAndImageLayout);
826
chudy@google.com902ebe52012-06-29 14:21:22 +0000827
chudy@google.comc432f002012-07-10 13:19:25 +0000828 fMainAndRightColumnLayout.setSpacing(6);
robertphillips@google.com6dec8fc2012-11-21 17:11:02 +0000829 fMainAndRightColumnLayout.addLayout(&fCanvasSettingsAndImageLayout);
chudy@google.comc432f002012-07-10 13:19:25 +0000830 fMainAndRightColumnLayout.addWidget(&fInspectorWidget);
chudy@google.com902ebe52012-06-29 14:21:22 +0000831
chudy@google.com2d537a12012-07-31 12:49:52 +0000832 fCentralWidget.setLayout(&fContainerLayout);
chudy@google.comc432f002012-07-10 13:19:25 +0000833 fContainerLayout.setSpacing(6);
834 fContainerLayout.setContentsMargins(11, 11, 11, 11);
835 fContainerLayout.addLayout(&fLeftColumnLayout);
836 fContainerLayout.addLayout(&fMainAndRightColumnLayout);
837
838 SkDebuggerGUI->setCentralWidget(&fCentralWidget);
839 SkDebuggerGUI->setStatusBar(&fStatusBar);
840
chudy@google.come504de02012-07-16 18:35:23 +0000841 fToolBar.setIconSize(QSize(32, 32));
chudy@google.comc432f002012-07-10 13:19:25 +0000842 fToolBar.setToolButtonStyle(Qt::ToolButtonTextUnderIcon);
843 SkDebuggerGUI->addToolBar(Qt::TopToolBarArea, &fToolBar);
chudy@google.com902ebe52012-06-29 14:21:22 +0000844
chudy@google.com0ab03392012-07-28 20:16:11 +0000845 fSpacer.setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
chudy@google.com902ebe52012-06-29 14:21:22 +0000846
chudy@google.comc432f002012-07-10 13:19:25 +0000847 fToolBar.addAction(&fActionRewind);
848 fToolBar.addAction(&fActionStepBack);
chudy@google.come504de02012-07-16 18:35:23 +0000849 fToolBar.addAction(&fActionPause);
chudy@google.comc432f002012-07-10 13:19:25 +0000850 fToolBar.addAction(&fActionStepForward);
851 fToolBar.addAction(&fActionPlay);
852 fToolBar.addSeparator();
chudy@google.come504de02012-07-16 18:35:23 +0000853 fToolBar.addAction(&fActionInspector);
chudy@google.comc432f002012-07-10 13:19:25 +0000854 fToolBar.addSeparator();
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000855 fToolBar.addAction(&fActionProfile);
856
857 fToolBar.addSeparator();
chudy@google.com0ab03392012-07-28 20:16:11 +0000858 fToolBar.addWidget(&fSpacer);
chudy@google.comc432f002012-07-10 13:19:25 +0000859 fToolBar.addWidget(&fFilter);
860 fToolBar.addAction(&fActionCancel);
chudy@google.com902ebe52012-06-29 14:21:22 +0000861
862 // TODO(chudy): Remove static call.
863 fDirectoryWidgetActive = false;
robertphillips@google.comd26c7062012-11-12 20:42:12 +0000864 fFileName = "";
robertphillips@google.comff6e6ba2013-01-28 17:43:26 +0000865 setupDirectoryWidget("");
chudy@google.com902ebe52012-06-29 14:21:22 +0000866 fDirectoryWidgetActive = true;
867
chudy@google.com902ebe52012-06-29 14:21:22 +0000868 // Menu Bar
chudy@google.comc432f002012-07-10 13:19:25 +0000869 fMenuFile.setTitle("File");
870 fMenuFile.addAction(&fActionOpen);
chudy@google.com0ab03392012-07-28 20:16:11 +0000871 fMenuFile.addAction(&fActionSave);
872 fMenuFile.addAction(&fActionSaveAs);
chudy@google.comc432f002012-07-10 13:19:25 +0000873 fMenuFile.addAction(&fActionClose);
chudy@google.come504de02012-07-16 18:35:23 +0000874
875 fMenuEdit.setTitle("Edit");
876 fMenuEdit.addAction(&fActionDelete);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000877 fMenuEdit.addAction(&fActionClearDeletes);
878 fMenuEdit.addSeparator();
chudy@google.come504de02012-07-16 18:35:23 +0000879 fMenuEdit.addAction(&fActionCreateBreakpoint);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000880 fMenuEdit.addAction(&fActionClearBreakpoints);
chudy@google.come504de02012-07-16 18:35:23 +0000881
chudy@google.comc432f002012-07-10 13:19:25 +0000882 fMenuNavigate.setTitle("Navigate");
chudy@google.come504de02012-07-16 18:35:23 +0000883 fMenuNavigate.addAction(&fActionRewind);
884 fMenuNavigate.addAction(&fActionStepBack);
885 fMenuNavigate.addAction(&fActionStepForward);
886 fMenuNavigate.addAction(&fActionPlay);
887 fMenuNavigate.addAction(&fActionPause);
chudy@google.comc432f002012-07-10 13:19:25 +0000888 fMenuNavigate.addAction(&fActionGoToLine);
chudy@google.come504de02012-07-16 18:35:23 +0000889
chudy@google.comc432f002012-07-10 13:19:25 +0000890 fMenuView.setTitle("View");
chudy@google.come504de02012-07-16 18:35:23 +0000891 fMenuView.addAction(&fActionBreakpoint);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +0000892 fMenuView.addAction(&fActionShowDeletes);
chudy@google.coma1226312012-07-26 20:26:44 +0000893 fMenuView.addAction(&fActionZoomIn);
894 fMenuView.addAction(&fActionZoomOut);
chudy@google.come504de02012-07-16 18:35:23 +0000895
896 fMenuWindows.setTitle("Window");
897 fMenuWindows.addAction(&fActionInspector);
898 fMenuWindows.addAction(&fActionDirectory);
chudy@google.comc432f002012-07-10 13:19:25 +0000899
900 fActionGoToLine.setText("Go to Line...");
901 fActionGoToLine.setDisabled(true);
902 fMenuBar.addAction(fMenuFile.menuAction());
chudy@google.come504de02012-07-16 18:35:23 +0000903 fMenuBar.addAction(fMenuEdit.menuAction());
chudy@google.comc432f002012-07-10 13:19:25 +0000904 fMenuBar.addAction(fMenuView.menuAction());
905 fMenuBar.addAction(fMenuNavigate.menuAction());
chudy@google.come504de02012-07-16 18:35:23 +0000906 fMenuBar.addAction(fMenuWindows.menuAction());
chudy@google.com902ebe52012-06-29 14:21:22 +0000907
chudy@google.com7dcae672012-07-09 20:26:53 +0000908 fPause = false;
909
chudy@google.comc432f002012-07-10 13:19:25 +0000910 SkDebuggerGUI->setMenuBar(&fMenuBar);
chudy@google.com902ebe52012-06-29 14:21:22 +0000911 QMetaObject::connectSlotsByName(SkDebuggerGUI);
912}
913
robertphillips@google.comff6e6ba2013-01-28 17:43:26 +0000914void SkDebuggerGUI::setupDirectoryWidget(const QString& path) {
915 fPath = path;
916 QDir dir(path);
chudy@google.com902ebe52012-06-29 14:21:22 +0000917 QRegExp r(".skp");
chudy@google.comc432f002012-07-10 13:19:25 +0000918 fDirectoryWidget.clear();
919 const QStringList files = dir.entryList();
chudy@google.com902ebe52012-06-29 14:21:22 +0000920 foreach (QString f, files) {
chudy@google.com7dcae672012-07-09 20:26:53 +0000921 if (f.contains(r))
chudy@google.comc432f002012-07-10 13:19:25 +0000922 fDirectoryWidget.addItem(f);
chudy@google.com902ebe52012-06-29 14:21:22 +0000923 }
924}
925
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000926// SkOffsetPicturePlayback records the offset of each command in the picture.
927// These are needed by the profiling system.
928class SkOffsetPicturePlayback : public SkPicturePlayback {
929public:
930 SkOffsetPicturePlayback(SkStream* stream, const SkPictInfo& info, bool* isValid,
931 SkSerializationHelpers::DecodeBitmap decoder)
932 : INHERITED(stream, info, isValid, decoder) {
933 }
934
935 const SkTDArray<size_t>& offsets() const { return fOffsets; }
936
937protected:
938 SkTDArray<size_t> fOffsets;
939
robertphillips@google.comca47aae2012-12-12 15:58:25 +0000940 virtual size_t preDraw(size_t offset, int type) {
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000941 *fOffsets.append() = offset;
robertphillips@google.com5f971142012-12-07 20:48:56 +0000942 return 0;
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000943 }
944
945private:
946 typedef SkPicturePlayback INHERITED;
947};
948
949// Picture to wrap an SkOffsetPicturePlayback.
950class SkOffsetPicture : public SkPicture {
951public:
skia.committer@gmail.com884e60b2012-11-16 02:01:17 +0000952 SkOffsetPicture(SkStream* stream,
953 bool* success,
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000954 SkSerializationHelpers::DecodeBitmap decoder) {
955 if (success) {
956 *success = false;
957 }
958 fRecord = NULL;
959 fPlayback = NULL;
960 fWidth = fHeight = 0;
961
962 SkPictInfo info;
963
964 if (!stream->read(&info, sizeof(info))) {
965 return;
966 }
967 if (PICTURE_VERSION != info.fVersion) {
968 return;
969 }
970
971 if (stream->readBool()) {
972 bool isValid = false;
973 fPlayback = SkNEW_ARGS(SkOffsetPicturePlayback, (stream, info, &isValid, decoder));
974 if (!isValid) {
975 SkDELETE(fPlayback);
976 fPlayback = NULL;
977 return;
978 }
979 }
980
981 // do this at the end, so that they will be zero if we hit an error.
982 fWidth = info.fWidth;
983 fHeight = info.fHeight;
984 if (success) {
985 *success = true;
986 }
987 }
988
skia.committer@gmail.com884e60b2012-11-16 02:01:17 +0000989 const SkTDArray<size_t>& offsets() const {
990 return ((SkOffsetPicturePlayback*) fPlayback)->offsets();
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000991 }
992
993private:
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +0000994 // disallow default ctor b.c. we don't have a good way to setup the fPlayback ptr
995 SkOffsetPicture();
996 // disallow the copy ctor - enabling would require copying code from SkPicture
997 SkOffsetPicture(const SkOffsetPicture& src);
998
robertphillips@google.com2bde91d2012-11-15 14:57:57 +0000999 typedef SkPicture INHERITED;
1000};
1001
1002
1003
robertphillips@google.comd26c7062012-11-12 20:42:12 +00001004void SkDebuggerGUI::loadPicture(const SkString& fileName) {
1005 fFileName = fileName;
chudy@google.comd3058f52012-07-19 13:41:27 +00001006 fLoading = true;
robertphillips@google.comd26c7062012-11-12 20:42:12 +00001007 SkStream* stream = SkNEW_ARGS(SkFILEStream, (fileName.c_str()));
robertphillips@google.com2d40ec42013-02-07 20:39:40 +00001008
1009 bool success = false;
1010
1011 SkOffsetPicture* picture = SkNEW_ARGS(SkOffsetPicture, (stream, &success, &SkImageDecoder::DecodeStream));
1012
1013 if (!success) {
1014 QMessageBox::critical(this, "Error loading file", "Couldn't read file, sorry.");
1015 SkSafeUnref(stream);
1016 return;
1017 }
robertphillips@google.com2bde91d2012-11-15 14:57:57 +00001018
chudy@google.com686e6802012-08-14 16:00:32 +00001019 fCanvasWidget.resetWidgetTransform();
chudy@google.com607357f2012-08-07 16:12:23 +00001020 fDebugger.loadPicture(picture);
chudy@google.com4c7962e2012-08-14 19:38:31 +00001021
robertphillips@google.com2bde91d2012-11-15 14:57:57 +00001022 fOffsets = picture->offsets();
1023
robertphillips@google.com5f971142012-12-07 20:48:56 +00001024 fSkipCommands.setCount(fOffsets.count());
1025 for (int i = 0; i < fOffsets.count(); ++i) {
1026 fSkipCommands[i] = false;
1027 }
1028
chudy@google.com607357f2012-08-07 16:12:23 +00001029 SkSafeUnref(stream);
1030 SkSafeUnref(picture);
1031
chudy@google.com97cee972012-08-07 20:41:37 +00001032 // Will this automatically clear out due to nature of refcnt?
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001033 SkTArray<SkString>* commands = fDebugger.getDrawCommandsAsStrings();
chudy@google.com607357f2012-08-07 16:12:23 +00001034
robertphillips@google.comfe830a42012-11-15 16:33:31 +00001035 // If SkPicturePlayback is compiled w/o SK_PICTURE_PROFILING_STUBS
1036 // the offset count will always be zero
1037 SkASSERT(0 == fOffsets.count() || commands->count() == fOffsets.count());
robertphillips@google.come099bc42012-11-19 16:26:40 +00001038 if (commands->count() == fOffsets.count()) {
1039 fActionProfile.setDisabled(false);
robertphillips@google.comfe830a42012-11-15 16:33:31 +00001040 }
robertphillips@google.com2bde91d2012-11-15 14:57:57 +00001041
chudy@google.com7dcae672012-07-09 20:26:53 +00001042 /* fDebugCanvas is reinitialized every load picture. Need it to retain value
chudy@google.com607357f2012-08-07 16:12:23 +00001043 * of the visibility filter.
1044 * TODO(chudy): This should be deprecated since fDebugger is not
1045 * recreated.
1046 * */
1047 fDebugger.highlightCurrentCommand(fSettingsWidget.getVisibilityButton()->isChecked());
1048
chudy@google.com97cee972012-08-07 20:41:37 +00001049 setupListWidget(commands);
1050 setupComboBox(commands);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001051 setupOverviewText(NULL, 0.0);
chudy@google.com7e4cfbf2012-07-17 15:40:51 +00001052 fInspectorWidget.setDisabled(false);
chudy@google.come606d6e2012-07-12 14:31:25 +00001053 fSettingsWidget.setDisabled(false);
chudy@google.comd3058f52012-07-19 13:41:27 +00001054 fMenuEdit.setDisabled(false);
1055 fMenuNavigate.setDisabled(false);
1056 fMenuView.setDisabled(false);
chudy@google.com0ab03392012-07-28 20:16:11 +00001057 fActionSave.setDisabled(false);
1058 fActionSaveAs.setDisabled(false);
chudy@google.comd3058f52012-07-19 13:41:27 +00001059 fLoading = false;
1060 actionPlay();
chudy@google.com902ebe52012-06-29 14:21:22 +00001061}
1062
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001063void SkDebuggerGUI::setupListWidget(SkTArray<SkString>* command) {
chudy@google.comc432f002012-07-10 13:19:25 +00001064 fListWidget.clear();
chudy@google.com902ebe52012-06-29 14:21:22 +00001065 int counter = 0;
robertphillips@google.com30d35f22012-11-06 16:45:36 +00001066 int indent = 0;
chudy@google.com97cee972012-08-07 20:41:37 +00001067 for (int i = 0; i < command->count(); i++) {
chudy@google.com902ebe52012-06-29 14:21:22 +00001068 QListWidgetItem *item = new QListWidgetItem();
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001069 item->setData(Qt::DisplayRole, (*command)[i].c_str());
chudy@google.com902ebe52012-06-29 14:21:22 +00001070 item->setData(Qt::UserRole + 1, counter++);
robertphillips@google.com30d35f22012-11-06 16:45:36 +00001071
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001072 if (0 == strcmp("Restore", (*command)[i].c_str())) {
robertphillips@google.com30d35f22012-11-06 16:45:36 +00001073 indent -= 10;
1074 }
1075
1076 item->setData(Qt::UserRole + 3, indent);
1077
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001078 if (0 == strcmp("Save", (*command)[i].c_str()) ||
1079 0 == strcmp("Save Layer", (*command)[i].c_str())) {
robertphillips@google.com30d35f22012-11-06 16:45:36 +00001080 indent += 10;
1081 }
1082
robertphillips@google.comd26c7062012-11-12 20:42:12 +00001083 item->setData(Qt::UserRole + 4, -1.0);
1084
chudy@google.comc432f002012-07-10 13:19:25 +00001085 fListWidget.addItem(item);
chudy@google.com902ebe52012-06-29 14:21:22 +00001086 }
1087}
1088
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001089void SkDebuggerGUI::setupOverviewText(const SkTDArray<double>* typeTimes, double totTime) {
chudy@google.com902ebe52012-06-29 14:21:22 +00001090
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001091 const SkTDArray<SkDrawCommand*>& commands = fDebugger.getDrawCommands();
1092
1093 SkTDArray<int> counts;
1094 counts.setCount(LAST_DRAWTYPE_ENUM+1);
1095 for (int i = 0; i < LAST_DRAWTYPE_ENUM+1; ++i) {
1096 counts[i] = 0;
1097 }
1098
1099 for (int i = 0; i < commands.count(); i++) {
1100 counts[commands[i]->getType()]++;
chudy@google.com902ebe52012-06-29 14:21:22 +00001101 }
1102
1103 QString overview;
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001104 int total = 0;
1105#ifdef SK_DEBUG
1106 double totPercent = 0, tempSum = 0;
1107#endif
1108 for (int i = 0; i < LAST_DRAWTYPE_ENUM+1; ++i) {
1109 if (0 == counts[i]) {
1110 // if there were no commands of this type then they should've consumed no time
1111 SkASSERT(NULL == typeTimes || 0.0 == (*typeTimes)[i]);
1112 continue;
1113 }
1114
1115 overview.append(SkDrawCommand::GetCommandString((DrawType) i));
chudy@google.com902ebe52012-06-29 14:21:22 +00001116 overview.append(": ");
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001117 overview.append(QString::number(counts[i]));
1118 if (NULL != typeTimes) {
1119 overview.append(" - ");
1120 overview.append(QString::number((*typeTimes)[i], 'f', 1));
1121 overview.append("ms");
1122 overview.append(" - ");
1123 double percent = 100.0*(*typeTimes)[i]/totTime;
1124 overview.append(QString::number(percent, 'f', 1));
1125 overview.append("%");
1126#ifdef SK_DEBUG
1127 totPercent += percent;
1128 tempSum += (*typeTimes)[i];
1129#endif
1130 }
chudy@google.com902ebe52012-06-29 14:21:22 +00001131 overview.append("<br/>");
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001132 total += counts[i];
chudy@google.com902ebe52012-06-29 14:21:22 +00001133 }
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001134#ifdef SK_DEBUG
1135 if (NULL != typeTimes) {
1136 SkASSERT(SkScalarNearlyEqual(totPercent, 100.0));
1137 SkASSERT(SkScalarNearlyEqual(tempSum, totTime));
1138 }
1139#endif
1140
1141 if (totTime > 0.0) {
1142 overview.append("Total Time: ");
1143 overview.append(QString::number(totTime, 'f', 2));
1144 overview.append("ms");
1145#ifdef SK_DEBUG
1146 overview.append(" ");
1147 overview.append(QString::number(totPercent));
1148 overview.append("% ");
1149#endif
1150 overview.append("<br/>");
1151 }
1152
1153 QString totalStr;
1154 totalStr.append("Total Draw Commands: ");
1155 totalStr.append(QString::number(total));
1156 totalStr.append("<br/>");
1157 overview.insert(0, totalStr);
chudy@google.com902ebe52012-06-29 14:21:22 +00001158
1159 overview.append("<br/>");
chudy@google.com607357f2012-08-07 16:12:23 +00001160 overview.append("SkPicture Width: ");
chudy@google.com902ebe52012-06-29 14:21:22 +00001161 // NOTE(chudy): This is where we can pull out the SkPictures width.
chudy@google.com607357f2012-08-07 16:12:23 +00001162 overview.append(QString::number(fDebugger.pictureWidth()));
chudy@google.com902ebe52012-06-29 14:21:22 +00001163 overview.append("px<br/>");
chudy@google.com607357f2012-08-07 16:12:23 +00001164 overview.append("SkPicture Height: ");
1165 overview.append(QString::number(fDebugger.pictureHeight()));
chudy@google.com902ebe52012-06-29 14:21:22 +00001166 overview.append("px");
chudy@google.com6bd109a2012-08-14 19:34:13 +00001167 fInspectorWidget.setText(overview, SkInspectorWidget::kOverview_TabType);
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001168}
1169
1170void SkDebuggerGUI::setupComboBox(SkTArray<SkString>* command) {
1171 fFilter.clear();
1172 fFilter.addItem("--Filter By Available Commands--");
1173
1174 std::map<std::string, int> map;
1175 for (int i = 0; i < command->count(); i++) {
1176 map[(*command)[i].c_str()]++;
1177 }
1178
skia.committer@gmail.com34587162012-11-20 02:01:23 +00001179 for (std::map<std::string, int>::iterator it = map.begin(); it != map.end();
robertphillips@google.com8a1cdae2012-11-19 20:44:29 +00001180 ++it) {
1181 fFilter.addItem((it->first).c_str());
1182 }
chudy@google.com902ebe52012-06-29 14:21:22 +00001183
1184 // NOTE(chudy): Makes first item unselectable.
chudy@google.com7dcae672012-07-09 20:26:53 +00001185 QStandardItemModel* model = qobject_cast<QStandardItemModel*>(
chudy@google.comc432f002012-07-10 13:19:25 +00001186 fFilter.model());
1187 QModelIndex firstIndex = model->index(0, fFilter.modelColumn(),
1188 fFilter.rootModelIndex());
chudy@google.com902ebe52012-06-29 14:21:22 +00001189 QStandardItem* firstItem = model->itemFromIndex(firstIndex);
1190 firstItem->setSelectable(false);
1191}