blob: 1c309b2095bc8aff24f6d3573ca078ae9510e23e [file] [log] [blame]
/*
* Copyright 2016 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "src/core/SkRecordedDrawable.h"
#include "include/core/SkPicture.h"
#include "include/core/SkPictureRecorder.h"
#include "include/core/SkSize.h"
#include "src/core/SkBigPicture.h"
#include "src/core/SkPictureData.h"
#include "src/core/SkPicturePlayback.h"
#include "src/core/SkPictureRecord.h"
#include "src/core/SkReadBuffer.h"
#include "src/core/SkRecordDraw.h"
#include "src/core/SkWriteBuffer.h"
class SkCanvas;
size_t SkRecordedDrawable::onApproximateBytesUsed() {
size_t drawablesSize = 0;
if (fDrawableList) {
for (auto&& drawable : *fDrawableList) {
drawablesSize += drawable->approximateBytesUsed();
}
}
return sizeof(*this) +
(fRecord ? fRecord->bytesUsed() : 0) +
(fBBH ? fBBH->bytesUsed() : 0) +
drawablesSize;
}
void SkRecordedDrawable::onDraw(SkCanvas* canvas) {
SkDrawable* const* drawables = nullptr;
int drawableCount = 0;
if (fDrawableList) {
drawables = fDrawableList->begin();
drawableCount = fDrawableList->count();
}
SkRecordDraw(*fRecord, canvas, nullptr, drawables, drawableCount, fBBH.get(), nullptr);
}
sk_sp<SkPicture> SkRecordedDrawable::onMakePictureSnapshot() {
// TODO: should we plumb-down the BBHFactory and recordFlags from our host
// PictureRecorder?
std::unique_ptr<SkBigPicture::SnapshotArray> pictList{
fDrawableList ? fDrawableList->newDrawableSnapshot() : nullptr
};
size_t subPictureBytes = 0;
for (int i = 0; pictList && i < pictList->count(); i++) {
subPictureBytes += pictList->begin()[i]->approximateBytesUsed();
}
return sk_make_sp<SkBigPicture>(fBounds, fRecord, std::move(pictList), fBBH, subPictureBytes);
}
void SkRecordedDrawable::flatten(SkWriteBuffer& buffer) const {
// Write the bounds.
buffer.writeRect(fBounds);
// Create an SkPictureRecord to record the draw commands.
SkPictInfo info;
SkPictureRecord pictureRecord(SkISize::Make(fBounds.width(), fBounds.height()), 0);
// If the query contains the whole picture, don't bother with the bounding box hierarchy.
SkBBoxHierarchy* bbh;
if (pictureRecord.getLocalClipBounds().contains(fBounds)) {
bbh = nullptr;
} else {
bbh = fBBH.get();
}
// Record the draw commands.
SkDrawable* const* drawables = fDrawableList ? fDrawableList->begin() : nullptr;
int drawableCount = fDrawableList ? fDrawableList->count() : 0;
pictureRecord.beginRecording();
SkRecordDraw(*fRecord, &pictureRecord, nullptr, drawables, drawableCount, bbh, nullptr);
pictureRecord.endRecording();
// Flatten the recorded commands and drawables.
SkPictureData pictureData(pictureRecord, info);
pictureData.flatten(buffer);
}
sk_sp<SkFlattenable> SkRecordedDrawable::CreateProc(SkReadBuffer& buffer) {
// Read the bounds.
SkRect bounds;
buffer.readRect(&bounds);
// Unflatten into a SkPictureData.
SkPictInfo info;
info.setVersion(buffer.getVersion());
info.fCullRect = bounds;
std::unique_ptr<SkPictureData> pictureData(SkPictureData::CreateFromBuffer(buffer, info));
if (!pictureData) {
return nullptr;
}
// Create a drawable.
SkPicturePlayback playback(pictureData.get());
SkPictureRecorder recorder;
playback.draw(recorder.beginRecording(bounds), nullptr, &buffer);
return recorder.finishRecordingAsDrawable();
}