| Index: ui/android/java/src/org/chromium/ui/SelectFileDialog.java | 
| diff --git a/ui/android/java/src/org/chromium/ui/SelectFileDialog.java b/ui/android/java/src/org/chromium/ui/SelectFileDialog.java | 
| new file mode 100644 | 
| index 0000000000000000000000000000000000000000..9098481636fd0d2b51267ab566babae50778398c | 
| --- /dev/null | 
| +++ b/ui/android/java/src/org/chromium/ui/SelectFileDialog.java | 
| @@ -0,0 +1,259 @@ | 
| +// Copyright (c) 2012 The Chromium Authors. All rights reserved. | 
| +// Use of this source code is governed by a BSD-style license that can be | 
| +// found in the LICENSE file. | 
| + | 
| +package org.chromium.ui; | 
| + | 
| +import android.app.Activity; | 
| +import android.content.ContentResolver; | 
| +import android.content.Intent; | 
| +import android.database.Cursor; | 
| +import android.net.Uri; | 
| +import android.os.Environment; | 
| +import android.provider.MediaStore; | 
| + | 
| +import java.io.File; | 
| +import java.util.ArrayList; | 
| +import java.util.Arrays; | 
| +import java.util.List; | 
| + | 
| +import org.chromium.base.CalledByNative; | 
| +import org.chromium.base.JNINamespace; | 
| +import org.chromium.ui.gfx.NativeWindow; | 
| + | 
| +/** | 
| + * A dialog that is triggered from a file input field that allows a user to select a file based on | 
| + * a set of accepted file types. The path of the selected file is passed to the native dialog. | 
| + */ | 
| +@JNINamespace("ui") | 
| +class SelectFileDialog implements NativeWindow.IntentCallback{ | 
| +    // TODO (aurimas): Swap these constants with AppResources when it gets moved to base to support | 
| +    //                 internationalization. | 
| +    private static final String LOW_MEMORY_ERROR = | 
| +            "Unable to complete previous operation due to low memory"; | 
| +    private static final String OPENING_FILE_ERROR = "Failed to open selected file"; | 
| + | 
| +    private static final String IMAGE_TYPE = "image/"; | 
| +    private static final String VIDEO_TYPE = "video/"; | 
| +    private static final String AUDIO_TYPE = "audio/"; | 
| +    private static final String ALL_IMAGE_TYPES = IMAGE_TYPE + "*"; | 
| +    private static final String ALL_VIDEO_TYPES = VIDEO_TYPE + "*"; | 
| +    private static final String ALL_AUDIO_TYPES = AUDIO_TYPE + "*"; | 
| +    private static final String ANY_TYPES = "*/*"; | 
| +    private static final String CAPTURE_CAMERA = "camera"; | 
| +    private static final String CAPTURE_CAMCORDER = "camcorder"; | 
| +    private static final String CAPTURE_MICROPHONE = "microphone"; | 
| +    private static final String CAPTURE_FILESYSTEM = "filesystem"; | 
| +    private static final String CAPTURE_IMAGE_DIRECTORY = "browser-photos"; | 
| + | 
| +    private final int mNativeSelectFileDialog; | 
| +    private List<String> mFileTypes; | 
| +    private String mCapture;  // May be null if no capture parameter was set. | 
| +    private Uri mCameraOutputUri; | 
| + | 
| +    private SelectFileDialog(int nativeSelectFileDialog) { | 
| +        mNativeSelectFileDialog = nativeSelectFileDialog; | 
| +    } | 
| + | 
| +    /** | 
| +     * Creates and starts an intent based on the passed fileTypes and capture value. | 
| +     * @param fileTypes MIME types requested (i.e. "image/*") | 
| +     * @param capture The capture value as described in http://www.w3.org/TR/html-media-capture/ | 
| +     * @param window The NativeWindow that can show intents | 
| +     */ | 
| +    @CalledByNative | 
| +    private void selectFile(String[] fileTypes, String capture, NativeWindow window) { | 
| +        mFileTypes = new ArrayList<String>(Arrays.asList(fileTypes)); | 
| +        mCapture = capture; | 
| + | 
| +        Intent chooser = new Intent(Intent.ACTION_CHOOSER); | 
| +        Intent camera = new Intent(MediaStore.ACTION_IMAGE_CAPTURE); | 
| +        mCameraOutputUri = Uri.fromFile(getFileForImageCapture()); | 
| +        camera.putExtra(MediaStore.EXTRA_OUTPUT, mCameraOutputUri); | 
| +        Intent camcorder = new Intent(MediaStore.ACTION_VIDEO_CAPTURE); | 
| +        Intent soundRecorder = new Intent( | 
| +                MediaStore.Audio.Media.RECORD_SOUND_ACTION); | 
| + | 
| +        // Quick check - if a capture parameter other than filesystem (the default) is specified we | 
| +        // should just launch the appropriate intent. Otherwise build up a chooser based on the | 
| +        // accept type and then display that to the user. | 
| +        if (captureCamera()) { | 
| +            if (window.showIntent(camera, this, LOW_MEMORY_ERROR)) return; | 
| +        } else if (captureCamcorder()) { | 
| +            if (window.showIntent(camcorder, this, LOW_MEMORY_ERROR)) return; | 
| +        } else if (captureMicrophone()) { | 
| +            if (window.showIntent(soundRecorder, this, LOW_MEMORY_ERROR)) return; | 
| +        } | 
| + | 
| +        Intent getContentIntent = new Intent(Intent.ACTION_GET_CONTENT); | 
| +        getContentIntent.addCategory(Intent.CATEGORY_OPENABLE); | 
| +        ArrayList<Intent> extraIntents = new ArrayList<Intent>(); | 
| +        if (!noSpecificType()) { | 
| +            // Create a chooser based on the accept type that was specified in the webpage. Note | 
| +            // that if the web page specified multiple accept types, we will have built a generic | 
| +            // chooser above. | 
| +            if (shouldShowImageTypes()) { | 
| +                extraIntents.add(camera); | 
| +                getContentIntent.setType("image/*"); | 
| +            } else if (shouldShowVideoTypes()) { | 
| +                extraIntents.add(camcorder); | 
| +                getContentIntent.setType("video/*"); | 
| +            } else if (shouldShowAudioTypes()) { | 
| +                extraIntents.add(soundRecorder); | 
| +                getContentIntent.setType("audio/*"); | 
| +            } | 
| +        } | 
| + | 
| +        if (extraIntents.isEmpty()) { | 
| +            // We couldn't resolve an accept type, so fallback to a generic chooser. | 
| +            getContentIntent.setType("*/*"); | 
| +            extraIntents.add(camera); | 
| +            extraIntents.add(camcorder); | 
| +            extraIntents.add(soundRecorder); | 
| +        } | 
| + | 
| +        chooser.putExtra(Intent.EXTRA_INITIAL_INTENTS, | 
| +                extraIntents.toArray(new Intent[] { })); | 
| + | 
| +        chooser.putExtra(Intent.EXTRA_INTENT, getContentIntent); | 
| + | 
| +        if (!window.showIntent(chooser, this, LOW_MEMORY_ERROR)) onFileNotSelected(); | 
| +    } | 
| + | 
| +    /** | 
| +     * Get a file for the image capture in the CAPTURE_IMAGE_DIRECTORY directory. | 
| +     */ | 
| +    private File getFileForImageCapture() { | 
| +        File externalDataDir = Environment.getExternalStoragePublicDirectory( | 
| +                Environment.DIRECTORY_DCIM); | 
| +        File cameraDataDir = new File(externalDataDir.getAbsolutePath() + | 
| +                File.separator + CAPTURE_IMAGE_DIRECTORY); | 
| +        if (!cameraDataDir.exists() && !cameraDataDir.mkdirs()) { | 
| +            cameraDataDir = externalDataDir; | 
| +        } | 
| +        File photoFile = new File(cameraDataDir.getAbsolutePath() + | 
| +                File.separator + System.currentTimeMillis() + ".jpg"); | 
| +        return photoFile; | 
| +    } | 
| + | 
| +    /** | 
| +     * Callback method to handle the intent results and pass on the path to the native | 
| +     * SelectFileDialog. | 
| +     * @param window The window that has access to the application activity. | 
| +     * @param resultCode The result code whether the intent returned successfully. | 
| +     * @param contentResolver The content resolver used to extract the path of the selected file. | 
| +     * @param results The results of the requested intent. | 
| +     */ | 
| +    @Override | 
| +    public void onIntentCompleted(NativeWindow window, int resultCode, | 
| +            ContentResolver contentResolver, Intent results) { | 
| +        if (resultCode != Activity.RESULT_OK) { | 
| +            onFileNotSelected(); | 
| +            return; | 
| +        } | 
| +        boolean success = false; | 
| +        if (results == null) { | 
| +            // If we have a successful return but no data, then assume this is the camera returning | 
| +            // the photo that we requested. | 
| +            nativeOnFileSelected(mNativeSelectFileDialog, mCameraOutputUri.getPath()); | 
| +            success = true; | 
| + | 
| +            // Broadcast to the media scanner that there's a new photo on the device so it will | 
| +            // show up right away in the gallery (rather than waiting until the next time the media | 
| +            // scanner runs). | 
| +            window.getActivity().sendBroadcast(new Intent( | 
| +                    Intent.ACTION_MEDIA_SCANNER_SCAN_FILE, mCameraOutputUri)); | 
| +        } else { | 
| +            // We get back a content:// URI from the system if the user picked a file from the | 
| +            // gallery. The ContentView has functionality that will convert that content:// URI to | 
| +            // a file path on disk that Chromium understands. | 
| +            Cursor c = contentResolver.query(results.getData(), | 
| +                    new String[] { MediaStore.MediaColumns.DATA }, | 
| +                    null, null, null); | 
| +            if (c != null) { | 
| +                if (c.getCount() == 1) { | 
| +                    c.moveToFirst(); | 
| +                    String path = c.getString(0); | 
| +                    if (path != null) { | 
| +                        // Not all providers support the MediaStore.DATA column. For example, | 
| +                        // Gallery3D (com.android.gallery3d.provider) does not support it for | 
| +                        // Picasa Web Album images. | 
| +                        nativeOnFileSelected(mNativeSelectFileDialog, path); | 
| +                        success = true; | 
| +                    } | 
| +                } | 
| +                c.close(); | 
| +            } | 
| +        } | 
| +        if (!success) { | 
| +            onFileNotSelected(); | 
| +            window.showError(LOW_MEMORY_ERROR); | 
| +        } | 
| +    } | 
| + | 
| +    private void onFileNotSelected() { | 
| +        nativeOnFileNotSelected(mNativeSelectFileDialog); | 
| +    } | 
| + | 
| +    private boolean noSpecificType() { | 
| +        // We use a single Intent to decide the type of the file chooser we display to the user, | 
| +        // which means we can only give it a single type. If there are multiple accept types | 
| +        // specified, we will fallback to a generic chooser (unless a capture parameter has been | 
| +        // specified, in which case we'll try to satisfy that first. | 
| +        return mFileTypes.size() != 1 || mFileTypes.contains(ANY_TYPES); | 
| +    } | 
| + | 
| +    private boolean shouldShowTypes(String allTypes, String specificType) { | 
| +        if (noSpecificType() || mFileTypes.contains(allTypes)) return true; | 
| +        return acceptSpecificType(specificType); | 
| +    } | 
| + | 
| +    private boolean shouldShowImageTypes() { | 
| +        return shouldShowTypes(ALL_IMAGE_TYPES,IMAGE_TYPE); | 
| +    } | 
| + | 
| + | 
| +    private boolean shouldShowVideoTypes() { | 
| +        return shouldShowTypes(ALL_VIDEO_TYPES, VIDEO_TYPE); | 
| +    } | 
| + | 
| +    private boolean shouldShowAudioTypes() { | 
| +        return shouldShowTypes(ALL_AUDIO_TYPES, AUDIO_TYPE); | 
| +    } | 
| + | 
| +    private boolean captureCamera() { | 
| +        return shouldShowImageTypes() && mCapture != null && mCapture.startsWith(CAPTURE_CAMERA); | 
| +    } | 
| + | 
| +    private boolean captureCamcorder() { | 
| +        return shouldShowVideoTypes() && mCapture != null && | 
| +                mCapture.startsWith(CAPTURE_CAMCORDER); | 
| +    } | 
| + | 
| +    private boolean captureMicrophone() { | 
| +        return shouldShowAudioTypes() && mCapture != null && | 
| +                mCapture.startsWith(CAPTURE_MICROPHONE); | 
| +    } | 
| + | 
| +    private boolean captureFilesystem() { | 
| +        return mCapture != null && mCapture.startsWith(CAPTURE_FILESYSTEM); | 
| +    } | 
| + | 
| +    private boolean acceptSpecificType(String accept) { | 
| +        for (String type : mFileTypes) { | 
| +            if (type.startsWith(accept)) { | 
| +                return true; | 
| +            } | 
| +        } | 
| +        return false; | 
| +    } | 
| + | 
| +    @CalledByNative | 
| +    private static SelectFileDialog create(int nativeSelectFileDialog) { | 
| +        return new SelectFileDialog(nativeSelectFileDialog); | 
| +    } | 
| + | 
| +    private native void nativeOnFileSelected(int nativeSelectFileDialogImpl, | 
| +            String filePath); | 
| +    private native void nativeOnFileNotSelected(int nativeSelectFileDialogImpl); | 
| +} | 
|  |