人気ブログランキング | 話題のタグを見る

[Delphi] TShellListViewからエクスプローラへドラッグ&ドロップする

こんにチワワ。どーもボキです。

TShellListViewに表示したフォルダ・ファイルを、デスクトップ等にコピーできるようにするサンプル。
Asuyu HomepageDelFusa Floorのアーカイブ(Delphi Users’ Forum)を参考にした。
[Delphi] TShellListViewからエクスプローラへドラッグ&ドロップする_a0021757_0372278.gif



原因不明のトラブルが2つ。
一つ目は、1つのファイル・フォルダのドラッグではマウスアイコンが表示されない。
2つ以上を選択しドラッグした後は、1つのみの選択時もアイコンが表示される。

二つ目は、ドロップ終了後にEXEウィンドウにカーソルを戻すと高確率で起きる読み込み違反エラー。
ドラッグ&ドロップ処理が多重に動くためと思われるが、回避方法が分からない。

----------
15.05.09
DragOverイベントで、コントロールから外れたときにDragDropを実行させていたが、
それをやめることで読み込み違反エラーは回避(大幅頻度低下)できそう。以下ソースは修正済み。
unit Unit1;

interface

uses
Windows, Messages, SysUtils, Variants, Classes, Graphics, Controls, Forms,
Dialogs, ComCtrls, ShellCtrls, ExtCtrls, StdCtrls, ShlObj, ActiveX, ComObj;

type
TForm1 = class(TForm, IDropSource)
GroupBox1: TGroupBox;
ShellTreeView1: TShellTreeView;
Splitter1: TSplitter;
GroupBox2: TGroupBox;
ShellListView: TShellListView;
procedure ShellListViewMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);
procedure ShellListViewDragOver(Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean);
procedure ShellListViewEndDrag(Sender, Target: TObject; X, Y: Integer);
private
{ Private 宣言 }
FDragStartPos : TPoint;

//IDropSource のメソッド 以下2つ
function QueryContinueDrag(fEscapePressed: BOOL; grfKeyState: Longint): HResult; stdcall;
function GiveFeedback(dwEffect: Longint): HResult; stdcall;
public
{ Public 宣言 }
end;

var
Form1: TForm1;

implementation


{$R *.dfm}

{ TForm1 }


//ファイル名からIDataObjectインターフェイスを取得
function GetFileListDataObject(const Directory: String; Files : TStringList) : IDataObject;
type
PArrayOfPItemIDList = ^TArrayOfPItemIDList;
TArrayOfPItemIDList = array[0..0] of PItemIDList;
var
Malloc : IMalloc;
fDesktop,fTargetFolder : IShellFolder;
i : Integer;
Dirpidl : PItemIDList;
p : PArrayOfPItemIDList;
chEaten,dwAttributes: ULONG;
begin
Result := nil;
if Files.Count = 0 then exit;

OleCheck(SHGetMalloc(Malloc));
OleCheck(SHGetDesktopFolder(fDesktop));
OleCheck(fDesktop.ParseDisplayName(0, nil, PWideChar(WideString(Directory)), chEaten, Dirpidl, dwAttributes));
try
OleCheck(fDesktop.BindToObject(Dirpidl, nil, IShellFolder, fTargetFolder));
p := AllocMem(SizeOf(PItemIDList) * Files.Count);
try
for i := 0 to Files.Count -1 do begin
OleCheck(fTargetFolder.ParseDisplayName(0, nil, PWideChar(WideString(Files[i])), chEaten, p[i], dwAttributes));
end;
OleCheck(fTargetFolder.GetUIObjectOf(0, Files.Count, p[0], IDataObject, nil, Pointer(Result)));
finally
for i := 0 to Files.Count -1 do begin
if p[i] <> nil then Malloc.Free(p[i]);
end;
FreeMem(p);
end;
finally
Malloc.Free(Dirpidl);
end;
end;


//現在行われようとしている操作に応じてカーソルを設定できる
function TForm1.GiveFeedback(dwEffect : Longint): HResult;
begin
//デフォルトのカーソルを使う
Result := DRAGDROP_S_USEDEFAULTCURSORS;
end;


//ユーザーがドラッグ操作の中止を求めた場合 fEscapePressed や
//マウスの状態に応じてドラッグの継続の可否をOLEに通知
function TForm1.QueryContinueDrag(fEscapePressed: BOOL; grfKeyState: Integer): HResult;
begin
//ESCが押されたか、両方のボタンが押されている場合は中止する
if fEscapePressed or ((MK_LBUTTON or MK_RBUTTON) = (grfKeyState and (MK_LBUTTON or MK_RBUTTON))) then
Result := DRAGDROP_S_CANCEL

//マウスの左ボタンが離された場合はドロップ処理へ
else if (grfKeyState and MK_LBUTTON) = 0 then begin
Result := DRAGDROP_S_DROP;

//それ以外はD&D継続
end else begin
Result := S_OK;
end;
end;


// マウスダウンによる初期化
procedure TForm1.ShellListViewMouseDown(Sender: TObject; Button: TMouseButton; Shift: TShiftState; X, Y: Integer);
begin
if Button <> mbLeft then exit;
if TShellListView(Sender).SelCount = 0 then exit;

FDragStartPos := Point(X, Y);
end;


// ファイル/フォルダのドラッグ
procedure TForm1.ShellListViewDragOver(Sender, Source: TObject; X, Y: Integer; State: TDragState; var Accept: Boolean);
var
_STV : TCustomShellTreeView;
_SLV : TShellListView;
slst : TStringList;
i,dwEffect : Integer;
s,dpth : String;
DataObject : IDataObject;
begin
//if State <> dsDragLeave then exit; ※コントロールから離れた時のみとすると、メモリアクセスエラーが置きやすい。
if ((Abs(X - FDragStartPos.X) < Mouse.DragThreshold) and (Abs(Y - FDragStartPos.Y) < Mouse.DragThreshold)) then exit;

_SLV := TShellListView(Sender);
_STV := _SLV.ShellTreeView;
dpth := _STV.Path;
try
slst := TStringList.Create;
slst.Capacity := _SLV.SelCount;
for i := 0 to _SLV.Items.Count -1 do begin
if _SLV.Items[i].Selected then begin
s := _SLV.Folders[i].PathName;
slst.Add(Copy(s, Pos(dpth,s)+Length(dpth)+1,Length(s)))
end;
end;
//ファイル名からIDataObjectを取得
DataObject := GetFileListDataObject(dpth,slst);
finally
slst.Free;
end;

//OLEドラッグ&ドロップ開始
dwEffect := DROPEFFECT_NONE;
DoDragDrop(DataObject, Self, DROPEFFECT_COPY, dwEffect);
end;


// 表示更新
procedure TForm1.ShellListViewEndDrag(Sender, Target: TObject; X, Y: Integer);
begin
TShellListView(Sender).Refresh
end;


initialization
OleInitialize(nil);


finalization
OleUninitialize;


end.



by yozda | 2015-04-28 16:20 | プログラミング | Comments(0)