ColorDrawBuf - performance optimizations

This commit is contained in:
Vadim Lopatin 2014-12-10 16:13:21 +03:00
parent 1bdf1ce3fa
commit 634789aa36
3 changed files with 65 additions and 29 deletions

View File

@ -456,16 +456,31 @@ class ColorDrawBufBase : DrawBuf {
for (int yy = 0; yy < dy; yy++) {
uint * srcrow = colorDrawBuf.scanLine(srcrect.top + yy) + srcrect.left;
uint * dstrow = scanLine(dstrect.top + yy) + dstrect.left;
if (!_alpha) {
// simplified version - no alpha blending
for (int i = 0; i < dx; i++) {
uint pixel = srcrow[i];
uint alpha = pixel >> 24;
if (!alpha)
dstrow[i] = pixel;
else if (alpha < 254) {
// apply blending
dstrow[i] = blendARGB(dstrow[i], pixel, alpha);
}
}
} else {
// combine two alphas
for (int i = 0; i < dx; i++) {
uint pixel = srcrow[i];
uint alpha = blendAlpha(_alpha, pixel >> 24);
if (!alpha)
dstrow[i] = pixel;
else if (alpha < 255) {
else if (alpha < 254) {
// apply blending
dstrow[i] = blendARGB(dstrow[i], pixel, alpha);
}
}
}
}
}
@ -482,12 +497,17 @@ class ColorDrawBufBase : DrawBuf {
res[i] = src0 + i * sd / dd;
return res;
}
/// draw source buffer rectangle contents to destination buffer rectangle applying rescaling
override void drawRescaled(Rect dstrect, DrawBuf src, Rect srcrect) {
//Log.d("drawRescaled ", dstrect, " <- ", srcrect);
if (_alpha >= 254)
return; // fully transparent - don't draw
if (applyClipping(dstrect, srcrect)) {
int[] xmap = createMap(dstrect.left, dstrect.right, srcrect.left, srcrect.right);
int[] ymap = createMap(dstrect.top, dstrect.bottom, srcrect.top, srcrect.bottom);
int[] xmapArray = createMap(dstrect.left, dstrect.right, srcrect.left, srcrect.right);
int[] ymapArray = createMap(dstrect.top, dstrect.bottom, srcrect.top, srcrect.bottom);
int * xmap = xmapArray.ptr;
int * ymap = ymapArray.ptr;
int dx = dstrect.width;
int dy = dstrect.height;
ColorDrawBufBase colorDrawBuf = cast(ColorDrawBufBase) src;
@ -495,10 +515,12 @@ class ColorDrawBufBase : DrawBuf {
for (int y = 0; y < dy; y++) {
uint * srcrow = colorDrawBuf.scanLine(ymap[y]);
uint * dstrow = scanLine(dstrect.top + y) + dstrect.left;
if (!_alpha) {
// simplified alpha calculation
for (int x = 0; x < dx; x++) {
uint srcpixel = srcrow[xmap[x]];
uint dstpixel = dstrow[x];
uint alpha = blendAlpha(_alpha, srcpixel >> 24);
uint alpha = srcpixel >> 24;
if (!alpha)
dstrow[x] = srcpixel;
else if (alpha < 255) {
@ -506,6 +528,21 @@ class ColorDrawBufBase : DrawBuf {
dstrow[x] = blendARGB(dstpixel, srcpixel, alpha);
}
}
} else {
// blending two alphas
for (int x = 0; x < dx; x++) {
uint srcpixel = srcrow[xmap[x]];
uint dstpixel = dstrow[x];
uint srca = srcpixel >> 24;
uint alpha = !srca ? _alpha : blendAlpha(_alpha, srca);
if (!alpha)
dstrow[x] = srcpixel;
else if (alpha < 255) {
// apply blending
dstrow[x] = blendARGB(dstpixel, srcpixel, alpha);
}
}
}
}
}
}
@ -583,6 +620,7 @@ class ColorDrawBufBase : DrawBuf {
//Log.d("NinePatch detected: frame=", p.frame, " padding=", p.padding, " left+right=", p.frame.left + p.frame.right, " dx=", _dx);
return true;
}
override void drawGlyph(int x, int y, Glyph * glyph, uint color) {
ubyte[] src = glyph.glyph;
int srcdx = glyph.blackBoxX;
@ -616,15 +654,16 @@ class ColorDrawBufBase : DrawBuf {
}
}
}
override void fillRect(Rect rc, uint color) {
if (applyClipping(rc)) {
for (int y = rc.top; y < rc.bottom; y++) {
uint * row = scanLine(y);
uint alpha = color >> 24;
if (!alpha) {
row[rc.left .. rc.right] = color;
} else if (alpha < 254) {
for (int x = rc.left; x < rc.right; x++) {
if (!alpha)
row[x] = color;
else if (alpha < 255) {
// apply blending
row[x] = blendARGB(row[x], color, alpha);
}

View File

@ -120,8 +120,9 @@ class Win32ColorDrawBuf : ColorDrawBufBase {
}
override void fill(uint color) {
int len = _dx * _dy;
for (int i = 0; i < len; i++)
_pixels[i] = color;
//for (int i = 0; i < len; i++)
// _pixels[i] = color;
_pixels[0 .. len - 1] = color;
}
void drawTo(HDC dc, int x, int y) {
BitBlt(dc, x, y, _dx, _dy, _drawdc, 0, 0, SRCCOPY);

View File

@ -357,11 +357,7 @@ class UndoBuffer {
return; // merged - no need to add new operation
}
}
long ts = currentTimeMillis();
_undoList.pushBack(op);
long duration = currentTimeMillis() - ts;
if (duration > 3)
Log.w("Too long saveForUndo: ", duration, " ms");
}
/// returns operation to be undone (put it to redo), null if no undo ops available