skia2/samplecode/SampleHairline.cpp

232 lines
6.4 KiB
C++
Raw Normal View History

Automatic update of all copyright notices to reflect new license terms. I have manually examined all of these diffs and restored a few files that seem to require manual adjustment. The following files still need to be modified manually, in a separate CL: android_sample/SampleApp/AndroidManifest.xml android_sample/SampleApp/res/layout/layout.xml android_sample/SampleApp/res/menu/sample.xml android_sample/SampleApp/res/values/strings.xml android_sample/SampleApp/src/com/skia/sampleapp/SampleApp.java android_sample/SampleApp/src/com/skia/sampleapp/SampleView.java experimental/CiCarbonSampleMain.c experimental/CocoaDebugger/main.m experimental/FileReaderApp/main.m experimental/SimpleCocoaApp/main.m experimental/iOSSampleApp/Shared/SkAlertPrompt.h experimental/iOSSampleApp/Shared/SkAlertPrompt.m experimental/iOSSampleApp/SkiOSSampleApp-Base.xcconfig experimental/iOSSampleApp/SkiOSSampleApp-Debug.xcconfig experimental/iOSSampleApp/SkiOSSampleApp-Release.xcconfig gpu/src/android/GrGLDefaultInterface_android.cpp gyp/common.gypi gyp_skia include/ports/SkHarfBuzzFont.h include/views/SkOSWindow_wxwidgets.h make.bat make.py src/opts/memset.arm.S src/opts/memset16_neon.S src/opts/memset32_neon.S src/opts/opts_check_arm.cpp src/ports/SkDebug_brew.cpp src/ports/SkMemory_brew.cpp src/ports/SkOSFile_brew.cpp src/ports/SkXMLParser_empty.cpp src/utils/ios/SkImageDecoder_iOS.mm src/utils/ios/SkOSFile_iOS.mm src/utils/ios/SkStream_NSData.mm tests/FillPathTest.cpp Review URL: http://codereview.appspot.com/4816058 git-svn-id: http://skia.googlecode.com/svn/trunk@1982 2bbb7eff-a529-9590-31e7-b0007b416f81
2011-07-28 14:26:00 +00:00
/*
* Copyright 2011 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#include "include/core/SkBitmap.h"
#include "include/core/SkCanvas.h"
#include "include/core/SkColorFilter.h"
#include "include/core/SkColorPriv.h"
#include "include/core/SkGraphics.h"
#include "include/core/SkPath.h"
#include "include/core/SkRegion.h"
#include "include/core/SkShader.h"
#include "include/core/SkStream.h"
#include "include/core/SkTime.h"
#include "include/core/SkTypeface.h"
#include "include/effects/SkCornerPathEffect.h"
#include "include/effects/SkGradientShader.h"
#include "include/private/SkTo.h"
#include "include/utils/SkRandom.h"
#include "samplecode/Sample.h"
#include "src/utils/SkUTF.h"
static SkRandom gRand;
static void generate_pts(SkPoint pts[], int count, int w, int h) {
for (int i = 0; i < count; i++) {
pts[i].set(gRand.nextUScalar1() * 3 * w - SkIntToScalar(w),
gRand.nextUScalar1() * 3 * h - SkIntToScalar(h));
}
}
static bool check_zeros(const SkPMColor pixels[], int count, int skip) {
for (int i = 0; i < count; i++) {
if (*pixels) {
return false;
}
pixels += skip;
}
return true;
}
static bool check_bitmap_margin(const SkBitmap& bm, int margin) {
size_t rb = bm.rowBytes();
for (int i = 0; i < margin; i++) {
if (!check_zeros(bm.getAddr32(0, i), bm.width(), 1)) {
return false;
}
int bottom = bm.height() - i - 1;
if (!check_zeros(bm.getAddr32(0, bottom), bm.width(), 1)) {
return false;
}
// left column
if (!check_zeros(bm.getAddr32(i, 0), bm.height(), SkToInt(rb >> 2))) {
return false;
}
int right = bm.width() - margin + i;
if (!check_zeros(bm.getAddr32(right, 0), bm.height(),
SkToInt(rb >> 2))) {
return false;
}
}
return true;
}
#define WIDTH 620
#define HEIGHT 460
#define MARGIN 10
static void line_proc(SkCanvas* canvas, const SkPaint& paint,
const SkBitmap& bm) {
const int N = 2;
SkPoint pts[N];
for (int i = 0; i < 400; i++) {
generate_pts(pts, N, WIDTH, HEIGHT);
canvas->drawLine(pts[0], pts[1], paint);
if (!check_bitmap_margin(bm, MARGIN)) {
SkDebugf("---- hairline failure (%g %g) (%g %g)\n",
pts[0].fX, pts[0].fY, pts[1].fX, pts[1].fY);
break;
}
}
}
static void poly_proc(SkCanvas* canvas, const SkPaint& paint,
const SkBitmap& bm) {
const int N = 8;
SkPoint pts[N];
for (int i = 0; i < 50; i++) {
generate_pts(pts, N, WIDTH, HEIGHT);
SkPath path;
path.moveTo(pts[0]);
for (int j = 1; j < N; j++) {
path.lineTo(pts[j]);
}
canvas->drawPath(path, paint);
}
}
static SkPoint ave(const SkPoint& a, const SkPoint& b) {
SkPoint c = a + b;
c.fX = SkScalarHalf(c.fX);
c.fY = SkScalarHalf(c.fY);
return c;
}
static void quad_proc(SkCanvas* canvas, const SkPaint& paint,
const SkBitmap& bm) {
const int N = 30;
SkPoint pts[N];
for (int i = 0; i < 10; i++) {
generate_pts(pts, N, WIDTH, HEIGHT);
SkPath path;
path.moveTo(pts[0]);
for (int j = 1; j < N - 2; j++) {
path.quadTo(pts[j], ave(pts[j], pts[j+1]));
}
path.quadTo(pts[N - 2], pts[N - 1]);
canvas->drawPath(path, paint);
}
}
static void add_cubic(SkPath* path, const SkPoint& mid, const SkPoint& end) {
SkPoint start;
path->getLastPt(&start);
path->cubicTo(ave(start, mid), ave(mid, end), end);
}
static void cube_proc(SkCanvas* canvas, const SkPaint& paint,
const SkBitmap& bm) {
const int N = 30;
SkPoint pts[N];
for (int i = 0; i < 10; i++) {
generate_pts(pts, N, WIDTH, HEIGHT);
SkPath path;
path.moveTo(pts[0]);
for (int j = 1; j < N - 2; j++) {
add_cubic(&path, pts[j], ave(pts[j], pts[j+1]));
}
add_cubic(&path, pts[N - 2], pts[N - 1]);
canvas->drawPath(path, paint);
}
}
typedef void (*HairProc)(SkCanvas*, const SkPaint&, const SkBitmap&);
static const struct {
const char* fName;
HairProc fProc;
} gProcs[] = {
{ "line", line_proc },
{ "poly", poly_proc },
{ "quad", quad_proc },
{ "cube", cube_proc },
};
static int cycle_hairproc_index(int index) {
return (index + 1) % SK_ARRAY_COUNT(gProcs);
}
class HairlineView : public Sample {
SkMSec fNow;
int fProcIndex;
bool fDoAA;
public:
HairlineView() {
fProcIndex = 0;
fDoAA = true;
fNow = 0;
}
protected:
SkString name() override { return SkStringPrintf("Hair-%s", gProcs[fProcIndex].fName); }
void show_bitmaps(SkCanvas* canvas, const SkBitmap& b0, const SkBitmap& b1,
const SkIRect& inset) {
canvas->drawBitmap(b0, 0, 0, nullptr);
canvas->drawBitmap(b1, SkIntToScalar(b0.width()), 0, nullptr);
}
void onDrawContent(SkCanvas* canvas) override {
gRand.setSeed(fNow);
SkBitmap bm, bm2;
bm.allocN32Pixels(WIDTH + MARGIN*2, HEIGHT + MARGIN*2);
// this will erase our margin, which we want to always stay 0
bm.eraseColor(SK_ColorTRANSPARENT);
bm2.installPixels(SkImageInfo::MakeN32Premul(WIDTH, HEIGHT),
bm.getAddr32(MARGIN, MARGIN), bm.rowBytes());
SkCanvas c2(bm2);
SkPaint paint;
paint.setAntiAlias(fDoAA);
paint.setStyle(SkPaint::kStroke_Style);
bm2.eraseColor(SK_ColorTRANSPARENT);
gProcs[fProcIndex].fProc(&c2, paint, bm);
canvas->drawBitmap(bm2, SkIntToScalar(10), SkIntToScalar(10), nullptr);
}
bool onAnimate(double /*nanos*/) override {
if (fDoAA) {
fProcIndex = cycle_hairproc_index(fProcIndex);
// todo: signal that we want to rebuild our TITLE
}
fDoAA = !fDoAA;
return true;
}
Sample::Click* onFindClickHandler(SkScalar x, SkScalar y, skui::ModifierKey modi) override {
fDoAA = !fDoAA;
return this->INHERITED::onFindClickHandler(x, y, modi);
}
private:
using INHERITED = Sample;
};
//////////////////////////////////////////////////////////////////////////////
DEF_SAMPLE( return new HairlineView(); )