blob: 308f383a6a63cf7d931ef2659f5591357f72817c [file] [log] [blame]
/*
* Copyright 2023 Google LLC
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "src/image/SkImage_Picture.h"
#include "include/core/SkCanvas.h"
#include "include/core/SkColor.h"
#include "include/core/SkColorSpace.h"
#include "include/core/SkColorType.h"
#include "include/core/SkImage.h"
#include "include/core/SkImageGenerator.h"
#include "include/core/SkImageInfo.h"
#include "include/core/SkMatrix.h"
#include "include/core/SkPicture.h"
#include "include/core/SkSurfaceProps.h"
#include "include/private/base/SkAssert.h"
#include "include/private/base/SkMutex.h"
#include "include/private/base/SkTFitsIn.h"
#include "src/base/SkTLazy.h"
#include "src/image/SkImageGeneratorPriv.h"
#include "src/image/SkImage_Lazy.h"
#include "src/image/SkPictureImageGenerator.h"
#include <cstring>
#include <memory>
#include <utility>
class SkPaint;
struct SkISize;
sk_sp<SkImage> SkImage_Picture::Make(sk_sp<SkPicture> picture, const SkISize& dimensions,
const SkMatrix* matrix, const SkPaint* paint,
SkImages::BitDepth bitDepth, sk_sp<SkColorSpace> colorSpace,
SkSurfaceProps props) {
auto gen = SkImageGenerators::MakeFromPicture(dimensions, std::move(picture), matrix, paint,
bitDepth, std::move(colorSpace), props);
SkImage_Lazy::Validator validator(
SharedGenerator::Make(std::move(gen)), nullptr, nullptr);
return validator ? sk_make_sp<SkImage_Picture>(&validator) : nullptr;
}
const SkSurfaceProps* SkImage_Picture::props() const {
auto pictureIG = static_cast<SkPictureImageGenerator*>(this->generator()->fGenerator.get());
return &pictureIG->fProps;
}
void SkImage_Picture::replay(SkCanvas* canvas) const {
auto sharedGenerator = this->generator();
SkAutoMutexExclusive mutex(sharedGenerator->fMutex);
auto pictureIG = static_cast<SkPictureImageGenerator*>(sharedGenerator->fGenerator.get());
canvas->clear(SkColors::kTransparent);
canvas->drawPicture(pictureIG->fPicture,
&pictureIG->fMatrix,
pictureIG->fPaint.getMaybeNull());
}
bool SkImage_Picture::getImageKeyValues(
uint32_t keyValues[SkTiledImageUtils::kNumImageKeyValues]) const {
auto sharedGenerator = this->generator();
SkAutoMutexExclusive mutex(sharedGenerator->fMutex);
auto pictureIG = static_cast<SkPictureImageGenerator*>(sharedGenerator->fGenerator.get());
if (pictureIG->fPaint.getMaybeNull()) {
// A full paint complicates the potential key too much.
return false;
}
const SkImageInfo& ii = sharedGenerator->getInfo();
if (!ii.colorSpace()->isSRGB()) {
// We only return key values if the colorSpace is sRGB.
return false;
}
const SkMatrix& m = pictureIG->fMatrix;
if (!m.isIdentity() && !m.isTranslate()) {
// To keep the key small we only cache simple (<= translation) matrices
return false;
}
bool isU8 = ii.colorType() != kRGBA_F16_SkColorType;
uint32_t pixelGeometry = this->props()->pixelGeometry();
uint32_t surfacePropFlags = this->props()->flags();
int width = ii.width();
int height = ii.height();
float transX = m.getTranslateX();
float transY = m.getTranslateY();
SkASSERT(pixelGeometry <= 4);
SkASSERT(surfacePropFlags < 8);
SkASSERT(SkTFitsIn<uint32_t>(width));
SkASSERT(SkTFitsIn<uint32_t>(height));
SkASSERT(sizeof(float) == sizeof(uint32_t));
// The 0th slot usually holds either the SkBitmap's ID or the image's. In those two cases
// slot #1 is zero so we can reuse the 0th slot here.
keyValues[0] = (isU8 ? 0x1 : 0x0) | // 1 bit
(pixelGeometry << 1) | // 3 bits
(surfacePropFlags << 4); // 3 bits
keyValues[1] = pictureIG->fPicture->uniqueID();
SkASSERT(keyValues[1] != 0); // Double check we don't collide w/ bitmap or image keys
keyValues[2] = width;
keyValues[3] = height;
memcpy(&keyValues[4], &transX, sizeof(uint32_t));
memcpy(&keyValues[5], &transY, sizeof(uint32_t));
return true;
}