Список изображений с иконками альфа-смеси теряет прозрачность - PullRequest
5 голосов
/ 30 марта 2012

Вот (более или менее) связанный вопрос: Delphi - заполнение списка изображений значками во время выполнения «уничтожает» прозрачность .

Я проверял @TOndrej ответ . Но, похоже, мне нужно включить визуальные стили (XP Manifest), чтобы это работало (будут использоваться общие элементы управления Windows версии 6.0 - чего я сейчас не хочу). Я добавляю значки во время выполнения через ExtractIconEx и ImageList_AddIcon.

Видимо, для установки ImageList.Handle для использования дескриптора System Image-List, не не требует XP Manifest. так что даже старая программа, которую я написал в D3, правильно отображается с иконками в альфа-смешении, когда я использую список образов системы для отображения списка файлов (с TListView).

Я бродил Что особенного в системном списке образов и как он создается, так что он поддерживает альфа-смешивание во всех случаях? Я не могу понять это. Вот пример кода:

unit Unit1;

interface

uses
  Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
  Dialogs, Menus, ImgList, StdCtrls, ShellAPI, ExtCtrls, Commctrl;

type
  TForm1 = class(TForm)
    ImageList1: TImageList;
    PopupMenu1: TPopupMenu;
    MenuItem1: TMenuItem;
    Button1: TButton;
    Button2: TButton;
    procedure FormCreate(Sender: TObject);
    procedure Button1Click(Sender: TObject);
    procedure Button2Click(Sender: TObject);
  private
    FileName: string;
  end;

var
  Form1: TForm1;

implementation

{$R *.dfm}
// {$R WindowsXP.res}

procedure TForm1.FormCreate(Sender: TObject);
begin
  PopupMenu1.Images := ImageList1;
  FileName := 'C:\Program Files\Mozilla Firefox\firefox.exe';
end;

procedure TForm1.Button1Click(Sender: TObject);
var
  IconPath: string;
  IconIndex: Integer;
  hIconLarge, hIconSmall: HICON;
begin
  IconPath := FileName;
  IconIndex := 0; // index can be other than 0

  ExtractIconEx(PChar(IconPath), IconIndex, hIconLarge, hIconSmall, 1);

  Self.Refresh; // erase form
  DrawIconEx(Canvas.Handle, 10, 10, hIconSmall, 0, 16, 16, 0,
    DI_IMAGE or DI_MASK); // this will draw ok on the form

  // ImageList1.DrawingStyle := dsTransparent;
  ImageList1.Handle := ImageList_Create(ImageList1.Width, ImageList1.Height,
    {ILC_COLORDDB} ILC_COLOR32 or ILC_MASK, 0, ImageList1.AllocBy);
  ImageList_AddIcon(ImageList1.Handle, hIconSmall);

  MenuItem1.ImageIndex := 0;

  DestroyIcon(hIconSmall);
  DestroyIcon(hIconLarge);

  PopupMenu1.Popup(Mouse.CursorPos.X, Mouse.CursorPos.Y);
end;

procedure TForm1.Button2Click(Sender: TObject);
// using sys image-list will work with or without Manifest
type
  DWORD_PTR = DWORD;
var
  ShFileINfo :TShFileInfo;
  SysImageList: DWORD_PTR;
  FileName: string;
begin
  SysImageList := ShGetFileInfo(nil, 0, ShFileInfo, SizeOf(ShFileInfo),
    SHGFI_SYSICONINDEX OR SHGFI_SMALLICON);

  if SysImageList = 0 then Exit;
  ImageList1.Handle := SysImageList;
  ImageList1.ShareImages := True;

  if ShGetFileInfo(PChar(FileName), 0, ShFileInfo, SizeOf(ShFileInfo),
    SHGFI_SYSICONINDEX OR SHGFI_ICON OR SHGFI_SMALLICON) <> 0 then
  begin
    MenuItem1.ImageIndex := ShFileInfo.IIcon;
    Self.Refresh; // erase form
    DrawIconEx(Canvas.Handle, 10, 10, ShFileInfo.hIcon, 0, 16, 16, 0,
      DI_IMAGE or DI_MASK);
    DestroyIcon(ShFileInfo.hIcon); // todo: do I need to destroy here? 

    PopupMenu1.Popup(Mouse.CursorPos.X, Mouse.CursorPos.Y);
  end;      
end;

end.

Визуальные стили Отключено :

enter image description here

Визуальные стили Включено :

enter image description here


Обходной путь - использовать промежуточный класс или подкласс TImageList и переопределить DoDraw , как показано здесь , но я действительно хочу знать, как создать мой список изображений, такой же, как системный список изображений.

Примечание: я знаю о TPngImageList и не хочу использовать его в этом случае.


Edit: @ Ответ Дэвида (и комментарии) были точными:

Вам придется явно ссылаться на ImageList_Create (v6), потому что в противном случае он неявно связан во время загрузки модуля и будет привязан к v5.8

Пример кода (без использования API контекста активации):

function ImageList_Create_V6(CX, CY: Integer; Flags: UINT; Initial, Grow: Integer): HIMAGELIST;
var
  h: HMODULE;
  _ImageList_Create: function(CX, CY: Integer; Flags: UINT;
    Initial, Grow: Integer): HIMAGELIST; stdcall;
begin
  // TODO: find comctl32.dll v6 path programmatically
  h := LoadLibrary('C:\WINDOWS\WinSxS\x86_Microsoft.Windows.Common-Controls_6595b64144ccf1df_6.0.2600.5512_x-ww_35d4ce83\comctl32.dll');
  if h <> 0 then
  try
    _ImageList_Create := GetProcAddress(h, 'ImageList_Create');
    if Assigned(_ImageList_Create) then
      Result := _ImageList_Create(CX, CY, Flags, Initial, Grow);
  finally
    FreeLibrary(h);
  end;
end;

procedure TForm1.Button1Click(Sender: TObject);
begin
  ...
  ImageList1.Handle := ImageList_Create_V6(ImageList1.Width, ImageList1.Height,
    ILC_COLOR32 or ILC_MASK, 0, ImageList1.AllocBy);
  ...
end;

Edi2: Пример кода от @ David , который показывает, как это делается правильно с помощью API контекста активации.

1 Ответ

5 голосов
/ 30 марта 2012

Существует две версии элементов управления списком изображений.Версия v5.8 и версия v6.Список образов системы является общим компонентом, принадлежащим системе, и использует версию v6.Это не особенное, просто список изображений v6.В вашем приложении ваш список изображений v5.8 или v6, в зависимости от того, включаете ли вы манифест.Но системный список изображений всегда v6.

Я не знаю, почему вы не хотите использовать общие элементы управления v6 в своем приложении.Но с этим ограничением вы могли бы использовать API контекста активации для локального использования общих элементов управления v6, пока вы создаете свой список изображений.Это решило бы вашу проблему и оставило бы остальную часть вашего приложения с общими элементами управления v5.8.

...