Got stuck with large amounts of files
I'm talking about Visual Studio 2005. C#, C++, VB.net and so on share the same IDE. I don't remember how to debug foreign processes with Visual C++ 6.0, but it is possible, too.
Crunching for Fab36_Folding-Division at Folding@Home. Join Fab36/Fab30! - Folding@Home and BOINC
Boycott DRM! Boycott HDCP!
Boycott DRM! Boycott HDCP!
If I understand the pIDL and SHITEMID structures right, I could do this:
But it prints garbage. What am I doing wrong?
Code: Select all
Dim pIDL As Long, pp As Long
Dim s%, i%
Dim b(1024) As Byte
Dim a$
SHGetFolderLocation Me.hWnd, CSIDL_DesktopDirectory, 0&, 0&, pIDL
If pIDL Then
pp = pIDL
Do
MoveMemory s, ByVal pp, 2 'get size of pIDL
If s Then
MoveMemory b(0), ByVal pp + 2, s - 2
pp = pp + s
a$ = ""
For i = 1 To s: a$ = a$ + Chr$(b(i)): Next
debug.print a$
End If
Loop Until s = 0
End If
SHFree pIDL
Nothing. I get this output, which is quite normal:
Code: Select all
PÃ OÃ
Crunching for Fab36_Folding-Division at Folding@Home. Join Fab36/Fab30! - Folding@Home and BOINC
Boycott DRM! Boycott HDCP!
Boycott DRM! Boycott HDCP!
No. You are supposed to use the Shell API.WoF wrote:Is this information structure documented somewhere?
Crunching for Fab36_Folding-Division at Folding@Home. Join Fab36/Fab30! - Folding@Home and BOINC
Boycott DRM! Boycott HDCP!
Boycott DRM! Boycott HDCP!
Well, I think it's better to not document this structure. The Shell API is relative abstract. You use COM interfaces for almost everything. IMHO this is a good design.WoF wrote:Oh. I see. That's Microsoft for you.
Documenting the layout of a pIDL wouldn't be possible anyway, because different namespaces use different layouts. If you're writing a shell namespace extension, you can store everything you like in the pIDL. The format is up to you. Isn't this freedom great?

Crunching for Fab36_Folding-Division at Folding@Home. Join Fab36/Fab30! - Folding@Home and BOINC
Boycott DRM! Boycott HDCP!
Boycott DRM! Boycott HDCP!
Sounds crazy. How would the Shell API know about the ID format I create for my namespace extension?
Ok. Maybe I'm still to ignorant on the matter of shell namespace and shellfolder stuff, but I would expect that there are some standards e.g. how to retrieve a name string or similar from an ID in a pIDL, but I can see there are different requirements for nonFS and FS objects, so the ID format may vary.
Maybe the filename sorting problem which is still waiting in the background will bring me back to pIDLs and shell objects.
Ok. Maybe I'm still to ignorant on the matter of shell namespace and shellfolder stuff, but I would expect that there are some standards e.g. how to retrieve a name string or similar from an ID in a pIDL, but I can see there are different requirements for nonFS and FS objects, so the ID format may vary.
Maybe the filename sorting problem which is still waiting in the background will bring me back to pIDLs and shell objects.

It doesn't and there's no need for it to do so. When writing a shell extension, you implement the shell interfaces like IShellFolder and so on. So if a shell browser calls e. g. IShellFolder::GetDisplayNameOf to retrieve the display name of a pIDL that identifies an item within your namespace, your code is called. An API function like SHGetPathFromIDList() uses the shell interfaces internally, too.WoF wrote:Sounds crazy. How would the Shell API know about the ID format I create for my namespace extension?
There aren't. The only standard about pIDLs is, that a pIDL points to an array of SHITEMID structures, the first 2 byte of such a SHITEMID structure hold its size and the last SHITEMID structure has a size of 0. Everything else is up to the namespace implementations.WoF wrote:Ok. Maybe I'm still to ignorant on the matter of shell namespace and shellfolder stuff, but I would expect that there are some standards e.g. how to retrieve a name string or similar from an ID in a pIDL
Name strings are retrieved through IShellFolder::GetDisplayNameOf. A namespace extension has to implement the IShellFolder interface for each (virtual) folder it wants to display. Of course you can write generic implementations.
File sorting is done through IShellFolder::CompareIDs.WoF wrote:Maybe the filename sorting problem which is still waiting in the background will bring me back to pIDLs and shell objects.
Crunching for Fab36_Folding-Division at Folding@Home. Join Fab36/Fab30! - Folding@Home and BOINC
Boycott DRM! Boycott HDCP!
Boycott DRM! Boycott HDCP!
That's what intrigues me. How should a compare routine know how a pIDL is greater or less than another one, when the format of the ID is somehow unknown.TiKu wrote:File sorting is done through IShellFolder::CompareIDs.
Ok. Ok. I forget there's that interface which has to return the name... so the IShellFolder::CompareIDs must somehow identify the ID to know whom to call...
But anyway I can't use this routine beause I don't have pIDLs. I only have the filenames in my ListView.
The implementation of IShellFolder, whose CompareIDs method you're calling, knows the format, because it created the pIDLs.WoF wrote:That's what intrigues me. How should a compare routine know how a pIDL is greater or less than another one, when the format of the ID is somehow unknown.TiKu wrote:File sorting is done through IShellFolder::CompareIDs.
Each folder-like shell item implements the IShellFolder interface. To get the Desktop's IShellFolder you use the SHGetDesktopFolder API function. Then you use this IShellFolder implementation's BindToObject method to browse to other folder-like shell items. BindToObject will give you a new IShellFolder implementation - that one of the folder whose pIDL you passed.
To enumerate the content of a folder-like shell item, you call IShellFolder::EnumObjects. This will give you an IEnumIDList implementation which you use to iterate through the sub-items.
To retrieve a shell item's name, you take the parent item's IShellFolder implementation and call its GetDisplayNameOf method.
To compare two shell items, you take the parent item's IShellFolder implementation and call its CompareIDs method.
To retrieve all columns that a folder-like shell item defines for its sub-items, you query its IShellFolder implementation for an IShellFolder2 implementation (or IShellDetails as a fallback) and call this implementation's GetDetailsOf method. You can also use IShellFolder2 to retrieve the search engines this folder implements for its sub-items.
I hope now it's more clear how the shell is working
TiKu
Crunching for Fab36_Folding-Division at Folding@Home. Join Fab36/Fab30! - Folding@Home and BOINC
Boycott DRM! Boycott HDCP!
Boycott DRM! Boycott HDCP!
Yes a little more. Seems quite a clever concept once you get deeper into it.
I halso had a look to your ExplorerClone program you kindly provided some time ago, to understand how sorting is done there.
I undertsand, you take two items which have to be compared, retrieve there pIDLs with ILFindLastID and call their parents CompareID. Clever. But it depends on using a listview which has custom sort facility...
I halso had a look to your ExplorerClone program you kindly provided some time ago, to understand how sorting is done there.
Code: Select all
Private Sub lvwFiles_CompareItems(ByVal firstItem As ExLVwLibUCtl.IListViewItem, ByVal secondItem As ExLVwLibUCtl.IListViewItem, result As ExLVwLibUCtl.CompareResultConstants)
Dim Flags As SHCIDSConstants
Dim pIDLToParent1 As Long
Dim pIDLToParent2 As Long
pIDLToParent1 = ILFindLastID(firstItem.ItemData)
pIDLToParent2 = ILFindLastID(secondItem.ItemData)
Flags = SHCIDSConstants.SHCIDS_ALLFIELDS
result = ISHFParent.CompareIDs(Flags, pIDLToParent1, pIDLToParent2)
End Sub
Almost. PIDLs are somewhat like paths, i. e. they usually start at the root (which is the Desktop), but can also be relative to an other IShellFolder implementation. So something like C:\Windows\system32 can be expressed as pIDL as well as Windows\system32. The only thing that can't be done is something like ..\Windows\system32.WoF wrote:I undertsand, you take two items which have to be compared, retrieve there pIDLs with ILFindLastID
In the ExplorerClone sample I store the item's fully qualified pIDL (i. e. starting at the Desktop) in the ItemData property. The pIDLs that you pass to IShellFolder::CompareIDs must be relative to this IShellFolder implementation. So I use ILFindLastID to retrieve the address of the last SHITEMID structure (before the terminating one which has a size of 0) within the fully qualified pIDL. The result is still a pIDL - it points to an array of SHITEMID structures and is terminated by two NULL byte.
Right.WoF wrote:But it depends on using a listview which has custom sort facility...
Crunching for Fab36_Folding-Division at Folding@Home. Join Fab36/Fab30! - Folding@Home and BOINC
Boycott DRM! Boycott HDCP!
Boycott DRM! Boycott HDCP!
I see. So the last ID is that of the file itself and the parent, which in this case is a folder, provides the CompareID procedure apropriate for files.
I'm afraid the chances for me to make use of this intelligent sorting are remote.
I remember there were a couple of problems which made it quite improbable for me to use your ExplorerListView...
I'm afraid the chances for me to make use of this intelligent sorting are remote.
I remember there were a couple of problems which made it quite improbable for me to use your ExplorerListView...