166 lines
5.8 KiB
C++
166 lines
5.8 KiB
C++
/*
|
|
* Copyright (C) 2015 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
#include "TestUtils.h"
|
|
|
|
#include "DeferredLayerUpdater.h"
|
|
#include "LayerRenderer.h"
|
|
|
|
#include <unistd.h>
|
|
#include <signal.h>
|
|
#include <setjmp.h>
|
|
|
|
namespace android {
|
|
namespace uirenderer {
|
|
|
|
SkColor TestUtils::interpolateColor(float fraction, SkColor start, SkColor end) {
|
|
int startA = (start >> 24) & 0xff;
|
|
int startR = (start >> 16) & 0xff;
|
|
int startG = (start >> 8) & 0xff;
|
|
int startB = start & 0xff;
|
|
|
|
int endA = (end >> 24) & 0xff;
|
|
int endR = (end >> 16) & 0xff;
|
|
int endG = (end >> 8) & 0xff;
|
|
int endB = end & 0xff;
|
|
|
|
return (int)((startA + (int)(fraction * (endA - startA))) << 24)
|
|
| (int)((startR + (int)(fraction * (endR - startR))) << 16)
|
|
| (int)((startG + (int)(fraction * (endG - startG))) << 8)
|
|
| (int)((startB + (int)(fraction * (endB - startB))));
|
|
}
|
|
|
|
sp<DeferredLayerUpdater> TestUtils::createTextureLayerUpdater(
|
|
renderthread::RenderThread& renderThread, uint32_t width, uint32_t height,
|
|
std::function<void(Matrix4*)> transformSetupCallback) {
|
|
bool isOpaque = true;
|
|
bool forceFilter = true;
|
|
GLenum renderTarget = GL_TEXTURE_EXTERNAL_OES;
|
|
|
|
Layer* layer = LayerRenderer::createTextureLayer(renderThread.renderState());
|
|
LayerRenderer::updateTextureLayer(layer, width, height, isOpaque, forceFilter,
|
|
renderTarget, Matrix4::identity().data);
|
|
transformSetupCallback(&(layer->getTransform()));
|
|
|
|
sp<DeferredLayerUpdater> layerUpdater = new DeferredLayerUpdater(layer);
|
|
return layerUpdater;
|
|
}
|
|
|
|
void TestUtils::drawTextToCanvas(TestCanvas* canvas, const char* text,
|
|
const SkPaint& paint, float x, float y) {
|
|
// drawing text requires GlyphID TextEncoding (which JNI layer would have done)
|
|
LOG_ALWAYS_FATAL_IF(paint.getTextEncoding() != SkPaint::kGlyphID_TextEncoding,
|
|
"must use glyph encoding");
|
|
SkSurfaceProps surfaceProps(0, kUnknown_SkPixelGeometry);
|
|
SkAutoGlyphCacheNoGamma autoCache(paint, &surfaceProps, &SkMatrix::I());
|
|
|
|
float totalAdvance = 0;
|
|
std::vector<glyph_t> glyphs;
|
|
std::vector<float> positions;
|
|
Rect bounds;
|
|
while (*text != '\0') {
|
|
SkUnichar unichar = SkUTF8_NextUnichar(&text);
|
|
glyph_t glyph = autoCache.getCache()->unicharToGlyph(unichar);
|
|
autoCache.getCache()->unicharToGlyph(unichar);
|
|
|
|
// push glyph and its relative position
|
|
glyphs.push_back(glyph);
|
|
positions.push_back(totalAdvance);
|
|
positions.push_back(0);
|
|
|
|
// compute bounds
|
|
SkGlyph skGlyph = autoCache.getCache()->getUnicharMetrics(unichar);
|
|
Rect glyphBounds(skGlyph.fWidth, skGlyph.fHeight);
|
|
glyphBounds.translate(totalAdvance + skGlyph.fLeft, skGlyph.fTop);
|
|
bounds.unionWith(glyphBounds);
|
|
|
|
// advance next character
|
|
SkScalar skWidth;
|
|
paint.getTextWidths(&glyph, sizeof(glyph), &skWidth, NULL);
|
|
totalAdvance += skWidth;
|
|
}
|
|
|
|
// apply alignment via x parameter (which JNI layer would have done)
|
|
if (paint.getTextAlign() == SkPaint::kCenter_Align) {
|
|
x -= totalAdvance / 2;
|
|
} else if (paint.getTextAlign() == SkPaint::kRight_Align) {
|
|
x -= totalAdvance;
|
|
}
|
|
|
|
bounds.translate(x, y);
|
|
|
|
// Force left alignment, since alignment offset is already baked in
|
|
SkPaint alignPaintCopy(paint);
|
|
alignPaintCopy.setTextAlign(SkPaint::kLeft_Align);
|
|
canvas->drawText(glyphs.data(), positions.data(), glyphs.size(), alignPaintCopy, x, y,
|
|
bounds.left, bounds.top, bounds.right, bounds.bottom, totalAdvance);
|
|
}
|
|
|
|
void TestUtils::drawTextToCanvas(TestCanvas* canvas, const char* text,
|
|
const SkPaint& paint, const SkPath& path) {
|
|
// drawing text requires GlyphID TextEncoding (which JNI layer would have done)
|
|
LOG_ALWAYS_FATAL_IF(paint.getTextEncoding() != SkPaint::kGlyphID_TextEncoding,
|
|
"must use glyph encoding");
|
|
SkSurfaceProps surfaceProps(0, kUnknown_SkPixelGeometry);
|
|
SkAutoGlyphCacheNoGamma autoCache(paint, &surfaceProps, &SkMatrix::I());
|
|
|
|
std::vector<glyph_t> glyphs;
|
|
while (*text != '\0') {
|
|
SkUnichar unichar = SkUTF8_NextUnichar(&text);
|
|
glyphs.push_back(autoCache.getCache()->unicharToGlyph(unichar));
|
|
}
|
|
canvas->drawTextOnPath(glyphs.data(), glyphs.size(), path, 0, 0, paint);
|
|
}
|
|
|
|
static void defaultCrashHandler() {
|
|
fprintf(stderr, "RenderThread crashed!");
|
|
}
|
|
|
|
static jmp_buf gErrJmpBuff;
|
|
static std::function<void()> gCrashHandler = defaultCrashHandler;
|
|
|
|
static void signalHandler(int sig) {
|
|
longjmp(gErrJmpBuff, 1);
|
|
}
|
|
|
|
void TestUtils::setRenderThreadCrashHandler(std::function<void()> crashHandler) {
|
|
gCrashHandler = crashHandler;
|
|
}
|
|
|
|
void TestUtils::TestTask::run() {
|
|
struct sigaction act;
|
|
memset(&act, 0, sizeof(act));
|
|
act.sa_handler = signalHandler;
|
|
|
|
if (setjmp(gErrJmpBuff)) {
|
|
gCrashHandler();
|
|
return;
|
|
}
|
|
|
|
sigaction(SIGABRT, &act, nullptr);
|
|
|
|
|
|
// RenderState only valid once RenderThread is running, so queried here
|
|
RenderState& renderState = renderthread::RenderThread::getInstance().renderState();
|
|
|
|
renderState.onGLContextCreated();
|
|
rtCallback(renderthread::RenderThread::getInstance());
|
|
renderState.onGLContextDestroyed();
|
|
}
|
|
|
|
} /* namespace uirenderer */
|
|
} /* namespace android */
|