Click here to Skip to main content
15,860,943 members
Articles / Programming Languages / C++
Article

How to create and resolve a shortcut

Rate me:
Please Sign up or sign in to vote.
4.62/5 (36 votes)
6 Oct 2003CPOL1 min read 299.4K   73   83
This article presents 2 ready-to-use functions to create and resolve a shortcut.

Introduction

Shortcut is a file with an extension .lnk. Each of these files contain a special COM object that points to another file. Usually, when you try to open a .lnk file, the system opens a file which this shortcut points to.

Let's do the following experiment. Create a text file (file with an extension .txt) somewhere. Then create a shortcut pointing to this file (send me a private e-mail if you do not know how to create a shortcut manually). Then try to open the shortcut with Microsoft Word, using File->Open command and select just created shortcut. MS Word will do it correctly: it will open the text file that is pointed by this shortcut. Now do the same with Notepad. Instead of the text file content, you will see garbage. It means that Notepad has no idea how to deal with shortcuts.

So we came to a conclusion: in Windows a program should have a built-in support for shortcuts in order to handle them correctly.

In this article I will show how to do it. I will present 2 functions: how to create and resolve the shortcut. The code is well commented (in my opinion) and self-explanatory.

Code

/**********************************************************************
* Function......: CreateShortcut
* Parameters....: lpszFileName - string that specifies a valid file name
*          lpszDesc - string that specifies a description for a 
                             shortcut
*          lpszShortcutPath - string that specifies a path and 
                                     file name of a shortcut
* Returns.......: S_OK on success, error code on failure
* Description...: Creates a Shell link object (shortcut)
**********************************************************************/
HRESULT CreateShortcut(/*in*/ LPCTSTR lpszFileName, 
                    /*in*/ LPCTSTR lpszDesc, 
                    /*in*/ LPCTSTR lpszShortcutPath)
{
    HRESULT hRes = E_FAIL;
    DWORD dwRet = 0;
    CComPtr<IShellLink> ipShellLink;
        // buffer that receives the null-terminated string 
        // for the drive and path
    TCHAR szPath[MAX_PATH];    
        // buffer that receives the address of the final 
        //file name component in the path
    LPTSTR lpszFilePart;    
    WCHAR wszTemp[MAX_PATH];
        
    // Retrieve the full path and file name of a specified file
    dwRet = GetFullPathName(lpszFileName, 
                       sizeof(szPath) / sizeof(TCHAR), 
                       szPath, &lpszFilePart);
    if (!dwRet)                                        
        return hRes;

    // Get a pointer to the IShellLink interface
    hRes = CoCreateInstance(CLSID_ShellLink,
                            NULL, 
                            CLSCTX_INPROC_SERVER,
                            IID_IShellLink,
                            (void**)&ipShellLink);

    if (SUCCEEDED(hRes))
    {
        // Get a pointer to the IPersistFile interface
        CComQIPtr<IPersistFile> ipPersistFile(ipShellLink);

        // Set the path to the shortcut target and add the description
        hRes = ipShellLink->SetPath(szPath);
        if (FAILED(hRes))
            return hRes;

        hRes = ipShellLink->SetDescription(lpszDesc);
        if (FAILED(hRes))
            return hRes;

        // IPersistFile is using LPCOLESTR, so make sure 
                // that the string is Unicode
#if !defined _UNICODE
        MultiByteToWideChar(CP_ACP, 0, 
                       lpszShortcutPath, -1, wszTemp, MAX_PATH);
#else
        wcsncpy(wszTemp, lpszShortcutPath, MAX_PATH);
#endif

        // Write the shortcut to disk
        hRes = ipPersistFile->Save(wszTemp, TRUE);
    }

    return hRes;
}

/*********************************************************************
* Function......: ResolveShortcut
* Parameters....: lpszShortcutPath - string that specifies a path 
                                     and file name of a shortcut
*          lpszFilePath - string that will contain a file name
* Returns.......: S_OK on success, error code on failure
* Description...: Resolves a Shell link object (shortcut)
*********************************************************************/
HRESULT ResolveShortcut(/*in*/ LPCTSTR lpszShortcutPath,
                        /*out*/ LPTSTR lpszFilePath)
{
    HRESULT hRes = E_FAIL;
    CComPtr<IShellLink> ipShellLink;
        // buffer that receives the null-terminated string 
        // for the drive and path
    TCHAR szPath[MAX_PATH];     
        // buffer that receives the null-terminated 
        // string for the description
    TCHAR szDesc[MAX_PATH]; 
        // structure that receives the information about the shortcut
    WIN32_FIND_DATA wfd;    
    WCHAR wszTemp[MAX_PATH];

    lpszFilePath[0] = '\0';

    // Get a pointer to the IShellLink interface
    hRes = CoCreateInstance(CLSID_ShellLink,
                            NULL, 
                            CLSCTX_INPROC_SERVER,
                            IID_IShellLink,
                            (void**)&ipShellLink); 

    if (SUCCEEDED(hRes)) 
    { 
        // Get a pointer to the IPersistFile interface
        CComQIPtr<IPersistFile> ipPersistFile(ipShellLink);

        // IPersistFile is using LPCOLESTR, 
                // so make sure that the string is Unicode
#if !defined _UNICODE
        MultiByteToWideChar(CP_ACP, 0, lpszShortcutPath,
                                       -1, wszTemp, MAX_PATH);
#else
        wcsncpy(wszTemp, lpszShortcutPath, MAX_PATH);
#endif

        // Open the shortcut file and initialize it from its contents
        hRes = ipPersistFile->Load(wszTemp, STGM_READ); 
        if (SUCCEEDED(hRes)) 
        {
            // Try to find the target of a shortcut, 
                        // even if it has been moved or renamed
            hRes = ipShellLink->Resolve(NULL, SLR_UPDATE); 
            if (SUCCEEDED(hRes)) 
            {
                // Get the path to the shortcut target
                hRes = ipShellLink->GetPath(szPath, 
                                     MAX_PATH, &wfd, SLGP_RAWPATH); 
                if (FAILED(hRes))
                    return hRes;

                // Get the description of the target
                hRes = ipShellLink->GetDescription(szDesc,
                                             MAX_PATH); 
                if (FAILED(hRes))
                    return hRes;

                lstrcpyn(lpszFilePath, szPath, MAX_PATH); 
            } 
        } 
    } 

    return hRes;
}

Using the code

The following code will show how you may use these functions.

void HowToCreateShortcut()
{
    LPCTSTR lpszFileName = _T("C:\\Work\\Window.exe");
    LPCTSTR lpszShortcutDesc = _T("Anything can go here");
    LPCTSTR lpszShortcutPath = 
_T("C:\\Documents and Settings\\Administrator\\Desktop\\Sample Shortcut.lnk");

    CreateShortcut(lpszFileName, lpszShortcutDesc, lpszShortcutPath);
}

void HowToResolveShortcut()
{
    LPCTSTR lpszShortcutPath = 
_T("C:\\Documents and Settings\\Administrator\\Desktop\\Sample Shortcut.lnk");
    TCHAR szFilePath[MAX_PATH];

    ResolveShortcut(lpszShortcutPath, szFilePath);
}

That's it.

License

This article, along with any associated source code and files, is licensed under The Code Project Open License (CPOL)


Written By
Web Developer
United States United States
This member has not yet provided a Biography. Assume it's interesting and varied, and probably something to do with programming.

Comments and Discussions

 
GeneralRe: get info from MSOffice shortcut Pin
Igor Vigdorchik4-Feb-04 13:47
Igor Vigdorchik4-Feb-04 13:47 
GeneralRe: get info from MSOffice shortcut Pin
Sergey Kovrizhkin4-Feb-04 20:19
Sergey Kovrizhkin4-Feb-04 20:19 
GeneralRe: get info from MSOffice shortcut Pin
Shaun Harrington28-Jun-06 14:27
Shaun Harrington28-Jun-06 14:27 
GeneralRe: get info from MSOffice shortcut [modified] Pin
Shaun Harrington28-Jun-06 15:12
Shaun Harrington28-Jun-06 15:12 
GeneralRe: get info from MSOffice shortcut Pin
Shaun Harrington28-Jun-06 15:15
Shaun Harrington28-Jun-06 15:15 
GeneralRe: get info from MSOffice shortcut Pin
Igor Vigdorchik28-Jun-06 16:50
Igor Vigdorchik28-Jun-06 16:50 
GeneralRe: get info from MSOffice shortcut Pin
Shaun Harrington30-Jun-06 4:07
Shaun Harrington30-Jun-06 4:07 
GeneralRe: get info from MSOffice shortcut Pin
Igor Vigdorchik6-Jul-06 11:14
Igor Vigdorchik6-Jul-06 11:14 
Shaun,
check this article[^] out. Could be useful.
GeneralRe: get info from MSOffice shortcut Pin
Shaun Harrington20-Jul-06 23:46
Shaun Harrington20-Jul-06 23:46 
GeneralRe: get info from MSOffice shortcut Pin
BillJam1112-Jul-06 6:04
BillJam1112-Jul-06 6:04 
GeneralRe: get info from MSOffice shortcut Pin
Radu Sebastian LAZIN14-Aug-06 12:09
Radu Sebastian LAZIN14-Aug-06 12:09 
GeneralCreate Desktop Shortcut Pin
scar_me28-Jan-04 18:07
scar_me28-Jan-04 18:07 
QuestionThere's a way to do this procedures in C? Pin
Chulips16-Jan-04 0:39
Chulips16-Jan-04 0:39 
AnswerRe: There's a way to do this procedures in C? Pin
Igor Vigdorchik19-Jan-04 17:30
Igor Vigdorchik19-Jan-04 17:30 
GeneralRe: There's a way to do this procedures in C? Pin
Chulips21-Jan-04 9:15
Chulips21-Jan-04 9:15 
GeneralRe: There's a way to do this procedures in C? Pin
Igor Vigdorchik21-Jan-04 12:47
Igor Vigdorchik21-Jan-04 12:47 
GeneralRe: There's a way to do this procedures in C? Pin
John T5-Apr-04 8:26
John T5-Apr-04 8:26 
GeneralRe: There's a way to do this procedures in C? Pin
John T5-Apr-04 8:28
John T5-Apr-04 8:28 
GeneralRe: There's a way to do this procedures in C? Pin
Mike D.1-Jul-04 11:34
Mike D.1-Jul-04 11:34 
Generalshortcut to network connection Pin
Simon Lao17-Oct-03 11:35
Simon Lao17-Oct-03 11:35 
GeneralRe: shortcut to network connection Pin
Igor Vigdorchik17-Oct-03 18:55
Igor Vigdorchik17-Oct-03 18:55 
GeneralRe: shortcut to network connection Pin
dberindei20-Oct-03 0:19
dberindei20-Oct-03 0:19 
GeneralRe: shortcut to network connection Pin
Igor Vigdorchik8-Nov-03 11:09
Igor Vigdorchik8-Nov-03 11:09 
GeneralRe: shortcut to network connection Pin
ohadhawk7-Jan-04 9:41
ohadhawk7-Jan-04 9:41 
GeneralRe: shortcut to network connection Pin
ohadhawk7-Jan-04 9:43
ohadhawk7-Jan-04 9:43 

General General    News News    Suggestion Suggestion    Question Question    Bug Bug    Answer Answer    Joke Joke    Praise Praise    Rant Rant    Admin Admin   

Use Ctrl+Left/Right to switch messages, Ctrl+Up/Down to switch threads, Ctrl+Shift+Left/Right to switch pages.