रीसायकल बिन के लिए एक फ़ाइल भेजें


83

वर्तमान में मैं निम्नलिखित फ़ंक्शन का उपयोग कर रहा हूं

file.Delete();

लेकिन मैं इस फ़ंक्शन का उपयोग केवल रीसायकल बिन में फ़ाइल भेजने के लिए कैसे कर सकता हूं, इसे केवल एकमुश्त हटाने के बजाय?



3
@ UweKeim का लिंक अब मृत हो गया है, आप यहाँ MSDN मैगज़ीन (दिसंबर 2007) का एक .chm प्रारूप संस्करण पा सकते हैं , यह लेख कहा जाता है .NET Matters: IFileOperation in Windows Vistaऔर यह Columnsफ़ोल्डर में पाया जाता है ।
18:

लेख मेरे लिए .chm फ़ाइल में नहीं खुलता है। यह लिंक काम करता है: docs.microsoft.com/en-us/archive/msdn-magazine/2007/deloy/…
RandomEngy

इसके अलावा, आपको FOFX_RECYCLEONDELETE = 0x00080000ऑपरेशन के झंडे में जोड़ने की आवश्यकता है , और यह झंडा केवल विंडोज 8 या इसके बाद के संस्करण पर समर्थित है।
रैंडम इनीज जूल

जवाबों:


52

नोट: यह भी Windows सेवाओं जैसे गैर UI इंटरएक्टिव एप्लिकेशन के साथ काम नहीं करता है

यह आवरण आपको आवश्यक कार्यक्षमता प्रदान कर सकता है:

using System.Runtime.InteropServices;

public class FileOperationAPIWrapper
    {
        /// <summary>
        /// Possible flags for the SHFileOperation method.
        /// </summary>
        [Flags]
        public enum FileOperationFlags : ushort
        {
            /// <summary>
            /// Do not show a dialog during the process
            /// </summary>
            FOF_SILENT = 0x0004,
            /// <summary>
            /// Do not ask the user to confirm selection
            /// </summary>
            FOF_NOCONFIRMATION = 0x0010,
            /// <summary>
            /// Delete the file to the recycle bin.  (Required flag to send a file to the bin
            /// </summary>
            FOF_ALLOWUNDO = 0x0040,
            /// <summary>
            /// Do not show the names of the files or folders that are being recycled.
            /// </summary>
            FOF_SIMPLEPROGRESS = 0x0100,
            /// <summary>
            /// Surpress errors, if any occur during the process.
            /// </summary>
            FOF_NOERRORUI = 0x0400,
            /// <summary>
            /// Warn if files are too big to fit in the recycle bin and will need
            /// to be deleted completely.
            /// </summary>
            FOF_WANTNUKEWARNING = 0x4000,
        }

        /// <summary>
        /// File Operation Function Type for SHFileOperation
        /// </summary>
        public enum FileOperationType : uint
        {
            /// <summary>
            /// Move the objects
            /// </summary>
            FO_MOVE = 0x0001,
            /// <summary>
            /// Copy the objects
            /// </summary>
            FO_COPY = 0x0002,
            /// <summary>
            /// Delete (or recycle) the objects
            /// </summary>
            FO_DELETE = 0x0003,
            /// <summary>
            /// Rename the object(s)
            /// </summary>
            FO_RENAME = 0x0004,
        }



        /// <summary>
        /// SHFILEOPSTRUCT for SHFileOperation from COM
        /// </summary>
        [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)]
        private struct SHFILEOPSTRUCT
        {

            public IntPtr hwnd;
            [MarshalAs(UnmanagedType.U4)]
            public FileOperationType wFunc;
            public string pFrom;
            public string pTo;
            public FileOperationFlags fFlags;
            [MarshalAs(UnmanagedType.Bool)]
            public bool fAnyOperationsAborted;
            public IntPtr hNameMappings;
            public string lpszProgressTitle;
        }

        [DllImport("shell32.dll", CharSet = CharSet.Auto)]
        private static extern int SHFileOperation(ref SHFILEOPSTRUCT FileOp);

        /// <summary>
        /// Send file to recycle bin
        /// </summary>
        /// <param name="path">Location of directory or file to recycle</param>
        /// <param name="flags">FileOperationFlags to add in addition to FOF_ALLOWUNDO</param>
        public static bool Send(string path, FileOperationFlags flags)
        {
            try
            {
                var fs = new SHFILEOPSTRUCT
                                        {
                                            wFunc = FileOperationType.FO_DELETE,
                                            pFrom = path + '\0' + '\0',
                                            fFlags = FileOperationFlags.FOF_ALLOWUNDO | flags
                                        };
                SHFileOperation(ref fs);
                return true;
            }
            catch (Exception)
            {
                return false;
            }
        }

        /// <summary>
        /// Send file to recycle bin.  Display dialog, display warning if files are too big to fit (FOF_WANTNUKEWARNING)
        /// </summary>
        /// <param name="path">Location of directory or file to recycle</param>
        public static bool Send(string path)
        {
            return Send(path, FileOperationFlags.FOF_NOCONFIRMATION | FileOperationFlags.FOF_WANTNUKEWARNING);
        }

        /// <summary>
        /// Send file silently to recycle bin.  Surpress dialog, surpress errors, delete if too large.
        /// </summary>
        /// <param name="path">Location of directory or file to recycle</param>
        public static bool MoveToRecycleBin(string path)
        {
            return Send(path, FileOperationFlags.FOF_NOCONFIRMATION | FileOperationFlags.FOF_NOERRORUI | FileOperationFlags.FOF_SILENT);

        }

        private static bool deleteFile(string path, FileOperationFlags flags)
        {
            try
            {
                var fs = new SHFILEOPSTRUCT
                                        {
                                            wFunc = FileOperationType.FO_DELETE,
                                            pFrom = path + '\0' + '\0',
                                            fFlags = flags
                                        };
                SHFileOperation(ref fs);
                return true;
            }
            catch (Exception)
            {
                return false;
            }
        }

        public static bool DeleteCompletelySilent(string path)
        {
            return deleteFile(path,
                              FileOperationFlags.FOF_NOCONFIRMATION | FileOperationFlags.FOF_NOERRORUI |
                              FileOperationFlags.FOF_SILENT);
        }
    }

मुझे समझ नहीं आ रहा है कि इसका उपयोग कैसे करें ... क्या आप समझा सकते हैं?
muttley91

4
64 बिट प्लेटफ़ॉर्म के लिए संकलन करने पर पैक = 1 निकालें (यह अन्यथा विफल हो जाएगा)। बिना पैक = 1 निर्दिष्ट यह 32 बिट और 64 बिट दोनों के लिए काम करेगा। pinvoke.net/default.aspx/Structures/SHFILEOPSTRUCT.html
शॉन

1
पैक = 1 का उपयोग करते समय, एक AccessViolationException को फेंक दिया गया था। इसे हटाकर चाल चली गई। 64-बिट विंडोज वैसे
P1nGu1n

1
इस कोड को चलाने के लिए क्या आवश्यकताएं हैं? मैं पैक = 1 को हटाता हूं लेकिन यह अभी भी संकलन नहीं करता है। DllImport, DllImportAttribute, MarshalAs, MarshalAsAttribute, StructLayout, StructLayoutAttribute एक नाम स्थान के रूप में मौजूद नहीं हैं। कोई मदद धन्यवाद कृपया :)
शुद्ध ४५

1
SHFileOperation लंबे रास्तों को संभालता नहीं है और MAX_PATH (यहां तक ​​कि a \\? \ Prefix) की तुलना में अधिक लंबे रास्तों से विफल हो जाएगा।
मेल्विन

155

FileSystem.DeleteFile का उपयोग करें और सही रीसायकल को निर्दिष्ट करें ।

हालांकि यह UI इंटरएक्टिव ऐप्स के साथ काम करेगा, लेकिन यह विंडोज सर्विस ऐप जैसे गैर-यूआई इंटरैक्टिव ऐप के साथ काम नहीं करेगा।


17
@ नोल्डोरिन यह एक पूरी तरह से ठीक समाधान है, एक चढ़ाव के लायक नहीं है। मुझे इस बात पर एक संदर्भ पसंद है कि विजुअलबैसिक लाइब्रेरी को एक्सेस करना "बदसूरत" क्यों है।
jsmith

7
@ नोल्डोरिन: विशेष रूप से इस मामले Microsoft.VisualBasic.FileIO.FileSystemमें मूल रूप से उपयोग किए गए उदाहरण के समान है SHFileOperation
डर्क वोल्मार

18
@ नोल्डोरिन: अग्ली, हुह? मेरे लिए WinAPI रास्ता बदसूरत है - साथ ही, आपने कुछ गड़बड़ करने के लिए बेहतर शॉट दिया है। मैं व्यक्तिगत रूप से VB सिंटैक्स को नापसंद करता हूं लेकिन विधानसभाओं में यह सिर्फ ILइतना है कि मुझे कोई आपत्ति नहीं है। VB असेंबली एक ही WinAPI फ़ंक्शन btw कहता है।
जारोस्लाव जंडेक

7
@ नोल्डोरिन: अप्रचलित? क्या आपने Microsoft.VisualBasic.Compatibilityसंयोगवश विधानसभा को गलत माना है ? कि मैं एक से बचना होगा । ऐसा नहीं लगता कि यह जल्द ही किसी भी समय पदावनत होने वाला है (इसका उपयोग आरडीएल रिपोर्टिंग इंजन, आदि में किया जाता है)।
जारोस्लाव जंडेक

6
@ नोल्डोरिन: बिल्ट-इन फ्रेमवर्क असेंबली का उपयोग करना हार्ड-स्टाइल मैपिंग से शेल 32.dll तक जाने की तुलना में बेहतर समाधान की तरह दिखता है। फ्रेमवर्क असेंबली का उपयोग करने से आपको परिवर्तन पोर्टेबल हो जाता है और बाद में विकसित हो रहा है। सिस्टम लाइब्रेरी के लिए मैपिंग, आप किसी भी दिन अप्रचलित होने के सभी अवसरों को ले लेते हैं ...
fredlegrain

41

से MSDN :

Microsoft.VisualBasic असेंबली में संदर्भ जोड़ें। इस पुस्तकालय में आवश्यक वर्ग पाया जाता है।

फ़ाइल के शीर्ष पर स्थित कथन का उपयोग करके इसे जोड़ें using Microsoft.VisualBasic.FileIO;

FileSystem.DeleteFileकिसी फ़ाइल को हटाने के लिए उपयोग करें , इसमें रीसायकल बिन या नहीं निर्दिष्ट करने का विकल्प है।

FileSystem.DeleteDirectoryरीसायकल बिन या नहीं भेजने के लिए निर्दिष्ट करने के विकल्प के साथ एक निर्देशिका को हटाने के लिए उपयोग करें ।


Microsoct.VisualBasic को शामिल करने में समस्या यह है कि यह मेरे प्रोग्राम में कहीं और SearchOption के मेरे उपयोग के साथ संघर्ष करता है (GetFiles () फ़ंक्शन का हिस्सा)।
muttley91

8
@rar Downvote अभी भी योग्य नहीं है क्योंकि यह इस प्रश्न में निर्दिष्ट नहीं किया गया था कि "VisualBasic लाइब्रेरी को संघर्ष के कारण संदर्भित नहीं किया जा सकता है।" जिसे आप आसानी से अपने कोड में हल कर सकते हैं। stackoverflow.com/questions/1317263/…
जुमिथ

1
यह विधि आंतरिक रूप से SHFileOperation का उपयोग करती प्रतीत होती है, जो लंबे रास्तों को संभालती नहीं है और MAX_PATH (यहां तक ​​कि a \\? \ Prefix) की तुलना में अधिक लंबे रास्तों से विफल हो जाएगी।
मेल्विन

17

निम्नलिखित समाधान अन्य लोगों की तुलना में सरल है:

using Shell32;

static class Program
{
    public static Shell shell = new Shell();
    public static Folder RecyclingBin = shell.NameSpace(10);

    static void Main()
    {
        RecyclingBin.MoveHere("PATH TO FILE/FOLDER")
    }
}

आप इस लाइब्रेरी का उपयोग करके रीसायकल बिन के अन्य कार्यों का उपयोग कर सकते हैं।

सबसे पहले, पुस्तकालय "Microsoft शैल नियंत्रण और स्वचालन" (COM मेनू से) को जोड़ने के लिए मत भूलना, Shell32नामस्थान का उपयोग करने में सक्षम होने के लिए । यह आपके प्रोग्राम के साथ-साथ संकलित होने के बजाय गतिशील रूप से आपकी परियोजना से जुड़ा होगा।

[१]: https://i.stack.imgur.com/erV


8
आपका उत्तर बेहतर होगा जब आप पहले पैराग्राफ में अन्य उत्तरों पर टिप्पणी करने के बजाय अपने समाधान पर ध्यान केंद्रित करेंगे। इसके अलावा, स्पष्टता के लिए मैं की जगह था 10द्वारा Shell32.ShellSpecialFolderConstants.ssfBITBUCKET। यह MoveHere64 ("यदि संभव हो तो जानकारी को संरक्षित करें") जैसे विकल्पों के संबंध में, दूसरे पैरामीटर का उल्लेख करने योग्य हो सकता है । MSDN से कुछ प्रलेखन स्रोतों को जोड़ना एक अच्छा परिष्करण होगा।
grek40

2
ऐसा लगता है कि MoveHere को कॉल करने से कोई त्रुटि नहीं होती है: इसे गैर-मौजूदा फ़ाइल पर कॉल करना चुपचाप विफल हो जाता है! यह MAX_CHARS की तुलना में लंबे समय तक मौन रूप से विफल रहता है, "\\? \" उपसर्ग के साथ या बिना ...
मेल्विन

13

दुर्भाग्य से आपको रीसायकल बिन में फ़ाइल हटाने के लिए Win32 API का सहारा लेना होगा। इस पोस्ट के आधार पर, निम्न कोड का प्रयास करें । यह SHFileOperationविंडोज शेल के माध्यम से फाइल सिस्टम ऑपरेशन के लिए सामान्य फ़ंक्शन का उपयोग करता है ।

निम्नलिखित को परिभाषित करें (एक उपयोगिता वर्ग में संभवतः सबसे अच्छा है)।

[StructLayout(LayoutKind.Sequential, CharSet=CharSet.Auto, Pack=1)]
public struct SHFILEOPSTRUCT
{
        public IntPtr hwnd;
        [MarshalAs(UnmanagedType.U4)] public int wFunc;
        public string pFrom;
        public string pTo;
        public short fFlags;
        [MarshalAs(UnmanagedType.Bool)] public bool fAnyOperationsAborted;
        public IntPtr hNameMappings;
        public string lpszProgressTitle;
}

[DllImport("shell32.dll", CharSet=CharSet.Auto)]
public static extern int SHFileOperation(ref SHFILEOPSTRUCT FileOp);

public const int FO_DELETE = 3;
public const int FOF_ALLOWUNDO = 0x40;
public const int FOF_NOCONFIRMATION = 0x10; // Don't prompt the user

और एक फ़ाइल को हटाने के लिए इसका उपयोग करने के लिए, इसे रीसायकल बिन में भेजते हुए, आप कुछ इस तरह चाहते हैं:

var shf = new SHFILEOPSTRUCT();
shf.wFunc = FO_DELETE;
shf.fFlags = FOF_ALLOWUNDO | FOF_NOCONFIRMATION;
shf.pFrom = @"C:\test.txt";
SHFileOperation(ref shf);

1
और डबल नल स्ट्रिंग को समाप्त करता है।
शॉन ई

1
SHFileOperation लंबे रास्तों को संभालता नहीं है और MAX_PATH (यहां तक ​​कि a \\? \ Prefix) की तुलना में अधिक लंबे रास्तों से विफल हो जाएगा।
मेल्विन

ध्यान दें कि यह रेखा shf.pFrom = @"C:\test.txt";गलत है - pFrom को डबल शून्य समाप्त किया जाना चाहिए। आपको \0फ़ाइल में जोड़ना चाहिए shf.pFrom = "C:\\text.txt\0";Docs.microsoft.com/en-us/windows/desktop/api/shellapi/…
lindexi

1

आप DllImport कर सकते हैं SHFileOperation ऐसा करने के लिए ।


1
SHFileOperation लंबे रास्तों को संभालता नहीं है और MAX_PATH (यहां तक ​​कि a \\? \ Prefix) की तुलना में अधिक लंबे रास्तों से विफल हो जाएगा।
मेल्विन

1

मैं इस एक्सटेंशन विधि का उपयोग करता हूं, फिर मैं केवल एक DirectoryInfo या FileInfo का उपयोग कर सकता हूं और इसे हटा सकता हूं।

public static class NativeMethods
{
    [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Auto)]
        struct SHFILEOPSTRUCT
    {
        public IntPtr hwnd;
        [MarshalAs(UnmanagedType.U4)]
        public int wFunc;
        public string pFrom;
        public string pTo;
        public short fFlags;
        [MarshalAs(UnmanagedType.Bool)]
        public bool fAnyOperationsAborted;
        public IntPtr hNameMappings;
        public string lpszProgressTitle;
    }
    private const int FO_DELETE = 0x0003;
    private const int FOF_ALLOWUNDO = 0x0040;           // Preserve undo information, if possible. 
    private const int FOF_NOCONFIRMATION = 0x0010;      // Show no confirmation dialog box to the user      


    [DllImport("shell32.dll", CharSet = CharSet.Auto)]
    static extern int SHFileOperation(ref SHFILEOPSTRUCT FileOp);

    static bool DeleteFileOrFolder(string path)
    {


        SHFILEOPSTRUCT fileop = new SHFILEOPSTRUCT();
        fileop.wFunc = FO_DELETE;
        fileop.pFrom = path + '\0' + '\0';            
        fileop.fFlags = FOF_ALLOWUNDO | FOF_NOCONFIRMATION;


        var rc= SHFileOperation(ref fileop);
        return rc==0;
    }

    public static bool ToRecycleBin(this DirectoryInfo dir)
    {
        dir?.Refresh();
        if(dir is null || !dir.Exists)
        {
            return false;
        }
        else
            return DeleteFileOrFolder(dir.FullName);
    }
    public static bool ToRecycleBin(this FileInfo file)
    {
        file?.Refresh();

        if(file is null ||!file.Exists)
        {
            return false;
        }
        return DeleteFileOrFolder(file.FullName);
    }
}

यह कैसे हो सकता है कॉल करने के लिए एक नमूना:

private void BtnDelete_Click(object sender, EventArgs e)
{
    if(MessageBox.Show("Are you sure you would like to delete this directory?", "Delete & Close", MessageBoxButtons.YesNo, MessageBoxIcon.Question) == DialogResult.No)
        return;

    var dir= new DirectoryInfo(directoryName);
    dir.ToRecycleBin();

}

-1

इसके लिए बिल्ट-इन लाइब्रेरी है।

पहले संदर्भ Microsoft जोड़ें। विजुअलबसिक फिर इस कोड को जोड़ें:

FileSystem.DeleteFile(path_of_the_file,
                        Microsoft.VisualBasic.FileIO.UIOption.AllDialogs,
                        Microsoft.VisualBasic.FileIO.RecycleOption.SendToRecycleBin,
                        Microsoft.VisualBasic.FileIO.UICancelOption.ThrowException);

मैंने इसे यहां पाया है

हमारी साइट का प्रयोग करके, आप स्वीकार करते हैं कि आपने हमारी Cookie Policy और निजता नीति को पढ़ और समझा लिया है।
Licensed under cc by-sa 3.0 with attribution required.