skia2/gm/orientation.cpp
Brian Salomon 87d42e5d12 A new way to specify YUVA planar data from SkCodec to SkImage_Lazy
Tunnels through SkImageGenerator as well.

The new SkCodec interface doesn't assume three 8 bit planes.

New SkYUVASpec more clearly defines chroma subsampling and siting of
the planes.

The intent is to use this for other YUVA APIs as well, in particular
SkImage factories in the future.

In this change we convert to the SkYUVASpec to SkYUVASizeInfo
and SkYUVAIndex[4] representation. But the intent is to use
the SkYUVASpec representation throughout the pipeline once
legacy APIs are removed.

orientation GM is replicated to test a variety of chroma
subsampling configs.

Bug: skia:10632

Change-Id: I3fad35752b87cac16c51b24824331f2ae7d458d3
Reviewed-on: https://skia-review.googlesource.com/c/skia/+/309658
Commit-Queue: Brian Salomon <bsalomon@google.com>
Reviewed-by: Robert Phillips <robertphillips@google.com>
Reviewed-by: Leon Scroggins <scroggo@google.com>
2020-08-24 14:25:32 +00:00

174 lines
5.2 KiB
C++

/*
* Copyright 2018 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "gm/gm.h"
#include "include/core/SkCanvas.h"
#include "include/core/SkFont.h"
#include "include/core/SkImage.h"
#include "include/core/SkMaskFilter.h"
#include "include/core/SkRefCnt.h"
#include "include/core/SkSize.h"
#include "include/core/SkString.h"
#include "include/core/SkSurface.h"
#include "tools/Resources.h"
#include "tools/ToolUtils.h"
static constexpr int kImgW = 100;
static constexpr int kImgH = 80;
/**
This function was used to create the images used by these test. It saves them as PNGs (so they
are lossless). Then the following bash script was used to create the oriented JPGs with
imagemagick:
#!/bin/bash
function originname() {
case $1 in
1)
echo -n "top-left"
;;
2)
echo -n "top-right"
;;
3)
echo -n "bottom-right"
;;
4)
echo -n "bottom-left"
;;
5)
echo -n "left-top"
;;
6)
echo -n "right-top"
;;
7)
echo -n "right-bottom"
;;
8)
echo -n "left-bottom"
;;
*)
echo -n "unknown"
;;
esac
}
for s in 444 422 420 440 411 410; do
for i in {1..8}; do
magick convert $i.png -sampling-factor ${s:0:1}:${s:1:1}:${s:2:1} \
-orient $(originname $i) $i\_$s.jpg;
done
done
*/
static void make_images() {
auto surf = SkSurface::MakeRaster(SkImageInfo::Make({kImgW, kImgH},
kRGBA_8888_SkColorType,
kPremul_SkAlphaType));
for (int i = 1; i <= 8; ++i) {
auto* canvas = surf->getCanvas();
canvas->clear(SK_ColorBLACK);
SkPaint paint;
paint.setColor(SK_ColorRED);
SkScalar midX = kImgW / 2.f;
SkScalar midY = kImgH / 2.f;
SkScalar w = midX - 1;
SkScalar h = midY - 1;
canvas->drawRect(SkRect::MakeXYWH(1, 1, w, h), paint);
paint.setColor(SK_ColorBLUE);
canvas->drawRect(SkRect::MakeXYWH(midX, 1, w, h), paint);
paint.setColor(SK_ColorGREEN);
canvas->drawRect(SkRect::MakeXYWH(1, midY, w, h), paint);
paint.setColor(SK_ColorYELLOW);
canvas->drawRect(SkRect::MakeXYWH(midX, midY, w, h), paint);
SkFont font(ToolUtils::create_portable_typeface(), kImgH / 4.f);
SkPaint blurPaint;
blurPaint.setMaskFilter(SkMaskFilter::MakeBlur(kNormal_SkBlurStyle, .75f));
blurPaint.setColor(SK_ColorBLACK);
paint.setColor(SK_ColorWHITE);
auto drawLabel = [&](const char* string, SkScalar x, SkScalar y) {
canvas->save();
canvas->translate(1, 1);
canvas->drawString(string, x, y, font, blurPaint);
canvas->restore();
canvas->drawString(string, x, y, font, paint);
};
auto measure = [&font](const char* text) {
SkRect bounds;
font.measureText(text, strlen(text), SkTextEncoding::kUTF8, &bounds);
return bounds;
};
static constexpr SkScalar kPad = 3.f;
SkRect bounds;
bounds = measure("top");
drawLabel("top", midX - bounds.centerX(), -bounds.top() + kPad);
bounds = measure("bottom");
drawLabel("bottom", midX - bounds.centerX(), kImgH - kPad - bounds.bottom());
// It looks weird if "left" and "right" and the number at the center aren't vertically
// aligned.
SkScalar baseY = midY - measure("leftright").centerY();
bounds = measure("left");
drawLabel("left", kPad - bounds.left(), baseY);
bounds = measure("right");
drawLabel("right", midX - kPad - bounds.right(), baseY);
SkString num = SkStringPrintf("%d", i);
bounds = measure(num.c_str());
drawLabel(num.c_str(), midX - bounds.centerX(), baseY);
num.append(".png");
SkPixmap pm;
surf->makeImageSnapshot()->peekPixels(&pm);
ToolUtils::EncodeImageToFile(num.c_str(), pm, SkEncodedImageFormat::kPNG, 100);
}
}
// This gm draws 8 images that are mostly the same when respecting the
// EXIF orientation tag. Each one has four quadrants (red, blue, green,
// yellow), and labels on the left, top, right and bottom. The only
// visual difference is a number in the middle corresponding to the
// EXIF tag for that image's jpg file.
static void draw(SkCanvas* canvas, const char* suffix) {
// Avoid unused function warning.
if (0) {
make_images();
}
canvas->save();
for (char i = '1'; i <= '8'; i++) {
SkString path = SkStringPrintf("images/orientation/%c%s.jpg", i, suffix);
auto image = GetResourceAsImage(path.c_str());
if (!image) {
continue;
}
canvas->drawImage(image, 0, 0);
if ('4' == i) {
canvas->restore();
canvas->translate(0, image->height());
} else {
canvas->translate(image->width(), 0);
}
}
}
#define MAKE_GM(subsample) DEF_SIMPLE_GM(orientation_##subsample, canvas, 4*kImgW, 2*kImgH) { \
draw(canvas, "_" #subsample); \
}
MAKE_GM(410)
MAKE_GM(411)
MAKE_GM(420)
MAKE_GM(422)
MAKE_GM(440)
MAKE_GM(444)