MODULE WMTransitions;
IMPORT
SYSTEM, BIT, Raster, WMGraphics, WMGraphicUtilities, WMRectangles, WMWindowManager, Machine, KernelLog;
CONST
DEBUG = FALSE;
TYPE
Transition* = OBJECT
PROCEDURE Init*(w, h : LONGINT);
END Init;
PROCEDURE CalcImage*(a, b, result : Raster.Image; per255 : LONGINT);
END CalcImage;
END Transition;
TYPE
TransitionMask* = OBJECT(Transition)
VAR
mW, mH : LONGINT;
mask : POINTER TO ARRAY OF CHAR;
PROCEDURE Init*(w, h : LONGINT);
BEGIN
mW := w; mH := h;
END Init;
PROCEDURE SetMask*(mi : Raster.Image);
VAR x, y, col, i, tr, tg, tb, ta : LONGINT;
adr: SYSTEM.ADDRESS;
ti : Raster.Image;
mode : Raster.Mode;
pix : Raster.Pixel;
c : WMGraphics.BufferCanvas;
BEGIN
IF (mi.width # mW) OR (mi.height # mH) THEN
NEW(ti); Raster.Create(ti, mW, mH, Raster.BGR888);
NEW(c, ti);
c.ScaleImage(mi, WMRectangles.MakeRect(0, 0, mi.width - 1, mi.height - 1),
WMRectangles.MakeRect(0, 0, ti.width - 1, ti.height - 1), WMGraphics.ModeCopy, WMGraphics.ScaleBilinear);
mi := ti
END;
NEW(SELF.mask, mW * mH);
i := 0;
IF (mi.fmt.code = Raster.bgr565) THEN
FOR y := 0 TO mi.height -1 DO
adr := mi.adr + y * mi.bpr;
FOR x := 0 TO mi.width - 1 DO
col := SYSTEM.GET16(adr + x * 2);
mask[i] := CHR((col DIV 32 MOD 64) * 4);
INC(i)
END
END
ELSIF (mi.fmt.code = Raster.bgr888) THEN
FOR y := 0 TO mi.height -1 DO
adr := mi.adr + y * mi.bpr;
FOR x := 0 TO mi.width - 1 DO
mask[i] := CHR(SYSTEM.GET8(adr + x * 3 + 1));
INC(i)
END
END
ELSE
Raster.InitMode(mode, Raster.srcCopy);
FOR y := 0 TO mi.height -1 DO
FOR x := 0 TO mi.width - 1 DO
Raster.Get(mi, x, y, pix, mode);
Raster.GetRGBA(pix, tr, tg, tb, ta);
mask[i] := CHR(tg);
INC(i)
END
END
END;
END SetMask;
PROCEDURE DumpMask;
VAR i, x, y : LONGINT;
w : WMWindowManager.BufferWindow;
BEGIN
NEW(w, mW, mH, FALSE);
i := 0;
FOR y := 0 TO mH -1 DO
FOR x := 0 TO mW - 1 DO
w.canvas.SetPixel(x, y, WMGraphics.RGBAToColor(0, ORD(mask[i]), 0, 255), WMGraphics.ModeCopy);
INC(i)
END
END;
WMWindowManager.DefaultAddWindow(w);
END DumpMask;
PROCEDURE CalcImage*(a, b, result : Raster.Image; per255 : LONGINT);
VAR i, x, y, col : LONGINT;
adra, adrb, adrr: SYSTEM.ADDRESS;
mode : Raster.Mode;
pix : Raster.Pixel;
BEGIN
IF (a = NIL) OR (b = NIL) OR (result = NIL) OR
(a.height # b.height) OR (a.width # b.width) OR (result.height # result.height) OR
(a.fmt.code # b.fmt.code) OR (a.fmt.code # b.fmt.code) OR (result.fmt.code # result.fmt.code)
THEN
RETURN
END;
i := 0;
IF (a.fmt.code = Raster.bgr565) THEN
FOR y := 0 TO a.height -1 DO
adra := a.adr + y * a.bpr;
adrb := b.adr + y * b.bpr;
adrr := result.adr + y * result.bpr;
FOR x := 0 TO a.width - 1 DO
IF ORD(mask[i]) <= per255 THEN col := SYSTEM.GET16(adra + x * 2)
ELSE col := SYSTEM.GET16(adrb + x * 2)
END;
SYSTEM.PUT16(adrr + x * 2, col);
INC(i)
END
END
ELSIF (a.fmt.code = Raster.bgr888) THEN
FOR y := 0 TO a.height -1 DO
adra := a.adr + y * a.bpr;
adrb := b.adr + y * b.bpr;
adrr := result.adr + y * result.bpr;
FOR x := 0 TO a.width - 1 DO
IF ORD(mask[i]) <= per255 THEN SYSTEM.MOVE(adra + x * 3, adrr + x * 3, 3)
ELSE SYSTEM.MOVE(adrb + x * 3, adrr + x * 3, 3)
END;
INC(i)
END
END
ELSE
Raster.InitMode(mode, Raster.srcCopy);
FOR y := 0 TO a.height -1 DO
FOR x := 0 TO a.width - 1 DO
IF ORD(mask[i]) <= per255 THEN Raster.Get(a, x, y, pix, mode)
ELSE Raster.Get(b, x, y, pix, mode)
END;
Raster.Put(result, x, y, pix, mode);
INC(i)
END
END
END;
END CalcImage;
END TransitionMask;
TransitionFade* = OBJECT(Transition)
VAR
mW, mH : LONGINT;
mode : Raster.Mode;
PROCEDURE Init*(w, h : LONGINT);
BEGIN
mW := w; mH := h;
Raster.InitMode(mode, Raster.srcCopy);
END Init;
PROCEDURE CalcImage*(a, b, result : Raster.Image; per255 : LONGINT);
BEGIN
IF (per255 < 0) THEN per255 := 0; END;
IF (per255 > 255) THEN per255 := 255; END;
IF (a.fmt.code = Raster.bgr565) & (b.fmt.code = Raster.bgr565) THEN
IF (MMXenabled) THEN
Calc565MMX(a, b, result, per255);
ELSE
Calc565Opt(a, b, result, per255);
END;
ELSIF (a.fmt.code = Raster.bgr888) & (b.fmt.code = Raster.bgr888) THEN
IF (MMXenabled) THEN
Calc888MMX(a, b, result, per255);
ELSE
Calc888Opt(a, b, result, per255);
END;
ELSIF (a.fmt.code = b.fmt.code) THEN
CalcGenUnknown(a, b, result, per255);
ELSE
IF (DEBUG) THEN KernelLog.String("Error: source formats not equal!"); KernelLog.Ln; END;
END;
END CalcImage;
END TransitionFade;
VAR
MMXenabled : BOOLEAN;
PROCEDURE Calc565Opt(a, b : Raster.Image; VAR result : Raster.Image; per255 : LONGINT);
VAR
x, y : LONGINT;
adra, adrb, adrr : SYSTEM.ADDRESS;
ar,ag,ab, br,bg,bb, cr,cg,cb : LONGINT;
height, width : LONGINT;
oddWidth : BOOLEAN; blocksOf4Bytes : LONGINT;
add64, alphaOver2 : LONGINT;
tmpA, tmpB : LONGINT;
BEGIN
IF (DEBUG) THEN KernelLog.String("Fade in 565-Format"); KernelLog.Ln; END;
height := a.height; width := a.width;
IF ( (width MOD 2)=0 ) THEN
oddWidth := FALSE;
blocksOf4Bytes := width DIV 2;
ELSE
oddWidth := TRUE;
blocksOf4Bytes := (width-1) DIV 2;
END;
add64 := BIT.LOR(64, ASH(64, 16));
alphaOver2 := BIT.LOR( (per255 DIV 4), ASH((per255 DIV 4),16) );
FOR y := 0 TO height -1 DO
adra := a.adr + y * a.bpr;
adrb := b.adr + y * b.bpr;
adrr := result.adr + y * result.bpr;
FOR x := 0 TO blocksOf4Bytes DO
tmpA := SYSTEM.GET32(adra + x * 4);
tmpB := SYSTEM.GET32(adrb + x * 4);
ar := BIT.LAND( ASH(tmpA, -11), 001F001FH );
br := BIT.LAND( ASH(tmpB, -11), 001F001FH );
ag := BIT.LAND( ASH(tmpA, -5), 003F003FH );
bg := BIT.LAND( ASH(tmpB, -5), 003F003FH );
ab := BIT.LAND( tmpA, 001F001FH );
bb := BIT.LAND( tmpB, 001F001FH );
cr := ASH( BIT.LAND( (ASH(per255*(br+add64-ar), -8) + ar-alphaOver2), 001F001FH) , 11 );
cg := ASH( BIT.LAND( (ASH(per255*(bg+add64-ag), -8) + ag-alphaOver2), 003F003FH) , 5 );
cb := BIT.LAND( (ASH(per255*(bb+add64-ab), -8) + ab-alphaOver2), 001F001FH);
SYSTEM.PUT32(adrr + 4*x, BIT.LOR( BIT.LOR(cr,cg), cb) );
END;
IF (oddWidth) THEN
tmpA := SYSTEM.GET16(adra + x * 4);
tmpB := SYSTEM.GET16(adrb + x * 4);
ar := BIT.LAND( ASH(tmpA, -11), 1FH);
br := BIT.LAND( ASH(tmpB, -11), 1FH);
ag := BIT.LAND( ASH(tmpA, -5), 3FH);
bg := BIT.LAND( ASH(tmpB, -5), 3FH);
ab := BIT.LAND(tmpA, 1FH);
bb := BIT.LAND(tmpB, 1FH);
cr := ASH(ASH(per255*(br-ar),-8)+ar, 11);
cg := ASH(ASH(per255*(bg-ag),-8)+ag, 5);
cb := ASH(per255*(bb-ab),-8)+ab;
SYSTEM.PUT16(adrr + 2*x, BIT.LOR( BIT.LOR(cr,cg), cb));
END;
END;
END Calc565Opt;
PROCEDURE Calc565MMX(a, b : Raster.Image; VAR result : Raster.Image; per255 : LONGINT);
VAR
x, y : LONGINT;
height, width : LONGINT;
remainder : LONGINT;
blocksOf8Bytes : LONGINT;
adra, adrb, adrr : SYSTEM.ADDRESS;
alpha64, maskRed64, maskGreen64, maskBlue64 : HUGEINT;
alpha32 : LONGINT;
tmpA, tmpB : LONGINT;
ar, ag, ab, br, bg, bb, cr, cg, cb : LONGINT;
BEGIN
IF (DEBUG) THEN KernelLog.String("Fade in 565-Format (MMX)"); KernelLog.Ln; END;
height := a.height; width := a.width;
remainder := width MOD 4;
blocksOf8Bytes := (width-remainder) DIV 4;
maskRed64 := 0F800F800F800F800H;
maskGreen64 := 007E007E007E007E0H;
maskBlue64 := 0001F001F001F001FH;
alpha64 := 0;
alpha32 := BIT.LOR( per255, ASH(per255,16) );
SYSTEM.PUT32( SYSTEM.ADR(alpha64), alpha32);
SYSTEM.PUT32( SYSTEM.ADR(alpha64)+4, alpha32);
FOR y := 0 TO height -1 DO
adra := a.adr + y * a.bpr;
adrb := b.adr + y * b.bpr;
adrr := result.adr + y * result.bpr;
Calc565MMXLine(adra, adrb, adrr, blocksOf8Bytes, per255, alpha64, maskRed64, maskGreen64, maskBlue64);
IF (remainder # 0) THEN
FOR x := 0 TO remainder-1 DO
tmpA := SYSTEM.GET16(adra + blocksOf8Bytes*8 + 2*x);
tmpB := SYSTEM.GET16(adrb + blocksOf8Bytes*8 + 2*x);
ar := BIT.LAND( ASH(tmpA, -11), 1FH);
br := BIT.LAND( ASH(tmpB, -11), 1FH);
ag := BIT.LAND( ASH(tmpA, -5), 3FH);
bg := BIT.LAND( ASH(tmpB, -5), 3FH);
ab := BIT.LAND(tmpA, 1FH);
bb := BIT.LAND(tmpB, 1FH);
cr := ASH(ASH(per255*(br-ar),-8)+ar, 11);
cg := ASH(ASH(per255*(bg-ag),-8)+ag, 5);
cb := ASH(per255*(bb-ab),-8)+ab;
SYSTEM.PUT16(adrr + blocksOf8Bytes*8 + 2*x, BIT.LOR( BIT.LOR(cr,cg), cb));
END;
END;
END;
END Calc565MMX;
PROCEDURE Calc565MMXLine (adra, adrb, adrr: SYSTEM.ADDRESS; i, alpha : LONGINT; a64, mr64, mg64, mb64 : HUGEINT);
CODE {SYSTEM.i386, SYSTEM.MMX}
;
; Initialize the counter and skip if the latter is equal to zero
;
PUSH ECX
MOV ECX, [EBP+i]
CMP ECX, 0
JZ skip565
;
; Load the frame buffer pointers into the registers
;
PUSH EDI
PUSH ESI
PUSH EAX
MOV EDI, [EBP+adra] ; source address of image A
MOV ESI, [EBP+adrb] ; source address of image B
MOV EAX, [EBP+adrr] ; destination address of image RESULT
doblend565:
;
; Alpha blend four target and source pixels
;
;
; The mmx registers will basically be used in the following way:
; MMX0: red source value A
; MMX1: red source value B
; MMX2: green source value A
; MMX3: green source value B
; MMX4: blue source value A
; MMX5: blue source value B
; MMX6: original source pixel A
; MMX7: original source pixel B
;
;
; Note: Two lines together are assumed to pair
; in the processornd V-pipes
;
MOVQ MMX6, [EDI] ; Load the original source pixel A
NOP
MOVQ MMX7, [ESI] ; Load the original source pixel B
MOVQ MMX0, MMX6 ; Load the register for the red source A
PAND MMX0, [EBP+mr64] ; Extract the red source A channel
MOVQ MMX1, MMX7 ; Load the register for the red source B
PAND MMX1, [EBP+mr64] ; Extract the red source B channel
PSRLW MMX0, 11 ; Shift down the red source A channel
MOVQ MMX2, MMX6 ; Load the register for the green source A
PSRLW MMX1, 11 ; Shift down the red source B channel
MOVQ MMX3, MMX7 ; Load the register for the green source B
PSUBW MMX1, MMX0 ; Calculate red source B minus red source A
PMULLW MMX1, [EBP+a64] ; Multiply the red result with alpha
NOP
PAND MMX2, [EBP+mg64] ; Extract the green source A channel
NOP
PAND MMX3, [EBP+mg64] ; Extract the green source B channel
PSRAW MMX1, 8 ; Divide the red result by 256
PSRLW MMX2, 5 ; Shift down the green source B channel
PADDW MMX1, MMX0 ; Add the red source B to the red result
PSLLW MMX1, 11 ; Shift up the red source A again
MOVQ MMX4, MMX6 ; Load the register for the blue source A
PSRLW MMX3, 5 ; Shift down the green source B channel
MOVQ MMX5, MMX7 ; Load the register for the blue source B
PAND MMX4, [EBP+mb64] ; Extract the blue source A channel
PSUBW MMX3, MMX2 ; Calculate green source B minus green source A
PAND MMX5, [EBP+mb64] ; Extract the blue source B channel
PMULLW MMX3, [EBP+a64] ; Multiply the green result with alpha
PSUBW MMX5, MMX4 ; Calculate blue source B minus blue source A
NOP
PMULLW MMX5, [EBP+a64] ; Multiply the blue result with alpha
PSRAW MMX3, 8 ; Divide the green result by 256
PADDW MMX3, MMX2 ; Add the green source A to the green result
NOP
PSRAW MMX5, 8 ; Divide the blue result by 256
PSLLW MMX3, 5 ; Shift up the green source B again
PADDW MMX5, MMX4 ; Add the blue source A to the blue result
POR MMX1, MMX3 ; Combine the new red and green values
POR MMX1, MMX5 ; Combine new blue value with the others to RESULT pixel
MOVQ [EAX], MMX1 ; Write back RESULT value
;
; Advance to the next four pixels
;
ADD EDI, 8
ADD ESI, 8
ADD EAX, 8
;
; Loop again or break
;
DEC ECX
JNZ doblend565
;
; Clean up
;
POP EAX
POP ESI
POP EDI
EMMS ; Declare FPU registers free
skip565:
POP ECX
END Calc565MMXLine;
PROCEDURE Calc888Opt(a, b : Raster.Image; VAR result : Raster.Image; per255 : LONGINT);
VAR
x, y : LONGINT;
height, width : LONGINT;
adra, adrb, adrr : SYSTEM.ADDRESS;
ar,ag,ab, br,bg,bb, cr,cg,cb : LONGINT;
tmpA, tmpB, tmpR : LONGINT;
BEGIN
IF (DEBUG) THEN KernelLog.String("Fade in 888-Format"); KernelLog.Ln; END;
height := a.height; width := a.width;
FOR y := 0 TO height -1 DO
adra := a.adr + y * a.bpr;
adrb := b.adr + y * b.bpr;
adrr := result.adr + y * result.bpr;
FOR x := 0 TO width-1 DO
IF (x = width-1) THEN
tmpA := BIT.LOR( SYSTEM.GET16(adra + x * 3), ASH(SYSTEM.GET8(adra + x * 3 + 2),16) );
tmpB := BIT.LOR( SYSTEM.GET16(adrb + x * 3), ASH(SYSTEM.GET8(adrb + x * 3 + 2),16) );
ELSE
tmpA := SYSTEM.GET32(adra + x * 3);
tmpB := SYSTEM.GET32(adrb + x * 3);
END;
ar := BIT.LAND( ASH(tmpA,-16), 0FFH );
br := BIT.LAND( ASH(tmpB,-16), 0FFH );
ag := BIT.LAND( ASH(tmpA,-8), 0FFH );
bg := BIT.LAND( ASH(tmpB,-8), 0FFH );
ab := BIT.LAND( tmpA, 0FFH );
bb := BIT.LAND( tmpB, 0FFH );
cr := ASH ( ASH( per255*(br-ar), -8) + ar , 16);
cg := ASH ( ASH( per255*(bg-ag), -8) + ag , 8);
cb := ASH( per255*(bb-ab), -8) + ab;
tmpR := BIT.LOR( BIT.LOR(cr,cg), cb );
SYSTEM.PUT16(adrr + x * 3 , BIT.LAND(tmpR, 0FFFFH) );
SYSTEM.PUT8 (adrr + x * 3 + 2, ASH(tmpR, -16) );
END;
END;
END Calc888Opt;
PROCEDURE Calc888MMX(a, b : Raster.Image; VAR result : Raster.Image; per255 : LONGINT);
VAR
y : LONGINT;
height, width : LONGINT;
adra, adrb, adrr : SYSTEM.ADDRESS;
alpha64, mask64 : HUGEINT;
BEGIN
IF (DEBUG) THEN KernelLog.String("Fade in 565-Format (MMX)"); KernelLog.Ln; END;
height := a.height; width := a.width;
mask64 := 0000000000FFFFFFH;
alpha64 := 0;
SYSTEM.PUT32( SYSTEM.ADR(alpha64), BIT.LOR( per255, ASH(per255,16)) );
SYSTEM.PUT32( SYSTEM.ADR(alpha64)+4, per255);
FOR y := 0 TO height -1 DO
adra := a.adr + y * a.bpr;
adrb := b.adr + y * b.bpr;
adrr := result.adr + y * result.bpr;
Calc888MMXLine(adra, adrb, adrr, width, alpha64, mask64);
END;
END Calc888MMX;
PROCEDURE Calc888MMXLine (adra, adrb, adrr: SYSTEM.ADDRESS; i : LONGINT; a64, m64 : HUGEINT);
CODE {SYSTEM.i386, SYSTEM.MMX}
; (re)load the width counter
PUSH ECX
MOV ECX, [EBP+i]
;
; Load the frame buffer pointers into the registers
;
PUSH EDI
PUSH ESI
PUSH EBX
MOV EDI, [EBP+adra] ; source address of image A
MOV ESI, [EBP+adrb] ; source address of image B
MOV EBX, [EBP+adrr] ; destination address of image RESULT
; Load the mask into an mmx register
MOVQ MMX3, [EBP+m64]
; Load the alpha value into an mmx register
MOVQ MMX5, [EBP+a64]
; Clear an mmx register to facilitate unpacking
PXOR MMX6, MMX6
doblend24:
; The mmx registers will basically be used in the following way:
;
; MMX0: source value A
; MMX1: source value B
; MMX2: working register
; MMX3: mask ( 0x00ffffff )
; MMX4: working register
; MMX5: alpha value
; MMX6: zero for unpacking
; MMX7: original result value
;
; Note: Two lines together are assumed to pair
; in the processornd V-pipes
MOVD MMX0, [EDI] ; Load the original source pixel A
MOVQ MMX4, MMX3 ; Reload the mask ( 0x00ffffff )
MOVQ MMX1, [ESI] ; Load the original source pixel B
MOVQ MMX7, MMX0 ; Save the original result pixel
PUNPCKLBW MMX0, MMX6 ; Unpack the source pixel A
PUNPCKLBW MMX1, MMX6 ; Unpack the source pixel B
MOVQ MMX2, MMX0 ; Save the unpacked source A values
NOP
PMULLW MMX0, MMX5 ; Multiply the source A with the alpha value
NOP
PMULLW MMX1, MMX5 ; Multiply the source B with the alpha value
NOP
PSRLW MMX0, 8 ; Divide the source A by 256
NOP
PSRLW MMX1, 8 ; Divide the source B by 256
NOP
PSUBW MMX1, MMX0 ; Calculate the source B minus source A
NOP
PADDW MMX2, MMX1 ; Add former result value to the new result
NOP
PACKUSWB MMX2, MMX2 ; Pack the new result
NOP
PAND MMX2, MMX4 ; Mask of unwanted bytes
NOP
PANDN MMX4, MMX7 ; Get the high order byte we must keep
NOP
POR MMX2, MMX4 ; Assemble the value to write back
NOP
MOVD [EBX], MMX2 ; Write back the new value to result image
;
; Advance to the next pixel
;
ADD EDI, 3
ADD ESI, 3
ADD EBX, 3
;
; Loop again or break
;
DEC ECX
JNZ doblend24
;
; Write back the frame buffer pointers and clean up
;
POP EBX
POP ESI
POP EDI
EMMS ; Declare FPU registers free
POP ECX
END Calc888MMXLine;
PROCEDURE CalcGenUnknown(a, b : Raster.Image; VAR result : Raster.Image; perc : LONGINT);
VAR
x, y : LONGINT;
mode : Raster.Mode;
canvas : WMGraphics.BufferCanvas;
pix : Raster.Pixel;
ca, cb : LONGINT;
red, green, blue, alpha : LONGINT;
BEGIN
IF (DEBUG) THEN
KernelLog.String("Fade in other Format ["); KernelLog.String("a.fmt.code= "); KernelLog.Int(a.fmt.code, 0); KernelLog.String("b.fmt.code= "); KernelLog.Int(b.fmt.code, 0); KernelLog.String("]"); KernelLog.Ln;
END;
Raster.InitMode(mode, Raster.srcCopy);
NEW(canvas, result);
IF (canvas = NIL) & (DEBUG) THEN
KernelLog.String("Error during calculating fade: couldn't allocate buffer canvas!"); KernelLog.Ln;
HALT(99);
END;
FOR y := 0 TO a.height -1 DO
FOR x := 0 TO a.width - 1 DO
Raster.Get(a, x, y, pix, mode); Raster.GetRGBA(pix, red, green, blue, alpha); ca := WMGraphics.RGBAToColor(red, green, blue, alpha);
Raster.Get(b, x, y, pix, mode); Raster.GetRGBA(pix, red, green, blue, alpha); cb := WMGraphics.RGBAToColor(red, green, blue, alpha);
canvas.SetPixel(x, y, WMGraphicUtilities.InterpolateColorLinear(ca, cb, perc), WMGraphics.ModeCopy);
END
END;
END CalcGenUnknown;
PROCEDURE LoadImage(CONST fileName : ARRAY OF CHAR) : Raster.Image;
VAR t, img : Raster.Image;
c : WMGraphics.BufferCanvas;
BEGIN
t := WMGraphics.LoadImage(fileName, TRUE);
IF t # NIL THEN
NEW(img);
Raster.Create(img, t.width, t.height, Raster.BGR565);
NEW(c, img);
c.DrawImage(0, 0, t, WMGraphics.ModeCopy)
END;
RETURN img
END LoadImage;
PROCEDURE Test*;
VAR w : WMWindowManager.BufferWindow;
t : TransitionMask;
a, b, m : Raster.Image;
i : LONGINT;
BEGIN
NEW(t);
a := LoadImage("Reto01.png");
b := LoadImage("Reto02.png");
m := WMGraphics.LoadImage("M_Art1.png", TRUE);
t.Init(a.width, a.height);
NEW(w, a.width, a.height, FALSE);
WMWindowManager.DefaultAddWindow(w);
t.SetMask(m);
FOR i := 0 TO 256 DO
t.CalcImage(a, b, w.img, i );
w.Invalidate(WMRectangles.MakeRect(0, 0, a.width, a.height));
END;
END Test;
PROCEDURE Test2*;
VAR w : WMWindowManager.BufferWindow;
t : TransitionFade;
a, b : Raster.Image;
i : LONGINT;
BEGIN
NEW(t);
a := LoadImage("Reto01.png");
b := LoadImage("Reto02.png");
t.Init(a.width, a.height);
NEW(w, a.width, a.height, FALSE);
WMWindowManager.DefaultAddWindow(w);
FOR i := 0 TO 256 DO
t.CalcImage(a, b, w.img, i );
w.Invalidate(WMRectangles.MakeRect(0, 0, a.width, a.height));
END;
END Test2;
BEGIN
MMXenabled := 23 IN Machine.features;
END WMTransitions.
SystemTools.Free WMTransitions ~
WMTransitions.Test ~
WMTransitions.Test2 ~