我正在努力实现以下目标:
假设一个大的png(透明背景16000 x 70px),其中包含50个不同的其他png文件...... 我需要加载那个png并从中提取单个png(最好是有一种函数,我可以用coords(左,顶,高,宽)来表示我想要提取的...提取的... png应该显示为timage然后......
当然,我可以使用Gif图像并重新创建动画,但出于某种原因我需要png ...
一个想法是将它加载到图像列表但是失败了,因为所有50个png的维度都是(320x70px),timagelist只支持256px宽度......
我的下一个想法可能是我可以做类似的事情:
将Png加载到TBitmapArray中。嗯,提取工作非常好,但副作用是所有都失去alphachannel不再是透明的,而是我得到一个黑色的边框: - (
type
TRectArray = array of TRect;
TBitmapArray = array of TBitmap;
// Zwei Funktionen die Rechtecke aufbereiten:
function FixRect(SrcRect: TRect): TRect;
procedure Switch(var a,b: integer);
var c: integer;
begin
c := a; a := b; b := c;
end;
begin
if SrcRect.Left > SrcRect.Right then
Switch(SrcRect.Left,SrcRect.Right);
if SrcRect.Top > SrcRect.Bottom then
Switch(SrcRect.Top,SrcRect.Bottom);
result := SrcRect;
end;
function TrimRect(SrcRect: TRect; minx,miny,maxx,maxy: integer): TRect;
begin
result := fixrect(srcrect);
if result.Left < minx then result.left := minx;
if result.top < miny then result.top := miny;
if result.right > maxx then result.right := maxx;
if result.bottom > maxy then result.bottom := maxy;
end;
// Stanzt die in SrcRect übergebenen rechtecke aus SrcPNG aus und lädt sie ins
// DstBitmapArray
procedure GetBitmaps(const SrcPNG: TPNGObject; const SrcRects: TRectArray;
var DstBitmapArray: TBitmapArray);
var
i: integer;
Rct: TRect;
Bmp: TBitmap;
begin
// Bitmap vom PNG Erzeugen
Bmp := TBitmap.Create;
Bmp.Assign(SrcPNG);
// Länge der auszugebenden Bilderliste festlegen (=Anzahl der Rechtecke)
setlength(DstBitmapArray,high(SrcRects)+1);
for i := 0 to high(SrcRects) do
begin
// Bitmap erzeugen
DstBitmapArray[i] := TBitmap.Create;
// Rechteck vorbereiten mit obigen Funktionen (ggf Zurechtschneiden,
// falls es über die Grenzen des PNGs hinausgeht)
Rct := TrimRect(SrcRects[i],0,0,SrcPng.Width,SrcPNG.Height);
// Größe des Bitmaps setzen
DstBitmapArray[i].SetSize(rct.Right-rct.left,rct.bottom-rct.top);
// rechteck ausstanzen und auf Bitmap kopieren
BitBlt(DstBitmapArray[i].Canvas.Handle,0,0,DstBitmapArray[i].width,
DstBitmapArray[i].Height,bmp.Canvas.handle,rct.left,rct.top,srccopy);
end;
Bmp.free;
end;
// Stanzt ebenfalls Bilder aus dem PNG aus, die rechtecke werden aber im
// Parameter Positions testbasiert übergeben. jede Zeile definiert ein rechteck
// Die Koordinaten des Rechtecks werden in der reihenfolge Left, Top, Right, Bottom
// angegeben und durch Kommata separiert. Beispiel:
// 0,0,100,50
// 100,0,100,100
// etc...
procedure LoadBitmaps(const SrcPNG: TPNGObject; const Positions: TStrings;
var DstBitmapArray: TBitmapArray);
var
i: integer;
l: integer;
rectarray: TRectArray;
tmp: tstringlist;
begin
setlength(rectarray,positions.Count);
l := 0;
tmp := tstringlist.Create;
tmp.Delimiter := ',';
for i := 0 to positions.count - 1 do
begin
tmp.DelimitedText := Positions[i];
if TryStrToInt(trim(tmp[0]),rectarray[l].Left) and
TryStrToInt(trim(tmp[1]),rectarray[l].Top) and
TryStrToInt(trim(tmp[2]),rectarray[l].Right) and
TryStrToInt(trim(tmp[3]),rectarray[l].Bottom) then
inc(l);
end;
setlength(rectarray,l);
GetBitmaps(srcpng,rectarray,dstbitmaparray);
tmp.free;
end;
//extract the second png from the large one
procedure TForm1.btnExtractClick(Sender: TObject);
var
src: TPNGImage;
begin
src := TPNGImage.Create;
src.Assign(img.Picture.Graphic);
try
myPictures[0] := TBitmap.Create;
// ok transparency is lost here!
LoadBitmaps(src, ImageListAreas, myPictures);
imgExtract.Picture.Assign(myPictures[0]);
finally
FreeAndNil(src);
end;
end;
也许有人知道如何在不失去透明度的情况下做到这一点...... 任何帮助都非常感谢,但是如果没有第三方组件也许会很好......至少Gr32也可以。
最基本的问候,
小号!
答案 0 :(得分:1)
我不确定是否有任何尺寸限制,但你是否尝试过来自PngComponents的TPngCollection(我希望你在D2009 +上)。与TPngImageList相反,TPngCollection中的每个条目可以具有不同的大小。虽然你可能不需要它,但它可能会打破尺寸障碍。
嗯,不是没有第三方......
答案 1 :(得分:0)
您实际上是在构建自己的图像列表。也许您可以找到现有的ImageList代码并对其进行修改。如果您有Delphi源代码,那应该不难。可能只是扩展一些常量让它使用更大的图像。我看到DevExpress的TcxImageList允许你做自定义大小。我只是尝试了500x500它让我(虽然没有测试它,但我希望它有效)。 TMS还有一个ImageList,不确定功能(现在没有,现在就没有)。