फ़ाइल संवाद चुनें [बंद]


146

क्या किसी को पूरी तरह से फ़ाइल संवाद चुनने का पता है? शायद एक जहाँ आप विशिष्ट एक्सटेंशन वाले लोगों को छोड़कर सभी फ़ाइलों को फ़िल्टर कर सकते हैं?

मुझे अपनी परियोजनाओं में आसानी से लागू करने के लिए कुछ भी हल्का नहीं मिला है। केवल अन्य विकल्प OI FileManager के खुले इरादों का उपयोग करता प्रतीत होता है, लेकिन इसके लिए उपयोगकर्ता को पहले से ही फ़ाइल प्रबंधक स्थापित करने की आवश्यकता होती है।

मैं आभारी रहूंगा अगर कोई डायलॉग बता सकता है जो उपयोगकर्ता को फ़ोल्डर्स ब्राउज़ करने और एक फ़ाइल का चयन करने और पथ को वापस करने की अनुमति देगा।


5
यदि, जैसा कि आप कहते हैं, "इंटरनेट को इस तरह के उदाहरण की आवश्यकता है," तो यह आपके लिए ऐसे महान उद्देश्य के लिए एक अवसर बनाने का अवसर है। एसओ "किराए का कोडर" साइट नहीं है। दूसरी ओर, यदि आप एक फ़ाइल चयन संवाद का निर्माण / उपयोग करने और समस्याओं में चलाने का प्रयास कर रहे हैं, तो यह आपके विशिष्ट प्रश्न के साथ आने का स्थान है।
कैल जैकबसन


33
सवाल यह है कि अगर इसकी एलाड्री जैसी कोई चीज मौजूद है, जो एक अच्छी बात है, क्योंकि आप मूत को फिर से लाना नहीं चाहते हैं।
वेलरोक

9
यह सवाल बंद नहीं होना चाहिए। मैं aFileChooser ( github.com/iPaulPro/aFileChooser ) के साथ एक उत्तर पोस्ट करने जा रहा था , लेकिन नहीं कर सकता, तो चलिए उम्मीद करते हैं कि जिन लोगों को यह टिप्पणी देखने की आवश्यकता है।
टिआगो

2
मैं सहमत हूं, यह एक उपयोगी प्रश्न है। मैं उत्तर के लिए इस सरल एकल-वर्ग कार्यान्वयन में योगदान करने की उम्मीद कर रहा था: ninthavenue.com.au/simple-android-file-chooser
रोजर कीज़

जवाबों:


184

आपको केवल onCreateDialogएक गतिविधि में ओवरराइड करने की आवश्यकता है ।

//In an Activity
private String[] mFileList;
private File mPath = new File(Environment.getExternalStorageDirectory() + "//yourdir//");
private String mChosenFile;
private static final String FTYPE = ".txt";    
private static final int DIALOG_LOAD_FILE = 1000;

private void loadFileList() {
    try {
        mPath.mkdirs();
    }
    catch(SecurityException e) {
        Log.e(TAG, "unable to write on the sd card " + e.toString());
    }
    if(mPath.exists()) {
        FilenameFilter filter = new FilenameFilter() {

            @Override
            public boolean accept(File dir, String filename) {
                File sel = new File(dir, filename);
                return filename.contains(FTYPE) || sel.isDirectory();
            }

        };
        mFileList = mPath.list(filter);
    }
    else {
        mFileList= new String[0];
    }
}

protected Dialog onCreateDialog(int id) {
    Dialog dialog = null;
    AlertDialog.Builder builder = new Builder(this);

    switch(id) {
        case DIALOG_LOAD_FILE:
            builder.setTitle("Choose your file");
            if(mFileList == null) {
                Log.e(TAG, "Showing file picker before loading the file list");
                dialog = builder.create();
                return dialog;
            }
            builder.setItems(mFileList, new DialogInterface.OnClickListener() {
                public void onClick(DialogInterface dialog, int which) {
                    mChosenFile = mFileList[which];
                    //you can do stuff with the file here too
                }
            });
            break;
    }
    dialog = builder.show();
    return dialog;
}

4
फ़ोल्डरों को नेविगेट करने और पैरेंट फ़ोल्डर में जाने की क्षमता जोड़ें, और आपको यह मिल गया
आयमन फोरनियर

48
यदि आप फ़ाइल सिस्टम को नेविगेट करने के लिए उपरोक्त को संशोधित नहीं कर सकते हैं, तो मुझे नहीं पता कि आप इसे अपने ऐप में पहली बार कैसे सबमिट करने जा रहे हैं। जब वह पहले से ही "नियमों" पर तुला हुआ है और आपके लिए कोड लिखा है, तो मुझे यकीन है कि आप वास्तव में उसके लिए इनाम की राशि रखने वाले नहीं हैं।
ब्लमर

6
मैंने फ़ोल्डर को शामिल करने का तरीका दिखाने के लिए ऊपर दिए गए कोड को संपादित किया। आपको बाकी का पता लगाने में सक्षम होना चाहिए। यदि आप यह पता लगाते हैं कि दबाया गया फ़ाइल ऑनलाईन में एक निर्देशिका है तो नया पथ सेट करें और ऑनक्रिएटडियल कॉल फिर से कॉल करें।
नाथन श्वरमन

1
अरे व्हाट्स "एनसाइट्स", यह एक चर है, वास्तव में मैं आपके कोड का उपयोग कर रहा हूं और इसके लिए व्हाट्स "पर्यावरण" का पता लगाने में सक्षम नहीं हूं।
TRONZ

6
जोड़ने के लिए मत भूलना <उपयोग-अनुमति एंड्रॉइड: नाम = "android.permission.READ_EXTERNAL_STORAGE" /> मैनिफेस्ट की अनुमति
ज़ार ई अहमर

73

विचार के लिए Thanx schwiz! यहाँ संशोधित समाधान है:

public class FileDialog {
    private static final String PARENT_DIR = "..";
    private final String TAG = getClass().getName();
    private String[] fileList;
    private File currentPath;
    public interface FileSelectedListener {
        void fileSelected(File file);
    }
    public interface DirectorySelectedListener {
        void directorySelected(File directory);
    }
    private ListenerList<FileSelectedListener> fileListenerList = new ListenerList<FileDialog.FileSelectedListener>();
    private ListenerList<DirectorySelectedListener> dirListenerList = new ListenerList<FileDialog.DirectorySelectedListener>();
    private final Activity activity;
    private boolean selectDirectoryOption;
    private String fileEndsWith;    

    /**
    * @param activity 
    * @param initialPath
    */
    public FileDialog(Activity activity, File initialPath) {
        this(activity, initialPath, null);
    }

    public FileDialog(Activity activity, File initialPath, String fileEndsWith) {
        this.activity = activity;
        setFileEndsWith(fileEndsWith);
        if (!initialPath.exists()) initialPath = Environment.getExternalStorageDirectory();
            loadFileList(initialPath);
    }

    /**
    * @return file dialog
    */
    public Dialog createFileDialog() {
        Dialog dialog = null;
        AlertDialog.Builder builder = new AlertDialog.Builder(activity);

        builder.setTitle(currentPath.getPath());
        if (selectDirectoryOption) {
            builder.setPositiveButton("Select directory", new OnClickListener() {
                public void onClick(DialogInterface dialog, int which) {
                    Log.d(TAG, currentPath.getPath());
                    fireDirectorySelectedEvent(currentPath);
                }
            });
        }

        builder.setItems(fileList, new DialogInterface.OnClickListener() {
            public void onClick(DialogInterface dialog, int which) {
                String fileChosen = fileList[which];
                File chosenFile = getChosenFile(fileChosen);
                if (chosenFile.isDirectory()) {
                    loadFileList(chosenFile);
                    dialog.cancel();
                    dialog.dismiss();
                    showDialog();
                } else fireFileSelectedEvent(chosenFile);
            }
        });

        dialog = builder.show();
        return dialog;
    }


    public void addFileListener(FileSelectedListener listener) {
        fileListenerList.add(listener);
    }

    public void removeFileListener(FileSelectedListener listener) {
        fileListenerList.remove(listener);
    }

    public void setSelectDirectoryOption(boolean selectDirectoryOption) {
        this.selectDirectoryOption = selectDirectoryOption;
    }

    public void addDirectoryListener(DirectorySelectedListener listener) {
        dirListenerList.add(listener);
    }

    public void removeDirectoryListener(DirectorySelectedListener listener) {
        dirListenerList.remove(listener);
    }

    /**
    * Show file dialog
    */
    public void showDialog() {
        createFileDialog().show();
    }

    private void fireFileSelectedEvent(final File file) {
        fileListenerList.fireEvent(new FireHandler<FileDialog.FileSelectedListener>() {
            public void fireEvent(FileSelectedListener listener) {
                listener.fileSelected(file);
            }
        });
    }

    private void fireDirectorySelectedEvent(final File directory) {
        dirListenerList.fireEvent(new FireHandler<FileDialog.DirectorySelectedListener>() {
            public void fireEvent(DirectorySelectedListener listener) {
                listener.directorySelected(directory);
            }
        });
    }

    private void loadFileList(File path) {
        this.currentPath = path;
        List<String> r = new ArrayList<String>();
        if (path.exists()) {
            if (path.getParentFile() != null) r.add(PARENT_DIR);
            FilenameFilter filter = new FilenameFilter() {
                public boolean accept(File dir, String filename) {
                    File sel = new File(dir, filename);
                    if (!sel.canRead()) return false;
                    if (selectDirectoryOption) return sel.isDirectory();
                    else {
                        boolean endsWith = fileEndsWith != null ? filename.toLowerCase().endsWith(fileEndsWith) : true;
                        return endsWith || sel.isDirectory();
                    }
                }
            };
            String[] fileList1 = path.list(filter);
            for (String file : fileList1) {
                r.add(file);
            }
        }
        fileList = (String[]) r.toArray(new String[]{});
    }

    private File getChosenFile(String fileChosen) {
        if (fileChosen.equals(PARENT_DIR)) return currentPath.getParentFile();
        else return new File(currentPath, fileChosen);
    }

    private void setFileEndsWith(String fileEndsWith) {
        this.fileEndsWith = fileEndsWith != null ? fileEndsWith.toLowerCase() : fileEndsWith;
    }
}

class ListenerList<L> {
    private List<L> listenerList = new ArrayList<L>();

    public interface FireHandler<L> {
        void fireEvent(L listener);
    }

    public void add(L listener) {
        listenerList.add(listener);
    }

    public void fireEvent(FireHandler<L> fireHandler) {
        List<L> copy = new ArrayList<L>(listenerList);
        for (L l : copy) {
            fireHandler.fireEvent(l);
        }
    }

    public void remove(L listener) {
        listenerList.remove(listener);
    }

    public List<L> getListenerList() {
        return listenerList;
    }
}

गतिविधि onCreate पर इसका उपयोग करें (निर्देशिका चयन विकल्प टिप्पणी की गई है):

protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    File mPath = new File(Environment.getExternalStorageDirectory() + "//DIR//");
    fileDialog = new FileDialog(this, mPath, ".txt");
    fileDialog.addFileListener(new FileDialog.FileSelectedListener() {
        public void fileSelected(File file) {
            Log.d(getClass().getName(), "selected file " + file.toString());
        }
    });
    //fileDialog.addDirectoryListener(new FileDialog.DirectorySelectedListener() {
    //  public void directorySelected(File directory) {
    //      Log.d(getClass().getName(), "selected dir " + directory.toString());
    //  }
    //});
    //fileDialog.setSelectDirectoryOption(false);
    fileDialog.showDialog();
}

8
महान सहायक वर्ग! मुझे एक छोटा सा गड़बड़ मिला - पहले रन लोडफिलिस्ट पर () फ़ाइल एक्सटेंशन द्वारा फ़िल्टर नहीं किया जाएगा, क्योंकि यह अभी तक SetFileEndsWith द्वारा सेट नहीं किया जाएगा। मैंने निर्माणकर्ता को तीसरे पैरामीटर फ़ाइल स्वीकार करने के लिए फिर से काम किया है, और इसे लोडफ़िलिस्ट () कॉल से पहले निर्माता में सेट किया है।
सूपरहार्टन

हाय अच्छा कोड, धन्यवाद। इस कोड को कई फ़ाइल प्रारूप यानी फाइलडायलॉग .सेटफाइल्डइंडस्विथ ("। txt", "। pdf") से चुना जा सकता है; या fileDialog.setFileEndsWith ("fle / *"); कृपया जवाब दें
अनीता

लेकिन, इसे संशोधित करना बहुत आसान है। समस्या यह है कि .setFileEndsWith () बिल्कुल काम नहीं करता है, क्योंकि फ़ाइल सूची को कंस्ट्रक्टर में आवंटित किया गया है। आपको कई इनपुट को स्वीकार करने के लिए कंस्ट्रक्टर को बदलना होगा और फिर लाइन को बदलना होगा: "बूलियन सिरों के साथ = fileEsWith! = Null; filename.toLowerCase ()। समाप्त होता है (fileEndsith): सच;" जो भी आप डेटा संरचना में डालते हैं उसे ठीक से मिलान करने के लिए। यह एक बहुत ही मामूली परिवर्तन है।
ताताराइज करें

ये सभी खूंखार श्रोता सूचियां और फायरईवेंट (फायरहैंडलर <ओएमजी>) अनावश्यक दिखते हैं (क्या कभी किसी ने उनका उपयोग किया है?), लेकिन कोड काम करता है।
18446744073709551615

नमस्ते, महान सहायक वर्ग के लिए धन्यवाद। मैं कैसे कर सकता हूँ इसके लिए CanceledOnTouchOutside। मैंने शो मेथडलॉग में शो मेथड में जोड़ा, लेकिन मेरे लिए काम नहीं करता है
Dauezevy

15

मैंने बनाया है FolderLayoutजो आपकी मदद कर सकता है। यह लिंक ने मेरी मदद की

folderview.xml

<?xml version="1.0" encoding="utf-8"?>

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="vertical" android:layout_width="match_parent"
    android:layout_height="match_parent">
    <TextView android:id="@+id/path" android:text="Path"
        android:layout_width="match_parent" android:layout_height="wrap_content"></TextView>
    <ListView android:layout_width="fill_parent"
        android:layout_height="wrap_content" android:id="@+id/list"></ListView>

</LinearLayout>

FolderLayout.java

package com.testsample.activity;




   public class FolderLayout extends LinearLayout implements OnItemClickListener {

    Context context;
    IFolderItemListener folderListener;
    private List<String> item = null;
    private List<String> path = null;
    private String root = "/";
    private TextView myPath;
    private ListView lstView;

    public FolderLayout(Context context, AttributeSet attrs) {
        super(context, attrs);

        // TODO Auto-generated constructor stub
        this.context = context;


        LayoutInflater layoutInflater = (LayoutInflater) context
                .getSystemService(Context.LAYOUT_INFLATER_SERVICE);
        View view = layoutInflater.inflate(R.layout.folderview, this);

        myPath = (TextView) findViewById(R.id.path);
        lstView = (ListView) findViewById(R.id.list);

        Log.i("FolderView", "Constructed");
        getDir(root, lstView);

    }

    public void setIFolderItemListener(IFolderItemListener folderItemListener) {
        this.folderListener = folderItemListener;
    }

    //Set Directory for view at anytime
    public void setDir(String dirPath){
        getDir(dirPath, lstView);
    }


    private void getDir(String dirPath, ListView v) {

        myPath.setText("Location: " + dirPath);
        item = new ArrayList<String>();
        path = new ArrayList<String>();
        File f = new File(dirPath);
        File[] files = f.listFiles();

        if (!dirPath.equals(root)) {

            item.add(root);
            path.add(root);
            item.add("../");
            path.add(f.getParent());

        }
        for (int i = 0; i < files.length; i++) {
            File file = files[i];
            path.add(file.getPath());
            if (file.isDirectory())
                item.add(file.getName() + "/");
            else
                item.add(file.getName());

        }

        Log.i("Folders", files.length + "");

        setItemList(item);

    }

    //can manually set Item to display, if u want
    public void setItemList(List<String> item){
        ArrayAdapter<String> fileList = new ArrayAdapter<String>(context,
                R.layout.row, item);

        lstView.setAdapter(fileList);
        lstView.setOnItemClickListener(this);
    }


    public void onListItemClick(ListView l, View v, int position, long id) {
        File file = new File(path.get(position));
        if (file.isDirectory()) {
            if (file.canRead())
                getDir(path.get(position), l);
            else {
//what to do when folder is unreadable
                if (folderListener != null) {
                    folderListener.OnCannotFileRead(file);

                }

            }
        } else {

//what to do when file is clicked
//You can add more,like checking extension,and performing separate actions
            if (folderListener != null) {
                folderListener.OnFileClicked(file);
            }

        }
    }

    public void onItemClick(AdapterView<?> arg0, View arg1, int arg2, long arg3) {
        // TODO Auto-generated method stub
        onListItemClick((ListView) arg0, arg0, arg2, arg3);
    }

}

और IFolderItemListenerजब जोड़ने के लिए एक इंटरफ़ेस एfileItem क्लिक किया जाता है

IFolderItemListener.java

public interface IFolderItemListener {

    void OnCannotFileRead(File file);//implement what to do folder is Unreadable
    void OnFileClicked(File file);//What to do When a file is clicked
}

पंक्ति को परिभाषित करने के लिए एक xml भी

row.xml

<?xml version="1.0" encoding="utf-8"?>
<TextView xmlns:android="http://schemas.android.com/apk/res/android"
    android:id="@+id/rowtext" android:layout_width="fill_parent"
    android:textSize="23sp" android:layout_height="match_parent"/>

अपने आवेदन में कैसे उपयोग करें

अपने xml में,

folders.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent" android:layout_height="match_parent"
    android:orientation="horizontal" android:weightSum="1">
    <com.testsample.activity.FolderLayout android:layout_height="match_parent" layout="@layout/folderview"
        android:layout_weight="0.35"
        android:layout_width="200dp" android:id="@+id/localfolders"></com.testsample.activity.FolderLayout></LinearLayout>

आपकी गतिविधि में,

SampleFolderActivity.java

public class SampleFolderActivity extends Activity implements IFolderItemListener {

    FolderLayout localFolders;

    /** Called when the activity is first created. */

    @Override
    public void onCreate(Bundle savedInstanceState) {

        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);

        localFolders = (FolderLayout)findViewById(R.id.localfolders);
        localFolders.setIFolderItemListener(this);
            localFolders.setDir("./sys");//change directory if u want,default is root   

    }

    //Your stuff here for Cannot open Folder
    public void OnCannotFileRead(File file) {
        // TODO Auto-generated method stub
        new AlertDialog.Builder(this)
        .setIcon(R.drawable.icon)
        .setTitle(
                "[" + file.getName()
                        + "] folder can't be read!")
        .setPositiveButton("OK",
                new DialogInterface.OnClickListener() {

                    public void onClick(DialogInterface dialog,
                            int which) {


                    }
                }).show();

    }


    //Your stuff here for file Click
    public void OnFileClicked(File file) {
        // TODO Auto-generated method stub
        new AlertDialog.Builder(this)
        .setIcon(R.drawable.icon)
        .setTitle("[" + file.getName() + "]")
        .setPositiveButton("OK",
                new DialogInterface.OnClickListener() {
                    public void onClick(DialogInterface dialog,
                            int which) {


                    }

                }).show();
    }

}

आवश्यक पुस्तकालयों को आयात करें। आशा है कि ये आपकी मदद करेंगे ...


बहुत बहुत धन्यवाद कि मेरे लिए काम करता है, बस एक साधारण फ़ाइल किसी भी अनावश्यक ब्लोट के बिना गतिविधि की खोज
माइक76

5

हाल ही में एक फ़ाइल / फ़ोल्डर ब्राउज़र की तलाश कर रहा था और एक नई एक्सप्लोरर गतिविधि (एंड्रॉइड लाइब्रेरी) बनाने का फैसला किया: https://github.com/vaal12/AndroidFileBrowser

मिलान परीक्षण आवेदन https://github.com/vaal12/FileBrowserTestApplication- उपयोग करने के लिए एक नमूना है।

फोन फ़ाइल संरचना से निर्देशिकाओं और फ़ाइलों को चुनने की अनुमति देता है।


इसे भी जांचें: stackoverflow.com/a/59104787/3141844
Criss

3

मिश्रण में जोड़ना: OI फ़ाइल प्रबंधक में openintents.org पर एक सार्वजनिक एपि पंजीकृत है

http://www.openintents.org/filemanager

http://www.openintents.org/action/org-openintents-action-pick-file/


उपरोक्त लिंक कोई और काम नहीं कर रहा है।
uaaquarius

हाँ मैं जानता हूँ। जो कोई भी openintents.org का रखरखाव कर रहा था, उसने इसे तोड़ दिया है।
एडवर्ड फाल्क

Juozas Kontvainis का धन्यवाद जिन्होंने नए लिंक का पता लगाया।
एडवर्ड फाल्क

इसके अलावा: क्या खोज करने का एक तरीका है, या यहां तक ​​कि पंजीकृत इरादों को ब्राउज़ करना है?
एडवर्ड फॉक

2

मैंने सैमसंग फ़ाइल चयनकर्ता डायलॉग लागू किया है, यह फ़ाइल को खोलने, सहेजने, फ़ाइल एक्सटेंशन फ़िल्टर करने और उसी निर्देशिका में नई निर्देशिका बनाने की क्षमता प्रदान करता है जो मुझे लगता है कि यह कोशिश करने लायक है। यहां वह लिंक है जिसे आपको सैमसंग डेवलपर साइट पर लॉग इन करना होगा। समाधान देखें

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