| Index: ppapi/proxy/file_chooser_resource.cc
|
| diff --git a/ppapi/proxy/file_chooser_resource.cc b/ppapi/proxy/file_chooser_resource.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..f4e21ec4f2054805bfdf66e78f9ae891bba18b98
|
| --- /dev/null
|
| +++ b/ppapi/proxy/file_chooser_resource.cc
|
| @@ -0,0 +1,149 @@
|
| +// 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.
|
| +
|
| +#include "ppapi/proxy/file_chooser_resource.h"
|
| +
|
| +#include "base/string_split.h"
|
| +#include "ipc/ipc_message.h"
|
| +#include "ppapi/c/pp_errors.h"
|
| +#include "ppapi/proxy/ppapi_messages.h"
|
| +#include "ppapi/proxy/ppb_file_ref_proxy.h"
|
| +#include "ppapi/shared_impl/var.h"
|
| +
|
| +namespace ppapi {
|
| +namespace proxy {
|
| +
|
| +FileChooserResource::FileChooserResource(IPC::Sender* sender,
|
| + PP_Instance instance,
|
| + PP_FileChooserMode_Dev mode,
|
| + const std::string& accept_types)
|
| + : PluginResource(sender, instance),
|
| + mode_(mode) {
|
| + PopulateAcceptTypes(accept_types, &accept_types_);
|
| +}
|
| +
|
| +FileChooserResource::~FileChooserResource() {
|
| +}
|
| +
|
| +thunk::PPB_FileChooser_API* FileChooserResource::AsPPB_FileChooser_API() {
|
| + return this;
|
| +}
|
| +
|
| +int32_t FileChooserResource::Show(const PP_ArrayOutput& output,
|
| + scoped_refptr<TrackedCallback> callback) {
|
| + return ShowWithoutUserGesture(PP_FALSE, PP_MakeUndefined(), output, callback);
|
| +}
|
| +
|
| +int32_t FileChooserResource::ShowWithoutUserGesture(
|
| + PP_Bool save_as,
|
| + PP_Var suggested_file_name,
|
| + const PP_ArrayOutput& output,
|
| + scoped_refptr<TrackedCallback> callback) {
|
| + int32_t result = ShowInternal(PP_FALSE, suggested_file_name, callback);
|
| + if (result == PP_OK_COMPLETIONPENDING)
|
| + output_.set_pp_array_output(output);
|
| + return result;
|
| +}
|
| +
|
| +int32_t FileChooserResource::Show0_5(scoped_refptr<TrackedCallback> callback) {
|
| + return ShowInternal(PP_FALSE, PP_MakeUndefined(), callback);
|
| +}
|
| +
|
| +PP_Resource FileChooserResource::GetNextChosenFile() {
|
| + if (file_queue_.empty())
|
| + return 0;
|
| +
|
| + // Return the next resource in the queue. It will already have been addrefed
|
| + // (they're currently owned by the FileChooser) and returning it transfers
|
| + // ownership of that reference to the plugin.
|
| + PP_Resource next = file_queue_.front();
|
| + file_queue_.pop();
|
| + return next;
|
| +}
|
| +
|
| +int32_t FileChooserResource::ShowWithoutUserGesture0_5(
|
| + PP_Bool save_as,
|
| + PP_Var suggested_file_name,
|
| + scoped_refptr<TrackedCallback> callback) {
|
| + return ShowInternal(save_as, suggested_file_name, callback);
|
| +}
|
| +
|
| +// static
|
| +void FileChooserResource::PopulateAcceptTypes(
|
| + const std::string& input,
|
| + std::vector<std::string>* output) {
|
| + if (input.empty())
|
| + return;
|
| +
|
| + std::vector<std::string> type_list;
|
| + base::SplitString(input, ',', &type_list);
|
| + output->reserve(type_list.size());
|
| +
|
| + for (size_t i = 0; i < type_list.size(); ++i) {
|
| + std::string type = type_list[i];
|
| + TrimWhitespaceASCII(type, TRIM_ALL, &type);
|
| +
|
| + // If the type is a single character, it definitely cannot be valid. In the
|
| + // case of a file extension it would be a single ".". In the case of a MIME
|
| + // type it would just be a "/".
|
| + if (type.length() < 2)
|
| + continue;
|
| + if (type.find_first_of('/') == std::string::npos && type[0] != '.')
|
| + continue;
|
| + StringToLowerASCII(&type);
|
| + output->push_back(type);
|
| + }
|
| +}
|
| +
|
| +void FileChooserResource::OnReplyReceived(int /* sequence */,
|
| + int32_t result,
|
| + const IPC::Message& msg) {
|
| + PpapiPluginMsg_FileChooser_ShowReply::Schema::Param param;
|
| + PpapiPluginMsg_FileChooser_ShowReply::Read(&msg, ¶m);
|
| + const std::vector<ppapi::PPB_FileRef_CreateInfo>& chosen_files = param.a;
|
| +
|
| + if (output_.is_valid()) {
|
| + // Using v0.6 of the API with the output array.
|
| + std::vector<PP_Resource> files;
|
| + for (size_t i = 0; i < chosen_files.size(); i++)
|
| + files.push_back(PPB_FileRef_Proxy::DeserializeFileRef(chosen_files[i]));
|
| + output_.StoreResourceVector(files);
|
| + } else {
|
| + // Convert each of the passed in file infos to resources. These will be
|
| + // owned by the FileChooser object until they're passed to the plugin.
|
| + DCHECK(file_queue_.empty());
|
| + for (size_t i = 0; i < chosen_files.size(); i++) {
|
| + file_queue_.push(PPB_FileRef_Proxy::DeserializeFileRef(
|
| + chosen_files[i]));
|
| + }
|
| + }
|
| +
|
| + // Notify the plugin of the new data.
|
| + TrackedCallback::ClearAndRun(&callback_, PP_OK);
|
| + // DANGER: May delete |this|!
|
| +}
|
| +
|
| +int32_t FileChooserResource::ShowInternal(
|
| + PP_Bool save_as,
|
| + const PP_Var& suggested_file_name,
|
| + scoped_refptr<TrackedCallback> callback) {
|
| + if (TrackedCallback::IsPending(callback_))
|
| + return PP_ERROR_INPROGRESS;
|
| +
|
| + if (!sent_create_to_renderer())
|
| + SendCreateToRenderer(PpapiHostMsg_FileChooser_Create());
|
| +
|
| + callback_ = callback;
|
| + StringVar* sugg_str = StringVar::FromPPVar(suggested_file_name);
|
| +
|
| + CallRenderer(PpapiHostMsg_FileChooser_Show(
|
| + PP_ToBool(save_as),
|
| + mode_ == PP_FILECHOOSERMODE_OPENMULTIPLE,
|
| + sugg_str ? sugg_str->value() : std::string(),
|
| + accept_types_));
|
| + return PP_OK_COMPLETIONPENDING;
|
| +}
|
| +
|
| +} // namespace proxy
|
| +} // namespace ppapi
|
|
|