i.e. it is essentially a .ini file, organized into different sections. If you ever wanted to retrieve these "Favourites" to show in your application you would need to open this file, and interrogate it for the relevant information. Thankfully, there are some PInvoke calls that can be used to make this task a little easier. This CodeBank submission aims to show how you can enumerate all your Internet Explorer Favourites and display them in your application, as well as how you can add new ones.
Internet Explorer Favourite
An Internet Explorer Favourite is essentially a file on your file system, with a .url extension. The name of the file, without the extenion, is used as the label for the Favourite within Internet Explorer, and the URL, contained within the file, is what is used to navigate to that Favourite. There are additional items in the .url file, including the icon to use for the shortcut, but these will not be discussed in this thread.
PInvoke Calls
Two PInvokes are going to be used to get/set the information that we need.
The first is GetPrivateProfileString, you can find information about this here:
With that in mind, the following code can be used to get all of the Favourites (including Favourites contained within nested folders) and put them into a TreeView (assuming you have a TreeView on your form called:
Code:
private uint result;
private StringBuilder sb = new StringBuilder(500);
private void GetFavouritesForDirectory(DirectoryInfo di, TreeNode dirNode)
{
foreach (FileInfo fileInfo in di.GetFiles())
{
result = GetPrivateProfileString("InternetShortcut", "URL", "", sb, (uint)sb.Capacity, fileInfo.FullName);
if (result > 0)
{
FavouriteTreeNode myFav = new FavouriteTreeNode();
myFav.Text = Path.GetFileNameWithoutExtension(fileInfo.FullName);
myFav.Url = new Uri(sb.ToString());
myFav.DirectoryPath = fileInfo.FullName;
if (dirNode == null)
{
FavouritesTreeView.Nodes.Add(myFav);
}
else
{
dirNode.Nodes.Add(myFav);
}
}
}
}
private void GetFavourites()
{
FavouritesTreeView.Nodes.Clear();
foreach(string dirName in Directory.GetDirectories(Environment.GetFolderPath(Environment.SpecialFolder.Favorites)))
{
DirectoryInfo dirInfo = new DirectoryInfo(dirName);
TreeNode newNode = new TreeNode();
newNode.Text = dirInfo.Name;
newNode.Tag = dirInfo.FullName;
FavouritesTreeView.Nodes.Add(newNode);
newNode.Nodes.Add("*");
}
GetFavouritesForDirectory(new DirectoryInfo(Environment.GetFolderPath(Environment.SpecialFolder.Favorites)), null);
}
The above code makes use of a custom class called FavouriteTreeNode. This is a simple class which inherits from TreeNode adding on a couple of specific properties to fit our needs:
Code:
public class FavouriteTreeNode : TreeNode
{
public Uri Url { get; set; }
public string DirectoryPath { get; set; }
public FavouriteTreeNode()
{
}
public FavouriteTreeNode(string displayText, Uri url)
{
this.Text = displayText;
this.Url = url;
}
}
With these two methods defined, it is a simple matter of calling GetFavourites on the Load Event of your form.
Due to the fact that there could literally be hundreds of Favourites stored on the user's machine, in various different nested folders, I have decided to only display the Favourites when required, rather than waste time loading them when they aren't needed. To that end, I have used the BeforeExpand event of the TreeView to decide whether or not I have to go and find some more Favourites for the currently selected Node.
To add a favourite, it's a simple call to the other PInvoke call, WritePrivateProfileString, providing the information for the new path to the Favourites file, as well as the Url that you want to save with it. Here is an example:
Code:
private void SaveFavouriteButton_Click(object sender, EventArgs e)
{
if (FavouriteDisplayNameTextBox.Text == string.Empty)
{
throw new ArgumentException("You must provide a Display Name for the Favourite");
}
if (FavouriteUrlTextBox.Text == string.Empty)
{
throw new ArgumentException("You must provide a Url for the Favourite");
}
Uri favouriteUri;
if(!Uri.TryCreate(FavouriteUrlTextBox.Text, UriKind.RelativeOrAbsolute, out favouriteUri))
{
throw new ArgumentException("Please provide a valid url");
}
string favouriteFileName = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.Favorites), FavouriteDisplayNameTextBox.Text + ".url");
WritePrivateProfileString("InternetShortcut", "URL", favouriteUri.ToString(), favouriteFileName);
GetFavourites();
}
Attached is a couple working sample of the above code. Let me know if you have any questions.