Home | History | Annotate | Download | only in proxy
      1 // Copyright (c) 2012 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #include "ppapi/proxy/file_chooser_resource.h"
      6 
      7 #include "base/bind.h"
      8 #include "base/strings/string_split.h"
      9 #include "ipc/ipc_message.h"
     10 #include "ppapi/c/pp_errors.h"
     11 #include "ppapi/proxy/dispatch_reply_message.h"
     12 #include "ppapi/proxy/file_ref_resource.h"
     13 #include "ppapi/proxy/ppapi_messages.h"
     14 #include "ppapi/shared_impl/var.h"
     15 
     16 namespace ppapi {
     17 namespace proxy {
     18 
     19 FileChooserResource::FileChooserResource(Connection connection,
     20                                          PP_Instance instance,
     21                                          PP_FileChooserMode_Dev mode,
     22                                          const std::string& accept_types)
     23     : PluginResource(connection, instance),
     24       mode_(mode) {
     25   PopulateAcceptTypes(accept_types, &accept_types_);
     26 }
     27 
     28 FileChooserResource::~FileChooserResource() {
     29 }
     30 
     31 thunk::PPB_FileChooser_API* FileChooserResource::AsPPB_FileChooser_API() {
     32   return this;
     33 }
     34 
     35 int32_t FileChooserResource::Show(const PP_ArrayOutput& output,
     36                                   scoped_refptr<TrackedCallback> callback) {
     37   return ShowWithoutUserGesture(PP_FALSE, PP_MakeUndefined(), output, callback);
     38 }
     39 
     40 int32_t FileChooserResource::ShowWithoutUserGesture(
     41     PP_Bool save_as,
     42     PP_Var suggested_file_name,
     43     const PP_ArrayOutput& output,
     44     scoped_refptr<TrackedCallback> callback) {
     45   int32_t result = ShowInternal(save_as, suggested_file_name, callback);
     46   if (result == PP_OK_COMPLETIONPENDING)
     47     output_.set_pp_array_output(output);
     48   return result;
     49 }
     50 
     51 int32_t FileChooserResource::Show0_5(scoped_refptr<TrackedCallback> callback) {
     52   return ShowInternal(PP_FALSE, PP_MakeUndefined(), callback);
     53 }
     54 
     55 PP_Resource FileChooserResource::GetNextChosenFile() {
     56  if (file_queue_.empty())
     57     return 0;
     58 
     59   // Return the next resource in the queue. It will already have been addrefed
     60   // (they're currently owned by the FileChooser) and returning it transfers
     61   // ownership of that reference to the plugin.
     62   PP_Resource next = file_queue_.front();
     63   file_queue_.pop();
     64   return next;
     65 }
     66 
     67 int32_t FileChooserResource::ShowWithoutUserGesture0_5(
     68     PP_Bool save_as,
     69     PP_Var suggested_file_name,
     70     scoped_refptr<TrackedCallback> callback) {
     71   return ShowInternal(save_as, suggested_file_name, callback);
     72 }
     73 
     74 // static
     75 void FileChooserResource::PopulateAcceptTypes(
     76     const std::string& input,
     77     std::vector<std::string>* output) {
     78   if (input.empty())
     79     return;
     80 
     81   std::vector<std::string> type_list;
     82   base::SplitString(input, ',', &type_list);
     83   output->reserve(type_list.size());
     84 
     85   for (size_t i = 0; i < type_list.size(); ++i) {
     86     std::string type = type_list[i];
     87     base::TrimWhitespaceASCII(type, base::TRIM_ALL, &type);
     88 
     89     // If the type is a single character, it definitely cannot be valid. In the
     90     // case of a file extension it would be a single ".". In the case of a MIME
     91     // type it would just be a "/".
     92     if (type.length() < 2)
     93       continue;
     94     if (type.find_first_of('/') == std::string::npos && type[0] != '.')
     95       continue;
     96     StringToLowerASCII(&type);
     97     output->push_back(type);
     98   }
     99 }
    100 
    101 void FileChooserResource::OnPluginMsgShowReply(
    102     const ResourceMessageReplyParams& params,
    103     const std::vector<FileRefCreateInfo>& chosen_files) {
    104   if (output_.is_valid()) {
    105     // Using v0.6 of the API with the output array.
    106     std::vector<PP_Resource> files;
    107     for (size_t i = 0; i < chosen_files.size(); i++) {
    108       files.push_back(FileRefResource::CreateFileRef(
    109           connection(),
    110           pp_instance(),
    111           chosen_files[i]));
    112     }
    113     output_.StoreResourceVector(files);
    114   } else {
    115     // Convert each of the passed in file infos to resources. These will be
    116     // owned by the FileChooser object until they're passed to the plugin.
    117     DCHECK(file_queue_.empty());
    118     for (size_t i = 0; i < chosen_files.size(); i++) {
    119       file_queue_.push(FileRefResource::CreateFileRef(
    120           connection(),
    121           pp_instance(),
    122           chosen_files[i]));
    123     }
    124   }
    125 
    126   // Notify the plugin of the new data.
    127   callback_->Run(params.result());
    128   // DANGER: May delete |this|!
    129 }
    130 
    131 int32_t FileChooserResource::ShowInternal(
    132     PP_Bool save_as,
    133     const PP_Var& suggested_file_name,
    134     scoped_refptr<TrackedCallback> callback) {
    135   if (TrackedCallback::IsPending(callback_))
    136     return PP_ERROR_INPROGRESS;
    137 
    138   if (!sent_create_to_renderer())
    139     SendCreate(RENDERER, PpapiHostMsg_FileChooser_Create());
    140 
    141   callback_ = callback;
    142   StringVar* sugg_str = StringVar::FromPPVar(suggested_file_name);
    143 
    144   PpapiHostMsg_FileChooser_Show msg(
    145         PP_ToBool(save_as),
    146         mode_ == PP_FILECHOOSERMODE_OPENMULTIPLE,
    147         sugg_str ? sugg_str->value() : std::string(),
    148         accept_types_);
    149   Call<PpapiPluginMsg_FileChooser_ShowReply>(RENDERER, msg,
    150       base::Bind(&FileChooserResource::OnPluginMsgShowReply, this));
    151   return PP_OK_COMPLETIONPENDING;
    152 }
    153 
    154 }  // namespace proxy
    155 }  // namespace ppapi
    156