ishelllink设置all user
如何用Shell实现程序组快捷方式的添加 作者 bood E-mail地址 boodweb@263.net 关键词:Shell函数 COM (一)前言 曾经在《电脑编程与维护》看到过一篇用DDE实现在程序组添加项目的方法,但是MSDN上明确指出应该用更加先进的Shell函数来实现,因此笔者在MSDN上仔细查找,终于在一篇名为“SHORTCUT: A SampleThat Manipulates Shortcuts”的文章中发现了这种方法。由于笔者初学COM不久,若有不当之处,请一定指出,感激不尽! (二)预备知识 Windows的程序组中的内容,实际上就是一个特定目录下的一些文件夹和文件(一般在c:\Windows\Start Menu\Programs目录下,我们可以用Shell函数SHGetSpecialFolderPath以CSIDL_PROGRAMS参数获得程序组的存放目录),Windows根据这些内容动态创建开始菜单和其下的子菜单。其中的文件夹代表一个弹出菜单,而文件则是快捷方式(.lnk文件),因此要在程序组建立快捷方式,实际上就是要在这个特定目录下新建一些.lnk文件,所幸Windows为我们提供了这样的接口。(当然,你要自己研究.lnk的文件结构然后自己一个字节一个字节的填,我也不反对)这些接口函数就是所谓的Shell函数的一部分。 Shell函数实际上就是一些Windows内置COM对象接口的函数,因此要使用他们,就应当知道一些有关COM的细节,您可以看看潘爱民的《COM原理与应用》。而我们要用的接口是IShellLink(用来设置要建立的快捷方式的一些信息)以及IPersistFile(用来以文件形式保存快捷方式)接口(均属于ShellLink对象),首先我们可以用CoCreateInstance库函数创建一个名为ShellLink的COM对象,同时得到其中一个接口指针,再用QueryInterface查询另外一个接口。详细实现以及接口函数的使用请看下面的代码。 (三)代码 这是我的实现,注意由于用了COM库函数,事先要调用CoInitialize()初始化,而在程序退出前要调用CoUninitialize() //bAdd=1表示添加快捷方式,bAdd=0表示删除此快捷方式 //此例中快捷方式名称为“自动运行.lnk”,创建在启动组内 BOOL CTestDlg::SetAutoRun(BOOL bAdd) { HRESULT hres; IShellLink *psl;//IShellLink接口指针 BOOL bRet=FALSE; char pszDesPath[MAX_PATH];//创建的目标路径 char pszShortcutFile[MAX_PATH];//创建的源文件 ::GetModuleFileName(NULL,pszShortcutFile,MAX_PATH);//得到本程序路径 SHGetSpecialFolderPath(m_hWnd,pszDesPath,CSIDL_PROGRAMS,0);//程序组路径 strcat(pszDesPath,"\\启动\\自动运行.lnk");//启动组 if(!bAdd) return DeleteFile(pszDesPath); hres = CoCreateInstance (CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER, IID_IShellLink, (void **)&psl);//得到CLSID_ShellLink标识的COM对象的IShellLink接口 if (!SUCCEEDED(hres)) goto error; IPersistFile *ppf;//IPersistFile接口指针 //查询IPersistFile接口以进行快捷方式的存储操作 hres = psl->QueryInterface (IID_IPersistFile, (void **)&ppf); if (!SUCCEEDED (hres)) goto error; WORD wsz [MAX_PATH]; //Unicode字符串的缓冲地址 //为适应COM标准一定要用Unicode //设置源文件地址 hres = psl->SetPath (pszShortcutFile); if (! SUCCEEDED (hres)) goto error; //设置参数 hres = psl->SetArguments("/ArgumentsHere"); if (! SUCCEEDED (hres)) goto error; //设置快捷方式的描述 hres = psl->SetDescription ("Shortcut to ScreenColor"); if (! SUCCEEDED (hres)) goto error; //将ANSI字符串转换为Unicode字符串 MultiByteToWideChar (CP_ACP, 0, pszDesPath, -1, wsz, MAX_PATH); //调用Save方法进行存储 hres = ppf->Save (wsz, TRUE); if (! SUCCEEDED (hres)) goto error; bRet=TRUE; error: //释放接口 ppf->Release (); psl->Release (); return TRUE; }
SHGetSpecialFolderPath(m_hWnd,pszDesPath,CSIDL_PROGRAMS,0)获得的路径是:"C:\Documents and Settings\Administrator\「开始」菜单\程序", 我需要的路径是"C:\Documents and Settings\All Users\「开始」菜单\程序"怎办? 我不想认为的对字符串操作
#define CSIDL_COMMON_STARTMENU 0x0016 // All Users\Start Menu #define CSIDL_COMMON_PROGRAMS 0X0017 // All Users\Programs #define CSIDL_COMMON_STARTUP 0x0018 // All Users\Startup #define CSIDL_COMMON_DESKTOPDIRECTORY 0x0019 // All Users\Desktop SHGetSpecialFolderLocation 参数改变一下就是All User 的目录了 具体参数参看shlobj.h 里面
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
//根据zswang和"超级猛料"部分函数改编
//比放到注册表里麻烦多了
function GetSpecialFolderDir(const folderid:integer): String;
var
pidl: pItemIDList;
buffer: array[0..255] of char;
begin
//取指定的文件夹项目表
//CSIDL_STARTUP:启动
SHGetSpecialFolderLocation(application.Handle , folderid, pidl);
SHGetPathFromIDList(pidl, buffer); //转换成文件系统的路径
Result:=strpas(buffer);
end;
function CreateShellLink(mPath: string; mWorkingDirectory: string;
mFileName: WideString): Boolean; { 返回创建快捷方式是否成功 }
var
vShellLink: IShellLink;
vPersistFile: IPersistFile;
vUnKnown: IUnKnown;
begin
Result := True;
try
vUnKnown := CreateComObject(CLSID_ShellLink);
vShellLink := vUnKnown as IShellLink;
vPersistFile := vUnKnown as IPersistFile;
vShellLink.SetPath(PChar(mPath));
vShellLink.SetWorkingDirectory(PChar(mWorkingDirectory));
if ExtractFileExt(mFileName) <> '.lnk' then
mFileName := mFileName + '.lnk';
vPersistFile.Save(PWChar(mFileName), False);
except
Result := False;
end;
end;
function CreateLinkToStartUp(AppName, WorkingDirectory,
LinkFileName: String; CurrentUser: Boolean = True): Boolean;
begin
if CurrentUser then //CurrentUser
Result := CreateShellLink(AppName, WorkingDirectory,
GetSpecialFolderDir(CSIDL_STARTUP) + '' + LinkFileName)
else//All User
Result := CreateShellLink(AppName, WorkingDirectory,
GetSpecialFolderDir(CSIDL_COMMON_STARTUP) + '' + LinkFileName);
end;
//创建到当前用户的[启动]里,CreateLinkToStartUp中最后一个参数设置为True
//创建到所有用户的启动里,CreateLinkToStartUp中最后一个参数设置为False
//创建到别的用户里,可以修改一下,不过。。。这样做不好吧。。。
***********************************************************8
- // AUTORUNDlg.cpp : implementation file
- //
- #include "stdafx.h"
- #include "AUTORUN.h"
- #include "AUTORUNDlg.h"
- #include <shlobj.h>
- #ifdef _DEBUG
- #define new DEBUG_NEW
- #undef THIS_FILE
- static char THIS_FILE[] = __FILE__;
- #endif
- /////////////////////////////////////////////////////////////////////////////
- // CAUTORUNDlg dialog
- CAUTORUNDlg::CAUTORUNDlg(CWnd* pParent /*=NULL*/)
- : CDialog(CAUTORUNDlg::IDD, pParent)
- {
- //{{AFX_DATA_INIT(CAUTORUNDlg)
- m_run = _T("");
- m_user = _T("");
- //}}AFX_DATA_INIT
- // Note that LoadIcon does not require a subsequent DestroyIcon in Win32
- m_hIcon = AfxGetApp()->LoadIcon(IDR_MAINFRAME);
- }
- void CAUTORUNDlg::DoDataExchange(CDataExchange* pDX)
- {
- CDialog::DoDataExchange(pDX);
- //{{AFX_DATA_MAP(CAUTORUNDlg)
- DDX_Text(pDX, IDC_RUNEDIT, m_run);
- DDX_Text(pDX, IDC_USEREDIT, m_user);
- //}}AFX_DATA_MAP
- }
- BEGIN_MESSAGE_MAP(CAUTORUNDlg, CDialog)
- //{{AFX_MSG_MAP(CAUTORUNDlg)
- ON_WM_PAINT()
- ON_WM_QUERYDRAGICON()
- ON_BN_CLICKED(IDC_OK1, OnOk)
- ON_BN_CLICKED(IDC_CANCEL, OnCancel)
- ON_BN_CLICKED(IDC_BUTTON2, OnButton1)
- //}}AFX_MSG_MAP
- END_MESSAGE_MAP()
- /////////////////////////////////////////////////////////////////////////////
- // CAUTORUNDlg message handlers
- BOOL CAUTORUNDlg::OnInitDialog()
- {
- CDialog::OnInitDialog();
- // Set the icon for this dialog. The framework does this automatically
- // when the application's main window is not a dialog
- SetIcon(m_hIcon, TRUE); // Set big icon
- SetIcon(m_hIcon, FALSE); // Set small icon
- // TODO: Add extra initialization here
- CoInitialize(NULL); //初始化COM接口
- return TRUE; // return TRUE unless you set the focus to a control
- }
- // If you add a minimize button to your dialog, you will need the code below
- // to draw the icon. For MFC applications using the document/view model,
- // this is automatically done for you by the framework.
- void CAUTORUNDlg::OnPaint()
- {
- if (IsIconic())
- {
- CPaintDC dc(this); // device context for painting
- SendMessage(WM_ICONERASEBKGND, (WPARAM) dc.GetSafeHdc(), 0);
- // Center icon in client rectangle
- int cxIcon = GetSystemMetrics(SM_CXICON);
- int cyIcon = GetSystemMetrics(SM_CYICON);
- CRect rect;
- GetClientRect(&rect);
- int x = (rect.Width() - cxIcon + 1) / 2;
- int y = (rect.Height() - cyIcon + 1) / 2;
- // Draw the icon
- dc.DrawIcon(x, y, m_hIcon);
- }
- else
- {
- CDialog::OnPaint();
- }
- }
- // The system calls this to obtain the cursor to display while the user drags
- // the minimized window.
- HCURSOR CAUTORUNDlg::OnQueryDragIcon()
- {
- return (HCURSOR) m_hIcon;
- }
- CString FName;
- CString Name;
- CString m_sDesc = _T("快捷方式");
- CString m_sFile;
- void CAUTORUNDlg::OnOk()
- {
- // TODO: Add your control notification handler code here
- m_sFile=_T("c:\\"+Name+".lnk");
- if (FName=="")
- {
- MessageBox("你选择的文件为空", "IShellLink Test", MB_ICONINFORMATION);
- CDialog::OnCancel();
- }
- else
- {
- if(CreateLink((LPCSTR)m_run, (LPCSTR)m_sFile, (LPCSTR)m_sDesc) == S_OK)
- MessageBox("完成!", "IShellLink Test", MB_ICONINFORMATION);
- int nOk;
- char m_s[30];
- sprintf(m_s,m_sFile,0);
- int len;
- len=strlen(m_s);
- m_s[18]='\0';
- m_s[19]='\0';
- //设置目的路径
- char strDst[]="C:\\Documents and Settings\\Administrator\\「开始」菜单\\程序\\启动\0";
- char strTitle[]="File copying";
- //进度题头
- SHFILEOPSTRUCT FileOp;
- FileOp.hwnd=m_hWnd;
- FileOp.wFunc=FO_MOVE;
- //执行文件拷贝
- FileOp.pFrom=m_s; //设置源路径
- FileOp.pTo=strDst; //设置目的路径
- FileOp.fFlags=FOF_ALLOWUNDO;
- FileOp.hNameMappings=NULL;
- FileOp.lpszProgressTitle=strTitle;
- nOk=SHFileOperation(&FileOp);
- if(FileOp.fAnyOperationsAborted)
- TRACE("Operation was aborted!\n");
- // MessageBox("Done!", "IShellLink Test", MB_ICONINFORMATION);
- }
- }
- void CAUTORUNDlg::OnCancel()
- {
- // TODO: Add your control notification handler code here
- CDialog::OnCancel();
- }
- HRESULT CAUTORUNDlg::CreateLink(LPCSTR pszShortcutFile, LPCSTR pszLink,
- LPCSTR pszDesc)
- {
- HRESULT hres;
- IShellLink *psl; //定义一个ISHELLLINK对象
- //创建实例,如果创建失败,返回
- hres = CoCreateInstance (CLSID_ShellLink, NULL, CLSCTX_INPROC_SERVER,
- IID_IShellLink, (void **)&psl);
- if (SUCCEEDED (hres))
- {
- IPersistFile *ppf; // 定义IpersistFile对象,用来保存ISHELLLINK对象
- hres = psl->QueryInterface (IID_IPersistFile, (void **)&ppf);
- //从ISHELLLINK对象中获得IpersistFile对象的接口。
- if (SUCCEEDED (hres))
- {
- WORD wsz [MAX_PATH];// 定义Unicode字符串
- //使用ISHELLLINK的SETPATH方法设置快捷方式中的程序路径
- hres = psl->SetPath (pszShortcutFile);
- if (! SUCCEEDED (hres))
- AfxMessageBox ("SetPath failed!");
- //设置快捷方式的提示信息.
- hres = psl->SetDescription (pszDesc);
- if (! SUCCEEDED (hres))
- AfxMessageBox ("SetDescription failed!");
- MultiByteToWideChar (CP_ACP, 0, pszLink, -1, wsz, MAX_PATH);
- // 确保快捷方式路径由ANSI字符组成
- hres = ppf->Save (wsz, TRUE);//保存快捷方式
- if (! SUCCEEDED (hres))
- AfxMessageBox ("Save failed!");
- ppf->Release ();//释放ISHELLLINK对象
- }
- psl->Release ();//释放ISHELLLINK对象
- }
- return hres;
- }
- void CAUTORUNDlg::OnButton1()
- {
- // TODO: Add your control notification handler code here
- CFileDialog dlg(TRUE, NULL, NULL, OFN_HIDEREADONLY | OFN_OVERWRITEPROMPT,
- "All files|*.*||");
- dlg.DoModal();
- FName=dlg.GetPathName(); // 取文件名全称,包括完整路径
- Name=dlg.GetFileName(); //取文件名全称
- m_run=FName;
- UpdateData(FALSE);
- }
**************************************************************************************************