2009-06-24 3 views
6

내 응용 프로그램에는 시작 프로그램 바로 가기가 있습니다 (사용자별로 아님). 설치 프로그램을 진행하는 동안 바로 가기를 만들 수 있습니다. 내 문제는 나중에 사용자가 제거한 다음 다시 사용하려고 할 때 발생합니다. 다른 말로하면 RunOnStartup (시작 잉크를 삭제 함)을 끄고 나중에 시작할 때 실행되도록 결정하여 기본 설정으로 돌아가서 다시 사용할 수 있도록 결정합니다.내 응용 프로그램의 바로 가기 (.lnk)를 시작 폴더에 프로그래밍 방식으로 .NET/C에 추가하려면 어떻게 만드십시오.

분명히, 이것은 바로 가기를 만드는 네이티브 방식이 아니라는 점에서 꽤 흔한 .NET입니다. 그러나, 아주 좋은 해결책을 찾지 못했습니다.

솔루션은 내가 고려/발견했습니다

  • 는 오히려 다음 바로 가기를 만들 수 있습니다. 그냥 하나 복사하십시오. 이것은 좋은 해결책 일 수 있습니다. 나는 Start Menu 링크가 있다는 것에 의존 할 수 없다. 하지만 아마도 하나 만들고 프로그램 디렉토리에 보관할 수있을 것 같아요 ...이 설치 프로그램에서 설치 시간에 지정된 적절한 경로로 바로 가기를 만들어야 문제를 이동합니다.

  • 무엇을 this other stackoverflow answer으로하고 COM 래퍼 개체를 사용하십시오. COM을 피하고 싶습니다. 또한 2003 년에 작성되었습니다. 따라서 비스타를 얼마나 잘 지원할 지 잘 모르겠습니다. 나는 그것에게 탄을 줄 것이다. 그러나 비스타 박스를 편리하게 가지지 마라.

  • 대신 레지스트리를 사용하십시오. 이것은 내가 현재 어떻게하고 있지만 ... 비스타에 관한 문제에 부딪치게됩니다. Startup Menu 단축키가이 작업을 수행하는 적절한 방법이라는 일반적인 합의가있는 것 같습니다. 이것이 나의 목표입니다.

또한, 나는 일반 사용자 (안 관리자)이이 환경 설정을 변경하려고 할 경우를 처리해야합니다. 이 경우 정상적으로 실패해야합니다. 또는 비스타의 경우 사용자가 관리자 비밀번호를 입력하여 관리자 보안 토큰을 얻을 수 있도록 허용해야합니다. 이미이 사건을 적절하게 고려한 답변은 굉장 할 것입니다.

이 주제가 이미 다루어지면 사과드립니다. 나는 게시하기 전에 주위를 수색했다.

업데이트 : 설치 관리자가 만든 바로 가기를 복사하는 것이 가장 좋은 방법입니다. 일단 코드가 끝나면 코드를 게시 할 것입니다. a) Environment.GetSpecialFolder가 해결 된 StartMenu에 대한 참조를 가지지 않는 장애물이 있습니다 ...하지만 이제는 파일을 적절한 위치. 이 주제에 대한 새로운 stackoverflow 질문을 만들었습니다. How can I copy a file as a "Standard User" in Vista (ie "An Administrative Choice Application") by prompting user for admin credentials?

+1

llLink는 Vista에서 잘 작동합니다. AFAIK은 바로 가기를 만들고 해결하는 가장 좋은 방법입니다. 아마도 IWshRuntimeLibrary.dll에 대한 interop assmbly를 사용하여이 작업을 수행 할 수도 있습니다. 즉, 추가 interop 어셈블리를 출하하게 될 것입니다. –

답변

8

, 적절한 솔루션은 설치시 프로그램 파일에 폴더를 바로 가기를 설치 한 다음 시작 폴더에 .LNK를 복사하는 것입니다. 바로 가기를 만드는 것은 더욱 어렵습니다.

아래의 코드는 다음을 수행

  • 그것은 당신을 위해 모든 사용자 시작 폴더의 경로를 가져옵니다. 환경.GetSpecialFolder는 상당히 제한되어 있으며이 폴더에 대한 참조가 없으므로 시스템 호출을해야합니다.
  • 바로 가기를 복사하고 제거하는 방법이 있습니다.

궁극적으로 응용 프로그램을 실행하는 사용자가 일반 사용자 인 경우 자격 증명을 입력하라는 메시지가 표시되면 정상적으로 처리되었는지도 확인하고 싶습니다. 제목에 게시물을 작성 했으므로 중요한 내용인지 여기에서 확인하십시오. 나는이 솔루션을 작성하는 동안 How can I copy a file as a "Standard User" in Vista (ie "An Administrative Choice Application") by prompting user for admin credentials?

using System; 
using System.Collections.Generic; 
using System.Text; 
using System.IO; 
using System.Security.Principal; 
using System.Windows.Forms; 
using System.Runtime.InteropServices; 

namespace BR.Util 
{ 
public class StartupLinkUtil 
{ 
    [DllImport("shell32.dll")] 
    static extern bool SHGetSpecialFolderPath(IntPtr hwndOwner, [Out] StringBuilder lpszPath, int nFolder, bool fCreate); 

    public static string getAllUsersStartupFolder() 
    { 
     StringBuilder path = new StringBuilder(200); 
     SHGetSpecialFolderPath(IntPtr.Zero, path, CSIDL_COMMON_STARTUP, false); 
     return path.ToString(); 
    } 


    private static string getStartupShortcutFilename() 
    { 
     return Path.Combine(getAllUsersStartupFolder(), Application.ProductName) + ".lnk"; 
    } 

    public static bool CopyShortcutToAllUsersStartupFolder(string pShortcutName) 
    { 
     bool retVal = false; 
     FileInfo shortcutFile = new FileInfo(pShortcutName); 
     FileInfo destination = new FileInfo(getStartupShortcutFilename()); 

     if (destination.Exists) 
     { 
      // Don't do anything file already exists. -- Potentially overwrite? 
     } 
     else if (!shortcutFile.Exists) 
     { 
      MessageBox.Show("Unable to RunOnStartup because '" + pShortcutName + "' can't be found. Was this application installed properly?"); 
     } 
     else 
     { 
      retVal = copyFile(shortcutFile, destination); 
     } 
     return retVal; 
    } 

    public static bool doesShortcutExistInAllUsersStartupFolder() 
    { 
     return File.Exists(getStartupShortcutFilename()); 
    } 

    public static bool RemoveShortcutFromAllUsersStartupFolder() { 
     bool retVal = false; 
     string path = Path.Combine(getAllUsersStartupFolder(), Application.ProductName) + ".lnk"; 
     if(File.Exists(path)) { 
      try 
      { 
       File.Delete(path); 
       retVal = true; 
      } 
      catch (Exception ex) 
      { 
       MessageBox.Show(string.Format("Unable to remove this application from the Startup list. Administrative privledges are required to perform this operation.\n\nDetails: SecurityException: {0}", ex.Message), "Update Startup Mode", MessageBoxButtons.OK, MessageBoxIcon.Error); 
      } 
     } 
     return retVal; 
    } 

    // TODO: Test this in vista to see if it prompts for credentials. 
    public static bool copyFile(FileInfo pSource, FileInfo pDestination) 
    { 
     bool retVal = false; 

     try 
     { 
      File.Copy(pSource.FullName, pDestination.FullName); 
      //MessageBox.Show("File has successfully been added.", "Copy File", MessageBoxButtons.OK, MessageBoxIcon.Information); 
      retVal = true; 
     } 
     catch (System.Security.SecurityException secEx) 
     { 
      MessageBox.Show(string.Format("SecurityException: {0}", secEx.Message), "Copy File", MessageBoxButtons.OK, MessageBoxIcon.Error); 
     } 
     catch (UnauthorizedAccessException authEx) 
     { 
      MessageBox.Show(string.Format("UnauthorizedAccessException: {0}", authEx.Message), "Copy File", MessageBoxButtons.OK, MessageBoxIcon.Error); 
     } 
     catch (Exception ex) 
     { 
      MessageBox.Show(ex.Message, "Copy File", MessageBoxButtons.OK, MessageBoxIcon.Error); 
     } 

     return retVal; 
    } 

    #region Special Folder constants 
    const int CSIDL_DESKTOP = 0x0000;  // <desktop> 
    const int CSIDL_INTERNET = 0x0001;  // Internet Explorer (icon on desktop) 
    const int CSIDL_PROGRAMS = 0x0002;  // Start Menu\Programs 
    const int CSIDL_CONTROLS = 0x0003;  // My Computer\Control Panel 
    const int CSIDL_PRINTERS = 0x0004;  // My Computer\Printers 
    const int CSIDL_PERSONAL = 0x0005;  // My Documents 
    const int CSIDL_FAVORITES = 0x0006;  // <user name>\Favorites 
    const int CSIDL_STARTUP = 0x0007;  // Start Menu\Programs\Startup 
    const int CSIDL_RECENT = 0x0008;  // <user name>\Recent 
    const int CSIDL_SENDTO = 0x0009;  // <user name>\SendTo 
    const int CSIDL_BITBUCKET = 0x000a;  // <desktop>\Recycle Bin 
    const int CSIDL_STARTMENU = 0x000b;  // <user name>\Start Menu 
    const int CSIDL_MYDOCUMENTS = CSIDL_PERSONAL; // Personal was just a silly name for My Documents 
    const int CSIDL_MYMUSIC = 0x000d;  // "My Music" folder 
    const int CSIDL_MYVIDEO = 0x000e;  // "My Videos" folder 
    const int CSIDL_DESKTOPDIRECTORY = 0x0010;  // <user name>\Desktop 
    const int CSIDL_DRIVES = 0x0011;  // My Computer 
    const int CSIDL_NETWORK = 0x0012;  // Network Neighborhood (My Network Places) 
    const int CSIDL_NETHOOD = 0x0013;  // <user name>\nethood 
    const int CSIDL_FONTS = 0x0014;  // windows\fonts 
    const int CSIDL_TEMPLATES = 0x0015; 
    const int CSIDL_COMMON_STARTMENU = 0x0016;  // All Users\Start Menu 
    const int CSIDL_COMMON_PROGRAMS = 0x0017;  // All Users\Start Menu\Programs 
    const int CSIDL_COMMON_STARTUP = 0x0018;  // All Users\Startup 
    const int CSIDL_COMMON_DESKTOPDIRECTORY = 0x0019;  // All Users\Desktop 
    const int CSIDL_APPDATA = 0x001a;  // <user name>\Application Data 
    const int CSIDL_PRINTHOOD = 0x001b;  // <user name>\PrintHood 
    const int CSIDL_LOCAL_APPDATA = 0x001c;  // <user name>\Local Settings\Applicaiton Data (non roaming) 
    const int CSIDL_ALTSTARTUP = 0x001d;  // non localized startup 
    const int CSIDL_COMMON_ALTSTARTUP = 0x001e;  // non localized common startup 
    const int CSIDL_COMMON_FAVORITES = 0x001f; 
    const int CSIDL_INTERNET_CACHE = 0x0020; 
    const int CSIDL_COOKIES = 0x0021; 
    const int CSIDL_HISTORY = 0x0022; 
    const int CSIDL_COMMON_APPDATA = 0x0023;  // All Users\Application Data 
    const int CSIDL_WINDOWS = 0x0024;  // GetWindowsDirectory() 
    const int CSIDL_SYSTEM = 0x0025;  // GetSystemDirectory() 
    const int CSIDL_PROGRAM_FILES = 0x0026;  // C:\Program Files 
    const int CSIDL_MYPICTURES = 0x0027;  // C:\Program Files\My Pictures 
    const int CSIDL_PROFILE = 0x0028;  // USERPROFILE 
    const int CSIDL_SYSTEMX86 = 0x0029;  // x86 system directory on RISC 
    const int CSIDL_PROGRAM_FILESX86 = 0x002a;  // x86 C:\Program Files on RISC 
    const int CSIDL_PROGRAM_FILES_COMMON = 0x002b;  // C:\Program Files\Common 
    const int CSIDL_PROGRAM_FILES_COMMONX86 = 0x002c;  // x86 Program Files\Common on RISC 
    const int CSIDL_COMMON_TEMPLATES = 0x002d;  // All Users\Templates 
    const int CSIDL_COMMON_DOCUMENTS = 0x002e;  // All Users\Documents 
    const int CSIDL_COMMON_ADMINTOOLS = 0x002f;  // All Users\Start Menu\Programs\Administrative Tools 
    const int CSIDL_ADMINTOOLS = 0x0030;  // <user name>\Start Menu\Programs\Administrative Tools 
    const int CSIDL_CONNECTIONS = 0x0031;  // Network and Dial-up Connections 
    const int CSIDL_COMMON_MUSIC = 0x0035;  // All Users\My Music 
    const int CSIDL_COMMON_PICTURES = 0x0036;  // All Users\My Pictures 
    const int CSIDL_COMMON_VIDEO = 0x0037;  // All Users\My Video 
    const int CSIDL_RESOURCES = 0x0038;  // Resource Direcotry 
    const int CSIDL_RESOURCES_LOCALIZED = 0x0039;  // Localized Resource Direcotry 
    const int CSIDL_COMMON_OEM_LINKS = 0x003a;  // Links to All Users OEM specific apps 
    const int CSIDL_CDBURN_AREA = 0x003b;  // USERPROFILE\Local Settings\Application Data\Microsoft\CD Burning 
    const int CSIDL_COMPUTERSNEARME = 0x003d;  // Computers Near Me (computered from Workgroup membership) 
    const int CSIDL_FLAG_CREATE = 0x8000;  // combine with CSIDL_ value to force folder creation in SHGetFolderPath() 
    const int CSIDL_FLAG_DONT_VERIFY = 0x4000;  // combine with CSIDL_ value to return an unverified folder path 
    const int CSIDL_FLAG_DONT_UNEXPAND = 0x2000;  // combine with CSIDL_ value to avoid unexpanding environment variables 
    const int CSIDL_FLAG_NO_ALIAS = 0x1000;  // combine with CSIDL_ value to insure non-alias versions of the pidl 
    const int CSIDL_FLAG_PER_USER_INIT = 0x0800;  // combine with CSIDL_ value to indicate per-user init (eg. upgrade) 
    #endregion 
} 

, 나는 시작에 실행을 해제하기 위해 에스컬레이션 privledges이 사용자가 필요하지 않습니다 문제를 처리하는 더 나은 방법을 생각했다. 내 솔루션은 사용자가 RunOnStartup이라는 설정 범위를 지정하면 프로그램이로드되는 즉시 확인했습니다. 시스템이로드되거나 로그인 할 때 응용 프로그램이 시작되었는지 감지하기 위해 바로 가기라는 모든 사용자 -> 시동 폴더에 추가 된 바로 가기에 인수를 추가했습니다.

 // Quit the application if the per user setting for RunOnStartup is false. 
      if (args != null && args.Length > 0 && args[0].Contains("startup")) 
      { 
       if (Settings1.Default.RunOnStartup == false) 
       { 
        Application.Exit(); 
       } 
      } 
+0

이 솔루션을 [C#을 통해 Windows 파일 및 폴더 바로 가기를 만들려면 어떻게 할 수 있습니까?] (http://stackoverflow.com/q/11513681/3090544) – MarkusEgle

3

앱의 바로 가기를 프로그램 파일 폴더에 설치 한 다음 필요에 따라 해당 .lnk 파일을 복사 할 수 있습니다.

0

저는이 솔루션을 오래 동안 사용 해왔고 멋지게 작동하는 것 같습니다. 조엘에 의해 제안

ShellLink

2

내가 그것을이

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text; 
using Shell32; 
using IWshRuntimeLibrary; 
using System.IO; 

namespace CMS.data 
{ 
    public class overall 
    { 
     public static void place_shortcut_on_desktop() 
     { 
      string desktopPath = Environment.GetFolderPath(Environment.SpecialFolder.Desktop) + "\\YourName.lnk"; 
      string shortcutto = System.Reflection.Assembly.GetExecutingAssembly().Location; 

      var wsh = new IWshShell_Class(); 
      IWshRuntimeLibrary.IWshShortcut shortcut = wsh.CreateShortcut(desktopPath) as IWshRuntimeLibrary.IWshShortcut; 
      shortcut.TargetPath = shortcutto; 
      shortcut.WorkingDirectory = Directory.GetParent(shortcutto).FullName; 
      shortcut.Save(); 
     } 
    }//class overall 
} 

으로 수행되었다 '는 그렇지

당신은 또한 아이콘이 방법을 추가 할 수 있습니다 문제를 만들 수있는 "작업 디렉토리"를 기억하지만 내 경우에는 내가 didn를 필요가 없다.

스택 오버플로에 대한 나의 첫 번째 대답은 정말 도움이 될 것입니다.

+1

2017 년에 이것을 보니 정말 고맙습니다. – Samuel