| /* | 
 |  * Copyright 2014 Google Inc. | 
 |  * | 
 |  * Use of this source code is governed by a BSD-style license that can be | 
 |  * found in the LICENSE file. | 
 |  */ | 
 |  | 
 | #include <stdio.h> | 
 |  | 
 | #include "SkPicturePriv.h" | 
 | #include "SkRecord.h" | 
 | #include "SkRecordDraw.h" | 
 |  | 
 | #include "DumpRecord.h" | 
 | #include "SkTime.h" | 
 |  | 
 | namespace { | 
 |  | 
 | class Dumper { | 
 | public: | 
 |     explicit Dumper(SkCanvas* canvas, int count, bool timeWithCommand) | 
 |         : fDigits(0) | 
 |         , fIndent(0) | 
 |         , fIndex(0) | 
 |         , fDraw(canvas, nullptr, nullptr, 0, nullptr) | 
 |         , fTimeWithCommand(timeWithCommand) { | 
 |         while (count > 0) { | 
 |             count /= 10; | 
 |             fDigits++; | 
 |         } | 
 |     } | 
 |  | 
 |     template <typename T> | 
 |     void operator()(const T& command) { | 
 |         auto start = SkTime::GetNSecs(); | 
 |         fDraw(command); | 
 |         this->print(command, SkTime::GetNSecs() - start); | 
 |     } | 
 |  | 
 |     void operator()(const SkRecords::NoOp&) { | 
 |         // Move on without printing anything. | 
 |     } | 
 |  | 
 |     template <typename T> | 
 |     void print(const T& command, double ns) { | 
 |         this->printNameAndTime(command, ns); | 
 |     } | 
 |  | 
 |     void print(const SkRecords::Restore& command, double ns) { | 
 |         --fIndent; | 
 |         this->printNameAndTime(command, ns); | 
 |     } | 
 |  | 
 |     void print(const SkRecords::Save& command, double ns) { | 
 |         this->printNameAndTime(command, ns); | 
 |         ++fIndent; | 
 |     } | 
 |  | 
 |     void print(const SkRecords::SaveLayer& command, double ns) { | 
 |         this->printNameAndTime(command, ns); | 
 |         ++fIndent; | 
 |     } | 
 |  | 
 |     void print(const SkRecords::DrawPicture& command, double ns) { | 
 |         this->printNameAndTime(command, ns); | 
 |  | 
 |         if (auto bp = SkPicturePriv::AsSkBigPicture(command.picture)) { | 
 |             ++fIndent; | 
 |  | 
 |             const SkRecord& record = *bp->record(); | 
 |             for (int i = 0; i < record.count(); i++) { | 
 |                 record.visit(i, *this); | 
 |             } | 
 |  | 
 |             --fIndent; | 
 |         } | 
 |     } | 
 |  | 
 | #if 1 | 
 |     void print(const SkRecords::DrawAnnotation& command, double ns) { | 
 |         int us = (int)(ns * 1e-3); | 
 |         if (!fTimeWithCommand) { | 
 |             printf("%6dus  ", us); | 
 |         } | 
 |         printf("%*d ", fDigits, fIndex++); | 
 |         for (int i = 0; i < fIndent; i++) { | 
 |             printf("    "); | 
 |         } | 
 |         if (fTimeWithCommand) { | 
 |             printf("%6dus  ", us); | 
 |         } | 
 |         printf("DrawAnnotation [%g %g %g %g] %s\n", | 
 |                command.rect.left(), command.rect.top(), command.rect.right(), command.rect.bottom(), | 
 |                command.key.c_str()); | 
 |     } | 
 | #endif | 
 |  | 
 | private: | 
 |     template <typename T> | 
 |     void printNameAndTime(const T& command, double ns) { | 
 |         int us = (int)(ns * 1e-3); | 
 |         if (!fTimeWithCommand) { | 
 |             printf("%6dus  ", us); | 
 |         } | 
 |         printf("%*d ", fDigits, fIndex++); | 
 |         for (int i = 0; i < fIndent; i++) { | 
 |             printf("    "); | 
 |         } | 
 |         if (fTimeWithCommand) { | 
 |             printf("%6dus  ", us); | 
 |         } | 
 |         puts(NameOf(command)); | 
 |     } | 
 |  | 
 |     template <typename T> | 
 |     static const char* NameOf(const T&) { | 
 |     #define CASE(U) case SkRecords::U##_Type: return #U; | 
 |         switch(T::kType) { SK_RECORD_TYPES(CASE); } | 
 |     #undef CASE | 
 |         SkDEBUGFAIL("Unknown T"); | 
 |         return "Unknown T"; | 
 |     } | 
 |  | 
 |     static const char* NameOf(const SkRecords::SaveLayer&) { | 
 |         return "\x1b[31;1mSaveLayer\x1b[0m";  // Bold red. | 
 |     } | 
 |  | 
 |     int fDigits; | 
 |     int fIndent; | 
 |     int fIndex; | 
 |     SkRecords::Draw fDraw; | 
 |     const bool fTimeWithCommand; | 
 | }; | 
 |  | 
 | }  // namespace | 
 |  | 
 | void DumpRecord(const SkRecord& record, | 
 |                   SkCanvas* canvas, | 
 |                   bool timeWithCommand) { | 
 |     Dumper dumper(canvas, record.count(), timeWithCommand); | 
 |     for (int i = 0; i < record.count(); i++) { | 
 |         record.visit(i, dumper); | 
 |     } | 
 | } |