/*
* 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 "SampleCode.h"
#include "SkView.h"
#include "SkCanvas.h"
#include "SkGradientShader.h"
#include "SkGraphics.h"
#include "SkImageDecoder.h"
#include "SkImageEncoder.h"
#include "SkPath.h"
#include "SkRegion.h"
#include "SkShader.h"
#include "SkUtils.h"
#include "SkXfermode.h"
#include "SkColorPriv.h"
#include "SkColorFilter.h"
#include "SkTime.h"
#include "SkTypeface.h"
#include "SkStream.h"
static void make_image(SkBitmap* bm, SkBitmap::Config config, int configIndex) {
const int width = 98;
const int height = 100;
SkBitmap device;
device.setConfig(SkBitmap::kARGB_8888_Config, width, height);
device.allocPixels();
SkCanvas canvas(device);
SkPaint paint;
paint.setAntiAlias(true);
canvas.drawColor(SK_ColorRED);
paint.setColor(SK_ColorBLUE);
canvas.drawCircle(SkIntToScalar(width)/2, SkIntToScalar(height)/2,
SkIntToScalar(width)/2, paint);
bm->setConfig(config, width, height);
switch (config) {
case SkBitmap::kARGB_8888_Config:
bm->swap(device);
break;
case SkBitmap::kRGB_565_Config: {
bm->allocPixels();
for (int y = 0; y < height; y++) {
for (int x = 0; x < width; x++) {
*bm->getAddr16(x, y) = SkPixel32ToPixel16(*device.getAddr32(x, y));
}
}
break;
}
case SkBitmap::kIndex8_Config: {
SkPMColor colors[256];
for (int i = 0; i < 256; i++) {
if (configIndex & 1) {
colors[i] = SkPackARGB32(255-i, 0, 0, 255-i);
} else {
colors[i] = SkPackARGB32(0xFF, i, 0, 255-i);
}
}
SkColorTable* ctable = new SkColorTable(colors, 256);
bm->allocPixels(ctable);
ctable->unref();
for (int y = 0; y < height; y++) {
for (int x = 0; x < width; x++) {
*bm->getAddr8(x, y) = SkGetPackedR32(*device.getAddr32(x, y));
}
}
break;
}
default:
break;
}
}
// configs to build the original bitmap in. Can be at most these 3
static const SkBitmap::Config gConfigs[] = {
SkBitmap::kARGB_8888_Config,
SkBitmap::kRGB_565_Config,
SkBitmap::kIndex8_Config, // opaque
SkBitmap::kIndex8_Config // alpha
};
static const char* const gConfigLabels[] = {
"8888", "565", "Index8", "Index8 alpha"
};
// types to encode into. Can be at most these 3. Must match up with gExt[]
static const SkImageEncoder::Type gTypes[] = {
SkImageEncoder::kJPEG_Type,
SkImageEncoder::kPNG_Type
};
// must match up with gTypes[]
static const char* const gExt[] = {
".jpg", ".png"
};
static const char* gPath = "/encoded/";
static void make_name(SkString* name, int config, int ext) {
name->set(gPath);
name->append(gConfigLabels[config]);
name->append(gExt[ext]);
}
#include <sys/stat.h>
class EncodeView : public SampleView {
public:
SkBitmap* fBitmaps;
size_t fBitmapCount;
EncodeView() {
#if 1
(void)mkdir(gPath, S_IRWXU | S_IRWXG | S_IRWXO);
fBitmapCount = SK_ARRAY_COUNT(gConfigs);
fBitmaps = new SkBitmap[fBitmapCount];
for (size_t i = 0; i < fBitmapCount; i++) {
make_image(&fBitmaps[i], gConfigs[i], i);
for (size_t j = 0; j < SK_ARRAY_COUNT(gExt); j++) {
SkString path;
make_name(&path, i, j);
// remove any previous run of this file
remove(path.c_str());
SkImageEncoder* codec = SkImageEncoder::Create(gTypes[j]);
if (NULL == codec ||
!codec->encodeFile(path.c_str(), fBitmaps[i], 100)) {
SkDebugf("------ failed to encode %s\n", path.c_str());
remove(path.c_str()); // remove any partial file
}
delete codec;
}
}
#else
fBitmaps = NULL;
fBitmapCount = 0;
#endif
this->setBGColor(0xFFDDDDDD);
}
virtual ~EncodeView() {
delete[] fBitmaps;
}
protected:
// overrides from SkEventSink
virtual bool onQuery(SkEvent* evt) {
if (SampleCode::TitleQ(*evt)) {
SampleCode::TitleR(evt, "ImageEncoder");
return true;
}
return this->INHERITED::onQuery(evt);
}
virtual void onDrawContent(SkCanvas* canvas) {
if (fBitmapCount == 0) {
return;
}
SkPaint paint;
paint.setAntiAlias(true);
paint.setTextAlign(SkPaint::kCenter_Align);
canvas->translate(SkIntToScalar(10), SkIntToScalar(20));
SkScalar x = 0, y = 0, maxX = 0;
const int SPACER = 10;
for (size_t i = 0; i < fBitmapCount; i++) {
canvas->drawText(gConfigLabels[i], strlen(gConfigLabels[i]),
x + SkIntToScalar(fBitmaps[i].width()) / 2, 0,
paint);
y = paint.getTextSize();
canvas->drawBitmap(fBitmaps[i], x, y);
SkScalar yy = y;
for (size_t j = 0; j < SK_ARRAY_COUNT(gExt); j++) {
yy += SkIntToScalar(fBitmaps[i].height() + 10);
SkBitmap bm;
SkString name;
make_name(&name, i, j);
SkImageDecoder::DecodeFile(name.c_str(), &bm);
canvas->drawBitmap(bm, x, yy);
}
x += SkIntToScalar(fBitmaps[i].width() + SPACER);
if (x > maxX) {
maxX = x;
}
}
y = (paint.getTextSize() + SkIntToScalar(fBitmaps[0].height())) * 3 / 2;
x = maxX + SkIntToScalar(10);
paint.setTextAlign(SkPaint::kLeft_Align);
for (size_t j = 0; j < SK_ARRAY_COUNT(gExt); j++) {
canvas->drawText(gExt[j], strlen(gExt[j]), x, y, paint);
y += SkIntToScalar(fBitmaps[0].height() + SPACER);
}
}
virtual SkView::Click* onFindClickHandler(SkScalar x, SkScalar y) {
this->inval(NULL);
return this->INHERITED::onFindClickHandler(x, y);
}
private:
typedef SampleView INHERITED;
};
//////////////////////////////////////////////////////////////////////////////
static SkView* MyFactory() { return new EncodeView; }
static SkViewRegister reg(MyFactory);