3a8b0d8288
Over-the-shoulder review from reed@ Review URL: https://codereview.appspot.com/6512043 git-svn-id: http://skia.googlecode.com/svn/trunk@5545 2bbb7eff-a529-9590-31e7-b0007b416f81
139 lines
3.9 KiB
C++
139 lines
3.9 KiB
C++
/*
|
|
* Copyright 2012 Google Inc.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
* found in the LICENSE file.
|
|
*/
|
|
|
|
#include "picture_utils.h"
|
|
#include "SkCanvas.h"
|
|
#include "SkColorPriv.h"
|
|
#include "SkBitmap.h"
|
|
#include "SkPicture.h"
|
|
#include "SkRefCnt.h"
|
|
#include "SkString.h"
|
|
#include "SkStream.h"
|
|
|
|
namespace sk_tools {
|
|
void force_all_opaque(const SkBitmap& bitmap) {
|
|
SkASSERT(NULL == bitmap.getTexture());
|
|
SkASSERT(SkBitmap::kARGB_8888_Config == bitmap.config());
|
|
if (NULL != bitmap.getTexture() || SkBitmap::kARGB_8888_Config == bitmap.config()) {
|
|
return;
|
|
}
|
|
|
|
SkAutoLockPixels lock(bitmap);
|
|
for (int y = 0; y < bitmap.height(); y++) {
|
|
for (int x = 0; x < bitmap.width(); x++) {
|
|
*bitmap.getAddr32(x, y) |= (SK_A32_MASK << SK_A32_SHIFT);
|
|
}
|
|
}
|
|
}
|
|
|
|
void make_filepath(SkString* path, const SkString& dir, const SkString& name) {
|
|
size_t len = dir.size();
|
|
path->set(dir);
|
|
if (0 < len && '/' != dir[len - 1]) {
|
|
path->append("/");
|
|
}
|
|
path->append(name);
|
|
}
|
|
|
|
namespace {
|
|
bool is_path_seperator(const char chr) {
|
|
#if defined(SK_BUILD_FOR_WIN)
|
|
return chr == '\\' || chr == '/';
|
|
#else
|
|
return chr == '/';
|
|
#endif
|
|
}
|
|
}
|
|
|
|
void get_basename(SkString* basename, const SkString& path) {
|
|
if (path.size() == 0) {
|
|
basename->reset();
|
|
return;
|
|
}
|
|
|
|
size_t end = path.size() - 1;
|
|
|
|
// Paths pointing to directories often have a trailing slash,
|
|
// we remove it so the name is not empty
|
|
if (is_path_seperator(path[end])) {
|
|
if (end == 0) {
|
|
basename->reset();
|
|
return;
|
|
}
|
|
|
|
end -= 1;
|
|
}
|
|
|
|
size_t i = end;
|
|
do {
|
|
--i;
|
|
if (is_path_seperator(path[i])) {
|
|
const char* basenameStart = path.c_str() + i + 1;
|
|
size_t basenameLength = end - i;
|
|
basename->set(basenameStart, basenameLength);
|
|
return;
|
|
}
|
|
} while (i > 0);
|
|
|
|
basename->set(path.c_str(), end + 1);
|
|
}
|
|
|
|
bool is_percentage(const char* const string) {
|
|
SkString skString(string);
|
|
return skString.endsWith("%");
|
|
}
|
|
|
|
void setup_bitmap(SkBitmap* bitmap, int width, int height) {
|
|
bitmap->setConfig(SkBitmap::kARGB_8888_Config, width, height);
|
|
bitmap->allocPixels();
|
|
bitmap->eraseColor(0);
|
|
}
|
|
|
|
bool area_too_big(int w, int h, SkISize* newSize) {
|
|
// just a guess, based on what seems to fail on smaller android devices
|
|
static const int64_t kMaxAreaForMemory = 4 * 1024 * 1024;
|
|
|
|
if ((int64_t)w * h > kMaxAreaForMemory) {
|
|
do {
|
|
w >>= 1;
|
|
h >>= 1;
|
|
} while ((int64_t)w * h > kMaxAreaForMemory);
|
|
if (0 == w) {
|
|
w = 1;
|
|
}
|
|
if (0 == h) {
|
|
h = 1;
|
|
}
|
|
newSize->set(w, h);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void resize_if_needed(SkAutoTUnref<SkPicture>* aur) {
|
|
SkISize newSize;
|
|
SkPicture* picture = aur->get();
|
|
if (area_too_big(picture->width(), picture->height(), &newSize)) {
|
|
SkPicture* pic = SkNEW(SkPicture);
|
|
picture->ref();
|
|
aur->reset(pic);
|
|
|
|
SkCanvas* canvas = pic->beginRecording(newSize.width(),
|
|
newSize.height());
|
|
SkScalar scale = SkIntToScalar(newSize.width()) / picture->width();
|
|
canvas->scale(scale, scale);
|
|
canvas->drawPicture(*picture);
|
|
pic->endRecording();
|
|
|
|
SkDebugf(
|
|
"... rescaling to [%d %d] to avoid overly large allocations\n",
|
|
newSize.width(), newSize.height());
|
|
picture->unref();
|
|
}
|
|
}
|
|
}
|