| OLD | NEW |
| (Empty) |
| 1 // Copyright (c) 2011 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 "chrome/browser/chromeos/extensions/file_browser_extension_api.h" | |
| 6 | |
| 7 #include "base/base64.h" | |
| 8 #include "base/bind.h" | |
| 9 #include "base/command_line.h" | |
| 10 #include "base/json/json_writer.h" | |
| 11 #include "base/logging.h" | |
| 12 #include "base/memory/singleton.h" | |
| 13 #include "base/string_util.h" | |
| 14 #include "base/stringprintf.h" | |
| 15 #include "base/time.h" | |
| 16 #include "base/values.h" | |
| 17 #include "chrome/browser/chromeos/extensions/file_browser_event_router.h" | |
| 18 #include "chrome/browser/chromeos/extensions/file_manager_util.h" | |
| 19 #include "chrome/browser/extensions/extension_event_router.h" | |
| 20 #include "chrome/browser/extensions/extension_function_dispatcher.h" | |
| 21 #include "chrome/browser/extensions/extension_process_manager.h" | |
| 22 #include "chrome/browser/extensions/extension_service.h" | |
| 23 #include "chrome/browser/extensions/extension_tab_util.h" | |
| 24 #include "chrome/browser/extensions/process_map.h" | |
| 25 #include "chrome/browser/prefs/pref_service.h" | |
| 26 #include "chrome/browser/prefs/scoped_user_pref_update.h" | |
| 27 #include "chrome/browser/profiles/profile.h" | |
| 28 #include "chrome/browser/ui/browser.h" | |
| 29 #include "chrome/browser/ui/views/select_file_dialog_extension.h" | |
| 30 #include "chrome/browser/ui/webui/extension_icon_source.h" | |
| 31 #include "chrome/common/chrome_switches.h" | |
| 32 #include "chrome/common/extensions/extension.h" | |
| 33 #include "chrome/common/extensions/file_browser_handler.h" | |
| 34 #include "chrome/common/pref_names.h" | |
| 35 #include "content/browser/child_process_security_policy.h" | |
| 36 #include "content/browser/renderer_host/render_view_host.h" | |
| 37 #include "content/browser/tab_contents/tab_contents.h" | |
| 38 #include "content/public/browser/browser_thread.h" | |
| 39 #include "content/public/browser/render_process_host.h" | |
| 40 #include "googleurl/src/gurl.h" | |
| 41 #include "grit/generated_resources.h" | |
| 42 #include "grit/platform_locale_settings.h" | |
| 43 #include "ui/base/l10n/l10n_util.h" | |
| 44 #include "webkit/fileapi/file_system_context.h" | |
| 45 #include "webkit/fileapi/file_system_file_util.h" | |
| 46 #include "webkit/fileapi/file_system_mount_point_provider.h" | |
| 47 #include "webkit/fileapi/file_system_operation.h" | |
| 48 #include "webkit/fileapi/file_system_operation_context.h" | |
| 49 #include "webkit/fileapi/file_system_path_manager.h" | |
| 50 #include "webkit/fileapi/file_system_types.h" | |
| 51 #include "webkit/fileapi/file_system_util.h" | |
| 52 | |
| 53 using content::BrowserThread; | |
| 54 | |
| 55 namespace { | |
| 56 | |
| 57 // Error messages. | |
| 58 const char kFileError[] = "File error %d"; | |
| 59 const char kInvalidFileUrl[] = "Invalid file URL"; | |
| 60 const char kVolumeDevicePathNotFound[] = "Device path not found"; | |
| 61 | |
| 62 #ifdef OS_CHROMEOS | |
| 63 // Volume type strings. | |
| 64 const char kVolumeTypeFlash[] = "flash"; | |
| 65 const char kVolumeTypeOptical[] = "optical"; | |
| 66 const char kVolumeTypeHardDrive[] = "hdd"; | |
| 67 const char kVolumeTypeUnknown[] = "undefined"; | |
| 68 #endif | |
| 69 | |
| 70 // Internal task ids. | |
| 71 const char kEnqueueTaskId[] = "enqueue"; | |
| 72 | |
| 73 const int kReadOnlyFilePermissions = base::PLATFORM_FILE_OPEN | | |
| 74 base::PLATFORM_FILE_READ | | |
| 75 base::PLATFORM_FILE_EXCLUSIVE_READ | | |
| 76 base::PLATFORM_FILE_ASYNC; | |
| 77 | |
| 78 const int kReadWriteFilePermissions = base::PLATFORM_FILE_OPEN | | |
| 79 base::PLATFORM_FILE_CREATE | | |
| 80 base::PLATFORM_FILE_OPEN_ALWAYS | | |
| 81 base::PLATFORM_FILE_CREATE_ALWAYS | | |
| 82 base::PLATFORM_FILE_OPEN_TRUNCATED | | |
| 83 base::PLATFORM_FILE_READ | | |
| 84 base::PLATFORM_FILE_WRITE | | |
| 85 base::PLATFORM_FILE_EXCLUSIVE_READ | | |
| 86 base::PLATFORM_FILE_EXCLUSIVE_WRITE | | |
| 87 base::PLATFORM_FILE_ASYNC | | |
| 88 base::PLATFORM_FILE_WRITE_ATTRIBUTES; | |
| 89 | |
| 90 struct LastUsedHandler { | |
| 91 LastUsedHandler(int t, const FileBrowserHandler* h, URLPatternSet p) | |
| 92 : timestamp(t), | |
| 93 handler(h), | |
| 94 patterns(p) { | |
| 95 } | |
| 96 | |
| 97 int timestamp; | |
| 98 const FileBrowserHandler* handler; | |
| 99 URLPatternSet patterns; | |
| 100 }; | |
| 101 | |
| 102 typedef std::vector<LastUsedHandler> LastUsedHandlerList; | |
| 103 typedef std::set<const FileBrowserHandler*> ActionSet; | |
| 104 | |
| 105 // Breaks down task_id that is used between getFileTasks() and executeTask() on | |
| 106 // its building blocks. task_id field the following structure: | |
| 107 // <extension-id>|<task-action-id> | |
| 108 // Currently, the only supported task-type is of 'context'. | |
| 109 bool CrackTaskIdentifier(const std::string& task_id, | |
| 110 std::string* target_extension_id, | |
| 111 std::string* action_id) { | |
| 112 std::vector<std::string> result; | |
| 113 int count = Tokenize(task_id, std::string("|"), &result); | |
| 114 if (count != 2) | |
| 115 return false; | |
| 116 *target_extension_id = result[0]; | |
| 117 *action_id = result[1]; | |
| 118 return true; | |
| 119 } | |
| 120 | |
| 121 std::string MakeTaskID(const char* extension_id, | |
| 122 const char* action_id) { | |
| 123 return base::StringPrintf("%s|%s", extension_id, action_id); | |
| 124 } | |
| 125 | |
| 126 bool GetFileBrowserHandlers(Profile* profile, | |
| 127 const GURL& selected_file_url, | |
| 128 ActionSet* results) { | |
| 129 ExtensionService* service = profile->GetExtensionService(); | |
| 130 if (!service) | |
| 131 return false; // In unit-tests, we may not have an ExtensionService. | |
| 132 | |
| 133 for (ExtensionList::const_iterator iter = service->extensions()->begin(); | |
| 134 iter != service->extensions()->end(); | |
| 135 ++iter) { | |
| 136 const Extension* extension = iter->get(); | |
| 137 if (!extension->file_browser_handlers()) | |
| 138 continue; | |
| 139 | |
| 140 for (Extension::FileBrowserHandlerList::const_iterator action_iter = | |
| 141 extension->file_browser_handlers()->begin(); | |
| 142 action_iter != extension->file_browser_handlers()->end(); | |
| 143 ++action_iter) { | |
| 144 const FileBrowserHandler* action = action_iter->get(); | |
| 145 if (!action->MatchesURL(selected_file_url)) | |
| 146 continue; | |
| 147 | |
| 148 results->insert(action_iter->get()); | |
| 149 } | |
| 150 } | |
| 151 return true; | |
| 152 } | |
| 153 | |
| 154 bool SortByLastUsedTimestampDesc(const LastUsedHandler& a, | |
| 155 const LastUsedHandler& b) { | |
| 156 return a.timestamp > b.timestamp; | |
| 157 } | |
| 158 | |
| 159 // TODO(zelidrag): Wire this with ICU to make this sort I18N happy. | |
| 160 bool SortByTaskName(const LastUsedHandler& a, const LastUsedHandler& b) { | |
| 161 return base::strcasecmp(a.handler->title().data(), | |
| 162 b.handler->title().data()) > 0; | |
| 163 } | |
| 164 | |
| 165 void SortLastUsedHandlerList(LastUsedHandlerList *list) { | |
| 166 // Sort by the last used descending. | |
| 167 std::sort(list->begin(), list->end(), SortByLastUsedTimestampDesc); | |
| 168 if (list->size() > 1) { | |
| 169 // Sort the rest by name. | |
| 170 std::sort(list->begin() + 1, list->end(), SortByTaskName); | |
| 171 } | |
| 172 } | |
| 173 | |
| 174 URLPatternSet GetAllMatchingPatterns(const FileBrowserHandler* handler, | |
| 175 const std::vector<GURL>& files_list) { | |
| 176 URLPatternSet matching_patterns; | |
| 177 const URLPatternSet& patterns = handler->file_url_patterns(); | |
| 178 for (URLPatternSet::const_iterator pattern_it = patterns.begin(); | |
| 179 pattern_it != patterns.end(); ++pattern_it) { | |
| 180 for (std::vector<GURL>::const_iterator file_it = files_list.begin(); | |
| 181 file_it != files_list.end(); ++file_it) { | |
| 182 if (pattern_it->MatchesURL(*file_it)) { | |
| 183 matching_patterns.AddPattern(*pattern_it); | |
| 184 break; | |
| 185 } | |
| 186 } | |
| 187 } | |
| 188 | |
| 189 return matching_patterns; | |
| 190 } | |
| 191 | |
| 192 // Given the list of selected files, returns array of context menu tasks | |
| 193 // that are shared | |
| 194 bool FindCommonTasks(Profile* profile, | |
| 195 const std::vector<GURL>& files_list, | |
| 196 LastUsedHandlerList* named_action_list) { | |
| 197 named_action_list->clear(); | |
| 198 ActionSet common_tasks; | |
| 199 for (std::vector<GURL>::const_iterator it = files_list.begin(); | |
| 200 it != files_list.end(); ++it) { | |
| 201 ActionSet file_actions; | |
| 202 if (!GetFileBrowserHandlers(profile, *it, &file_actions)) | |
| 203 return false; | |
| 204 // If there is nothing to do for one file, the intersection of tasks for all | |
| 205 // files will be empty at the end. | |
| 206 if (!file_actions.size()) | |
| 207 return true; | |
| 208 | |
| 209 // For the very first file, just copy elements. | |
| 210 if (it == files_list.begin()) { | |
| 211 common_tasks = file_actions; | |
| 212 } else { | |
| 213 if (common_tasks.size()) { | |
| 214 // For all additional files, find intersection between the accumulated | |
| 215 // and file specific set. | |
| 216 ActionSet intersection; | |
| 217 std::set_intersection(common_tasks.begin(), common_tasks.end(), | |
| 218 file_actions.begin(), file_actions.end(), | |
| 219 std::inserter(intersection, | |
| 220 intersection.begin())); | |
| 221 common_tasks = intersection; | |
| 222 } | |
| 223 } | |
| 224 } | |
| 225 | |
| 226 const DictionaryValue* prefs_tasks = | |
| 227 profile->GetPrefs()->GetDictionary(prefs::kLastUsedFileBrowserHandlers); | |
| 228 for (ActionSet::const_iterator iter = common_tasks.begin(); | |
| 229 iter != common_tasks.end(); ++iter) { | |
| 230 // Get timestamp of when this task was used last time. | |
| 231 int last_used_timestamp = 0; | |
| 232 prefs_tasks->GetInteger(MakeTaskID((*iter)->extension_id().data(), | |
| 233 (*iter)->id().data()), | |
| 234 &last_used_timestamp); | |
| 235 URLPatternSet matching_patterns = GetAllMatchingPatterns(*iter, files_list); | |
| 236 named_action_list->push_back(LastUsedHandler(last_used_timestamp, *iter, | |
| 237 matching_patterns)); | |
| 238 } | |
| 239 | |
| 240 SortLastUsedHandlerList(named_action_list); | |
| 241 return true; | |
| 242 } | |
| 243 | |
| 244 ListValue* URLPatternSetToStringList(const URLPatternSet& patterns) { | |
| 245 ListValue* list = new ListValue(); | |
| 246 for (URLPatternSet::const_iterator it = patterns.begin(); | |
| 247 it != patterns.end(); ++it) { | |
| 248 list->Append(new StringValue(it->GetAsString())); | |
| 249 } | |
| 250 | |
| 251 return list; | |
| 252 } | |
| 253 | |
| 254 // Update file handler usage stats. | |
| 255 void UpdateFileHandlerUsageStats(Profile* profile, const std::string& task_id) { | |
| 256 if (!profile || !profile->GetPrefs()) | |
| 257 return; | |
| 258 DictionaryPrefUpdate prefs_usage_update(profile->GetPrefs(), | |
| 259 prefs::kLastUsedFileBrowserHandlers); | |
| 260 prefs_usage_update->SetWithoutPathExpansion(task_id, | |
| 261 new base::FundamentalValue( | |
| 262 static_cast<int>(base::Time::Now().ToInternalValue()/ | |
| 263 base::Time::kMicrosecondsPerSecond))); | |
| 264 } | |
| 265 | |
| 266 #ifdef OS_CHROMEOS | |
| 267 base::DictionaryValue* MountPointToValue(Profile* profile, | |
| 268 const chromeos::disks::DiskMountManager::MountPointInfo& mount_point_info) { | |
| 269 | |
| 270 base::DictionaryValue *mount_info = new base::DictionaryValue(); | |
| 271 | |
| 272 mount_info->SetString("mountType", | |
| 273 chromeos::disks::DiskMountManager::MountTypeToString( | |
| 274 mount_point_info.mount_type)); | |
| 275 | |
| 276 if (mount_point_info.mount_type == chromeos::MOUNT_TYPE_ARCHIVE) { | |
| 277 GURL source_url; | |
| 278 if (file_manager_util::ConvertFileToFileSystemUrl(profile, | |
| 279 FilePath(mount_point_info.source_path), | |
| 280 file_manager_util::GetFileBrowserExtensionUrl().GetOrigin(), | |
| 281 &source_url)) { | |
| 282 mount_info->SetString("sourceUrl", source_url.spec()); | |
| 283 } | |
| 284 } else { | |
| 285 mount_info->SetString("sourceUrl", mount_point_info.source_path); | |
| 286 } | |
| 287 | |
| 288 FilePath relative_mount_path; | |
| 289 // Convert mount point path to relative path with the external file system | |
| 290 // exposed within File API. | |
| 291 if (file_manager_util::ConvertFileToRelativeFileSystemPath(profile, | |
| 292 FilePath(mount_point_info.mount_path), | |
| 293 &relative_mount_path)) { | |
| 294 mount_info->SetString("mountPath", relative_mount_path.value()); | |
| 295 } | |
| 296 | |
| 297 mount_info->SetString("mountCondition", | |
| 298 chromeos::disks::DiskMountManager::MountConditionToString( | |
| 299 mount_point_info.mount_condition)); | |
| 300 | |
| 301 return mount_info; | |
| 302 } | |
| 303 #endif | |
| 304 | |
| 305 } // namespace | |
| 306 | |
| 307 class RequestLocalFileSystemFunction::LocalFileSystemCallbackDispatcher | |
| 308 : public fileapi::FileSystemCallbackDispatcher { | |
| 309 public: | |
| 310 explicit LocalFileSystemCallbackDispatcher( | |
| 311 RequestLocalFileSystemFunction* function, | |
| 312 Profile* profile, | |
| 313 int child_id, | |
| 314 scoped_refptr<const Extension> extension) | |
| 315 : function_(function), | |
| 316 profile_(profile), | |
| 317 child_id_(child_id), | |
| 318 extension_(extension) { | |
| 319 DCHECK(function_); | |
| 320 } | |
| 321 | |
| 322 // fileapi::FileSystemCallbackDispatcher overrides. | |
| 323 virtual void DidSucceed() OVERRIDE { | |
| 324 NOTREACHED(); | |
| 325 } | |
| 326 | |
| 327 virtual void DidReadMetadata(const base::PlatformFileInfo& info, | |
| 328 const FilePath& unused) OVERRIDE { | |
| 329 NOTREACHED(); | |
| 330 } | |
| 331 | |
| 332 virtual void DidReadDirectory( | |
| 333 const std::vector<base::FileUtilProxy::Entry>& entries, | |
| 334 bool has_more) OVERRIDE { | |
| 335 NOTREACHED(); | |
| 336 } | |
| 337 | |
| 338 virtual void DidWrite(int64 bytes, bool complete) OVERRIDE { | |
| 339 NOTREACHED(); | |
| 340 } | |
| 341 | |
| 342 virtual void DidOpenFileSystem(const std::string& name, | |
| 343 const GURL& root_path) OVERRIDE { | |
| 344 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); | |
| 345 // Set up file permission access. | |
| 346 if (!SetupFileSystemAccessPermissions()) { | |
| 347 DidFail(base::PLATFORM_FILE_ERROR_SECURITY); | |
| 348 return; | |
| 349 } | |
| 350 | |
| 351 BrowserThread::PostTask( | |
| 352 BrowserThread::UI, FROM_HERE, | |
| 353 base::Bind( | |
| 354 &RequestLocalFileSystemFunction::RespondSuccessOnUIThread, | |
| 355 function_, | |
| 356 name, | |
| 357 root_path)); | |
| 358 } | |
| 359 | |
| 360 virtual void DidFail(base::PlatformFileError error_code) OVERRIDE { | |
| 361 BrowserThread::PostTask( | |
| 362 BrowserThread::UI, FROM_HERE, | |
| 363 base::Bind( | |
| 364 &RequestLocalFileSystemFunction::RespondFailedOnUIThread, | |
| 365 function_, | |
| 366 error_code)); | |
| 367 } | |
| 368 | |
| 369 private: | |
| 370 | |
| 371 // Grants file system access permissions to file browser component. | |
| 372 bool SetupFileSystemAccessPermissions() { | |
| 373 if (!extension_.get()) | |
| 374 return false; | |
| 375 | |
| 376 // Make sure that only component extension can access the entire | |
| 377 // local file system. | |
| 378 if (extension_->location() != Extension::COMPONENT | |
| 379 #ifndef NDEBUG | |
| 380 && !CommandLine::ForCurrentProcess()->HasSwitch( | |
| 381 switches::kExposePrivateExtensionApi) | |
| 382 #endif | |
| 383 ) { | |
| 384 NOTREACHED() << "Private method access by non-component extension " | |
| 385 << extension_->id(); | |
| 386 return false; | |
| 387 } | |
| 388 | |
| 389 fileapi::FileSystemPathManager* path_manager = | |
| 390 profile_->GetFileSystemContext()->path_manager(); | |
| 391 fileapi::ExternalFileSystemMountPointProvider* provider = | |
| 392 path_manager->external_provider(); | |
| 393 if (!provider) | |
| 394 return false; | |
| 395 | |
| 396 // Grant full access to File API from this component extension. | |
| 397 provider->GrantFullAccessToExtension(extension_->id()); | |
| 398 | |
| 399 // Grant R/W file permissions to the renderer hosting component | |
| 400 // extension for all paths exposed by our local file system provider. | |
| 401 std::vector<FilePath> root_dirs = provider->GetRootDirectories(); | |
| 402 for (std::vector<FilePath>::iterator iter = root_dirs.begin(); | |
| 403 iter != root_dirs.end(); | |
| 404 ++iter) { | |
| 405 ChildProcessSecurityPolicy::GetInstance()->GrantPermissionsForFile( | |
| 406 child_id_, *iter, kReadWriteFilePermissions); | |
| 407 } | |
| 408 return true; | |
| 409 } | |
| 410 | |
| 411 RequestLocalFileSystemFunction* function_; | |
| 412 Profile* profile_; | |
| 413 // Renderer process id. | |
| 414 int child_id_; | |
| 415 // Extension source URL. | |
| 416 scoped_refptr<const Extension> extension_; | |
| 417 DISALLOW_COPY_AND_ASSIGN(LocalFileSystemCallbackDispatcher); | |
| 418 }; | |
| 419 | |
| 420 void RequestLocalFileSystemFunction::RequestOnFileThread( | |
| 421 const GURL& source_url, int child_id) { | |
| 422 fileapi::FileSystemOperation* operation = | |
| 423 new fileapi::FileSystemOperation( | |
| 424 new LocalFileSystemCallbackDispatcher( | |
| 425 this, | |
| 426 profile(), | |
| 427 child_id, | |
| 428 GetExtension()), | |
| 429 BrowserThread::GetMessageLoopProxyForThread(BrowserThread::FILE), | |
| 430 profile()->GetFileSystemContext()); | |
| 431 GURL origin_url = source_url.GetOrigin(); | |
| 432 operation->OpenFileSystem(origin_url, fileapi::kFileSystemTypeExternal, | |
| 433 false); // create | |
| 434 } | |
| 435 | |
| 436 bool RequestLocalFileSystemFunction::RunImpl() { | |
| 437 if (!dispatcher() || !render_view_host() || !render_view_host()->process()) | |
| 438 return false; | |
| 439 | |
| 440 BrowserThread::PostTask( | |
| 441 BrowserThread::FILE, FROM_HERE, | |
| 442 base::Bind( | |
| 443 &RequestLocalFileSystemFunction::RequestOnFileThread, | |
| 444 this, | |
| 445 source_url_, | |
| 446 render_view_host()->process()->GetID())); | |
| 447 // Will finish asynchronously. | |
| 448 return true; | |
| 449 } | |
| 450 | |
| 451 void RequestLocalFileSystemFunction::RespondSuccessOnUIThread( | |
| 452 const std::string& name, const GURL& root_path) { | |
| 453 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 454 result_.reset(new DictionaryValue()); | |
| 455 DictionaryValue* dict = reinterpret_cast<DictionaryValue*>(result_.get()); | |
| 456 dict->SetString("name", name); | |
| 457 dict->SetString("path", root_path.spec()); | |
| 458 dict->SetInteger("error", base::PLATFORM_FILE_OK); | |
| 459 SendResponse(true); | |
| 460 } | |
| 461 | |
| 462 void RequestLocalFileSystemFunction::RespondFailedOnUIThread( | |
| 463 base::PlatformFileError error_code) { | |
| 464 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 465 error_ = base::StringPrintf(kFileError, static_cast<int>(error_code)); | |
| 466 SendResponse(false); | |
| 467 } | |
| 468 | |
| 469 bool FileWatchBrowserFunctionBase::GetLocalFilePath( | |
| 470 const GURL& file_url, FilePath* local_path, FilePath* virtual_path) { | |
| 471 fileapi::FileSystemPathManager* path_manager = | |
| 472 profile_->GetFileSystemContext()->path_manager(); | |
| 473 GURL file_origin_url; | |
| 474 fileapi::FileSystemType type; | |
| 475 if (!CrackFileSystemURL(file_url, &file_origin_url, &type, | |
| 476 virtual_path)) { | |
| 477 return false; | |
| 478 } | |
| 479 if (type != fileapi::kFileSystemTypeExternal) | |
| 480 return false; | |
| 481 | |
| 482 FilePath root_path = | |
| 483 path_manager->ValidateFileSystemRootAndGetPathOnFileThread( | |
| 484 file_origin_url, | |
| 485 fileapi::kFileSystemTypeExternal, | |
| 486 *virtual_path, | |
| 487 false); | |
| 488 if (root_path == FilePath()) | |
| 489 return false; | |
| 490 | |
| 491 *local_path = root_path.Append(*virtual_path); | |
| 492 return true; | |
| 493 } | |
| 494 | |
| 495 void FileWatchBrowserFunctionBase::RespondOnUIThread(bool success) { | |
| 496 result_.reset(Value::CreateBooleanValue(success)); | |
| 497 SendResponse(success); | |
| 498 } | |
| 499 | |
| 500 bool FileWatchBrowserFunctionBase::RunImpl() { | |
| 501 if (!render_view_host() || !render_view_host()->process()) | |
| 502 return false; | |
| 503 | |
| 504 // First param is url of a file to watch. | |
| 505 std::string url; | |
| 506 if (!args_->GetString(0, &url) || url.empty()) | |
| 507 return false; | |
| 508 | |
| 509 GURL file_watch_url(url); | |
| 510 BrowserThread::PostTask( | |
| 511 BrowserThread::FILE, FROM_HERE, | |
| 512 base::Bind( | |
| 513 &FileWatchBrowserFunctionBase::RunFileWatchOperationOnFileThread, | |
| 514 this, | |
| 515 file_watch_url, | |
| 516 extension_id())); | |
| 517 | |
| 518 return true; | |
| 519 } | |
| 520 | |
| 521 void FileWatchBrowserFunctionBase::RunFileWatchOperationOnFileThread( | |
| 522 const GURL& file_url, const std::string& extension_id) { | |
| 523 FilePath local_path; | |
| 524 FilePath virtual_path; | |
| 525 if (!GetLocalFilePath(file_url, &local_path, &virtual_path) || | |
| 526 local_path == FilePath()) { | |
| 527 BrowserThread::PostTask( | |
| 528 BrowserThread::UI, FROM_HERE, | |
| 529 base::Bind( | |
| 530 &FileWatchBrowserFunctionBase::RespondOnUIThread, | |
| 531 this, | |
| 532 false)); | |
| 533 } | |
| 534 if (!PerformFileWatchOperation(local_path, virtual_path, extension_id)) { | |
| 535 BrowserThread::PostTask( | |
| 536 BrowserThread::UI, FROM_HERE, | |
| 537 base::Bind( | |
| 538 &FileWatchBrowserFunctionBase::RespondOnUIThread, | |
| 539 this, | |
| 540 false)); | |
| 541 } | |
| 542 BrowserThread::PostTask( | |
| 543 BrowserThread::UI, FROM_HERE, | |
| 544 base::Bind( | |
| 545 &FileWatchBrowserFunctionBase::RespondOnUIThread, | |
| 546 this, | |
| 547 true)); | |
| 548 } | |
| 549 | |
| 550 bool AddFileWatchBrowserFunction::PerformFileWatchOperation( | |
| 551 const FilePath& local_path, const FilePath& virtual_path, | |
| 552 const std::string& extension_id) { | |
| 553 #if defined(OS_CHROMEOS) | |
| 554 return profile_->GetExtensionService()->file_browser_event_router()-> | |
| 555 AddFileWatch(local_path, virtual_path, extension_id); | |
| 556 #else | |
| 557 return true; | |
| 558 #endif // OS_CHROMEOS | |
| 559 } | |
| 560 | |
| 561 bool RemoveFileWatchBrowserFunction::PerformFileWatchOperation( | |
| 562 const FilePath& local_path, const FilePath& unused, | |
| 563 const std::string& extension_id) { | |
| 564 #if defined(OS_CHROMEOS) | |
| 565 profile_->GetExtensionService()->file_browser_event_router()-> | |
| 566 RemoveFileWatch(local_path, extension_id); | |
| 567 #endif | |
| 568 return true; | |
| 569 } | |
| 570 | |
| 571 bool GetFileTasksFileBrowserFunction::RunImpl() { | |
| 572 ListValue* files_list = NULL; | |
| 573 if (!args_->GetList(0, &files_list)) | |
| 574 return false; | |
| 575 | |
| 576 std::vector<GURL> file_urls; | |
| 577 for (size_t i = 0; i < files_list->GetSize(); ++i) { | |
| 578 std::string file_url; | |
| 579 if (!files_list->GetString(i, &file_url)) | |
| 580 return false; | |
| 581 | |
| 582 // We need case-insensitive matching, and pattern in handler is already | |
| 583 // in lower case. | |
| 584 StringToLowerASCII(&file_url); | |
| 585 file_urls.push_back(GURL(file_url)); | |
| 586 } | |
| 587 | |
| 588 ListValue* result_list = new ListValue(); | |
| 589 result_.reset(result_list); | |
| 590 | |
| 591 LastUsedHandlerList common_tasks; | |
| 592 if (!FindCommonTasks(profile_, file_urls, &common_tasks)) | |
| 593 return false; | |
| 594 | |
| 595 ExtensionService* service = profile_->GetExtensionService(); | |
| 596 for (LastUsedHandlerList::const_iterator iter = common_tasks.begin(); | |
| 597 iter != common_tasks.end(); | |
| 598 ++iter) { | |
| 599 const FileBrowserHandler* handler = iter->handler; | |
| 600 const std::string extension_id = handler->extension_id(); | |
| 601 const Extension* extension = service->GetExtensionById(extension_id, false); | |
| 602 CHECK(extension); | |
| 603 DictionaryValue* task = new DictionaryValue(); | |
| 604 task->SetString("taskId", MakeTaskID(extension_id.data(), | |
| 605 handler->id().data())); | |
| 606 task->SetString("title", handler->title()); | |
| 607 task->Set("patterns", URLPatternSetToStringList(iter->patterns)); | |
| 608 // TODO(zelidrag): Figure out how to expose icon URL that task defined in | |
| 609 // manifest instead of the default extension icon. | |
| 610 GURL icon = | |
| 611 ExtensionIconSource::GetIconURL(extension, | |
| 612 Extension::EXTENSION_ICON_BITTY, | |
| 613 ExtensionIconSet::MATCH_BIGGER, | |
| 614 false, NULL); // grayscale | |
| 615 task->SetString("iconUrl", icon.spec()); | |
| 616 result_list->Append(task); | |
| 617 } | |
| 618 | |
| 619 // TODO(zelidrag, serya): Add intent content tasks to result_list once we | |
| 620 // implement that API. | |
| 621 SendResponse(true); | |
| 622 return true; | |
| 623 } | |
| 624 | |
| 625 class ExecuteTasksFileBrowserFunction::ExecuteTasksFileSystemCallbackDispatcher | |
| 626 : public fileapi::FileSystemCallbackDispatcher { | |
| 627 public: | |
| 628 explicit ExecuteTasksFileSystemCallbackDispatcher( | |
| 629 ExecuteTasksFileBrowserFunction* function, | |
| 630 Profile* profile, | |
| 631 int child_id, | |
| 632 const GURL& source_url, | |
| 633 scoped_refptr<const Extension> extension, | |
| 634 const std::string task_id, | |
| 635 const std::vector<GURL>& file_urls) | |
| 636 : function_(function), | |
| 637 target_process_id_(0), | |
| 638 profile_(profile), | |
| 639 source_url_(source_url), | |
| 640 extension_(extension), | |
| 641 task_id_(task_id), | |
| 642 origin_file_urls_(file_urls) { | |
| 643 DCHECK(function_); | |
| 644 ExtractTargetExtensionAndProcessID(); | |
| 645 } | |
| 646 | |
| 647 // fileapi::FileSystemCallbackDispatcher overrides. | |
| 648 virtual void DidSucceed() OVERRIDE { | |
| 649 NOTREACHED(); | |
| 650 } | |
| 651 | |
| 652 virtual void DidReadMetadata(const base::PlatformFileInfo& info, | |
| 653 const FilePath& unused) OVERRIDE { | |
| 654 NOTREACHED(); | |
| 655 } | |
| 656 | |
| 657 virtual void DidReadDirectory( | |
| 658 const std::vector<base::FileUtilProxy::Entry>& entries, | |
| 659 bool has_more) OVERRIDE { | |
| 660 NOTREACHED(); | |
| 661 } | |
| 662 | |
| 663 virtual void DidWrite(int64 bytes, bool complete) OVERRIDE { | |
| 664 NOTREACHED(); | |
| 665 } | |
| 666 | |
| 667 virtual void DidOpenFileSystem(const std::string& file_system_name, | |
| 668 const GURL& file_system_root) OVERRIDE { | |
| 669 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); | |
| 670 ExecuteTasksFileBrowserFunction::FileDefinitionList file_list; | |
| 671 for (std::vector<GURL>::iterator iter = origin_file_urls_.begin(); | |
| 672 iter != origin_file_urls_.end(); | |
| 673 ++iter) { | |
| 674 // Set up file permission access. | |
| 675 ExecuteTasksFileBrowserFunction::FileDefinition file; | |
| 676 if (!SetupFileAccessPermissions(*iter, &file.target_file_url, | |
| 677 &file.virtual_path, &file.is_directory)) { | |
| 678 continue; | |
| 679 } | |
| 680 file_list.push_back(file); | |
| 681 } | |
| 682 if (file_list.empty()) { | |
| 683 BrowserThread::PostTask( | |
| 684 BrowserThread::UI, FROM_HERE, | |
| 685 base::Bind( | |
| 686 &ExecuteTasksFileBrowserFunction::ExecuteFailedOnUIThread, | |
| 687 function_)); | |
| 688 return; | |
| 689 } | |
| 690 | |
| 691 BrowserThread::PostTask( | |
| 692 BrowserThread::UI, FROM_HERE, | |
| 693 base::Bind( | |
| 694 &ExecuteTasksFileBrowserFunction::ExecuteFileActionsOnUIThread, | |
| 695 function_, | |
| 696 task_id_, | |
| 697 file_system_name, | |
| 698 file_system_root, | |
| 699 file_list)); | |
| 700 } | |
| 701 | |
| 702 virtual void DidFail(base::PlatformFileError error_code) OVERRIDE { | |
| 703 BrowserThread::PostTask( | |
| 704 BrowserThread::UI, FROM_HERE, | |
| 705 base::Bind( | |
| 706 &ExecuteTasksFileBrowserFunction::ExecuteFailedOnUIThread, | |
| 707 function_)); | |
| 708 } | |
| 709 | |
| 710 private: | |
| 711 // Extracts target extension's id and process from the tasks's id. | |
| 712 void ExtractTargetExtensionAndProcessID() { | |
| 713 // Get task details. | |
| 714 std::string action_id; | |
| 715 if (!CrackTaskIdentifier(task_id_, &target_extension_id_, &action_id)) | |
| 716 return; | |
| 717 | |
| 718 GURL extension_url = | |
| 719 Extension::GetBaseURLFromExtensionId(target_extension_id_); | |
| 720 ExtensionProcessManager* manager = profile_->GetExtensionProcessManager(); | |
| 721 | |
| 722 SiteInstance* site_instance = manager->GetSiteInstanceForURL(extension_url); | |
| 723 if (!site_instance || !site_instance->HasProcess()) | |
| 724 return; | |
| 725 content::RenderProcessHost* process = site_instance->GetProcess(); | |
| 726 | |
| 727 target_process_id_ = process->GetID(); | |
| 728 } | |
| 729 | |
| 730 // Checks legitimacy of file url and grants file RO access permissions from | |
| 731 // handler (target) extension and its renderer process. | |
| 732 bool SetupFileAccessPermissions(const GURL& origin_file_url, | |
| 733 GURL* target_file_url, FilePath* file_path, bool* is_directory) { | |
| 734 if (!extension_.get()) | |
| 735 return false; | |
| 736 | |
| 737 if (target_process_id_ == 0) | |
| 738 return false; | |
| 739 | |
| 740 GURL file_origin_url; | |
| 741 FilePath virtual_path; | |
| 742 fileapi::FileSystemType type; | |
| 743 if (!CrackFileSystemURL(origin_file_url, &file_origin_url, &type, | |
| 744 &virtual_path)) { | |
| 745 return false; | |
| 746 } | |
| 747 | |
| 748 if (type != fileapi::kFileSystemTypeExternal) | |
| 749 return false; | |
| 750 | |
| 751 fileapi::FileSystemPathManager* path_manager = | |
| 752 profile_->GetFileSystemContext()->path_manager(); | |
| 753 if (!path_manager->IsAccessAllowed(file_origin_url, | |
| 754 type, | |
| 755 virtual_path)) { | |
| 756 return false; | |
| 757 } | |
| 758 | |
| 759 // Make sure this url really being used by the right caller extension. | |
| 760 if (source_url_.GetOrigin() != file_origin_url) { | |
| 761 DidFail(base::PLATFORM_FILE_ERROR_SECURITY); | |
| 762 return false; | |
| 763 } | |
| 764 | |
| 765 FilePath root_path = | |
| 766 path_manager->ValidateFileSystemRootAndGetPathOnFileThread( | |
| 767 file_origin_url, | |
| 768 fileapi::kFileSystemTypeExternal, | |
| 769 virtual_path, | |
| 770 false); // create | |
| 771 FilePath final_file_path = root_path.Append(virtual_path); | |
| 772 | |
| 773 fileapi::ExternalFileSystemMountPointProvider* external_provider = | |
| 774 path_manager->external_provider(); | |
| 775 if (!external_provider) | |
| 776 return false; | |
| 777 | |
| 778 // Check if this file system entry exists first. | |
| 779 base::PlatformFileInfo file_info; | |
| 780 | |
| 781 if (!file_util::PathExists(final_file_path) || | |
| 782 file_util::IsLink(final_file_path) || | |
| 783 !file_util::GetFileInfo(final_file_path, &file_info)) | |
| 784 return false; | |
| 785 | |
| 786 // TODO(zelidrag): Let's just prevent all symlinks for now. We don't want a | |
| 787 // USB drive content to point to something in the rest of the file system. | |
| 788 // Ideally, we should permit symlinks within the boundary of the same | |
| 789 // virtual mount point. | |
| 790 if (file_info.is_symbolic_link) | |
| 791 return false; | |
| 792 | |
| 793 // TODO(zelidrag): Add explicit R/W + R/O permissions for non-component | |
| 794 // extensions. | |
| 795 | |
| 796 // Grant R/O access permission to non-component extension and R/W to | |
| 797 // component extensions. | |
| 798 ChildProcessSecurityPolicy::GetInstance()->GrantPermissionsForFile( | |
| 799 target_process_id_, | |
| 800 final_file_path, | |
| 801 extension_->location() != Extension::COMPONENT ? | |
| 802 kReadOnlyFilePermissions : kReadWriteFilePermissions); | |
| 803 | |
| 804 // Grant access to this particular file to target extension. This will | |
| 805 // ensure that the target extension can access only this FS entry and | |
| 806 // prevent from traversing FS hierarchy upward. | |
| 807 external_provider->GrantFileAccessToExtension(target_extension_id_, | |
| 808 virtual_path); | |
| 809 | |
| 810 // Output values. | |
| 811 GURL target_origin_url(Extension::GetBaseURLFromExtensionId( | |
| 812 target_extension_id_)); | |
| 813 GURL base_url = fileapi::GetFileSystemRootURI(target_origin_url, | |
| 814 fileapi::kFileSystemTypeExternal); | |
| 815 *target_file_url = GURL(base_url.spec() + virtual_path.value()); | |
| 816 FilePath root(FILE_PATH_LITERAL("/")); | |
| 817 *file_path = root.Append(virtual_path); | |
| 818 *is_directory = file_info.is_directory; | |
| 819 return true; | |
| 820 } | |
| 821 | |
| 822 ExecuteTasksFileBrowserFunction* function_; | |
| 823 int target_process_id_; | |
| 824 Profile* profile_; | |
| 825 // Extension source URL. | |
| 826 GURL source_url_; | |
| 827 scoped_refptr<const Extension> extension_; | |
| 828 std::string task_id_; | |
| 829 std::string target_extension_id_; | |
| 830 std::vector<GURL> origin_file_urls_; | |
| 831 DISALLOW_COPY_AND_ASSIGN(ExecuteTasksFileSystemCallbackDispatcher); | |
| 832 }; | |
| 833 | |
| 834 bool ExecuteTasksFileBrowserFunction::RunImpl() { | |
| 835 // First param is task id that was to the extension with getFileTasks call. | |
| 836 std::string task_id; | |
| 837 if (!args_->GetString(0, &task_id) || !task_id.size()) | |
| 838 return false; | |
| 839 | |
| 840 // The second param is the list of files that need to be executed with this | |
| 841 // task. | |
| 842 ListValue* files_list = NULL; | |
| 843 if (!args_->GetList(1, &files_list)) | |
| 844 return false; | |
| 845 | |
| 846 if (!files_list->GetSize()) | |
| 847 return true; | |
| 848 | |
| 849 return InitiateFileTaskExecution(task_id, files_list); | |
| 850 } | |
| 851 | |
| 852 bool ExecuteTasksFileBrowserFunction::InitiateFileTaskExecution( | |
| 853 const std::string& task_id, ListValue* files_list) { | |
| 854 std::vector<GURL> file_urls; | |
| 855 for (size_t i = 0; i < files_list->GetSize(); i++) { | |
| 856 std::string origin_file_url; | |
| 857 if (!files_list->GetString(i, &origin_file_url)) { | |
| 858 error_ = kInvalidFileUrl; | |
| 859 return false; | |
| 860 } | |
| 861 file_urls.push_back(GURL(origin_file_url)); | |
| 862 } | |
| 863 // Get local file system instance on file thread. | |
| 864 BrowserThread::PostTask( | |
| 865 BrowserThread::FILE, FROM_HERE, | |
| 866 base::Bind( | |
| 867 &ExecuteTasksFileBrowserFunction::RequestFileEntryOnFileThread, | |
| 868 this, | |
| 869 source_url_, | |
| 870 task_id, | |
| 871 file_urls)); | |
| 872 result_.reset(new base::FundamentalValue(true)); | |
| 873 return true; | |
| 874 } | |
| 875 | |
| 876 void ExecuteTasksFileBrowserFunction::RequestFileEntryOnFileThread( | |
| 877 const GURL& source_url, const std::string& task_id, | |
| 878 const std::vector<GURL>& file_urls) { | |
| 879 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); | |
| 880 fileapi::FileSystemOperation* operation = | |
| 881 new fileapi::FileSystemOperation( | |
| 882 new ExecuteTasksFileSystemCallbackDispatcher( | |
| 883 this, | |
| 884 profile(), | |
| 885 render_view_host()->process()->GetID(), | |
| 886 source_url, | |
| 887 GetExtension(), | |
| 888 task_id, | |
| 889 file_urls), | |
| 890 BrowserThread::GetMessageLoopProxyForThread(BrowserThread::FILE), | |
| 891 profile()->GetFileSystemContext()); | |
| 892 GURL origin_url = source_url.GetOrigin(); | |
| 893 operation->OpenFileSystem(origin_url, fileapi::kFileSystemTypeExternal, | |
| 894 false); // create | |
| 895 } | |
| 896 | |
| 897 void ExecuteTasksFileBrowserFunction::ExecuteFailedOnUIThread() { | |
| 898 SendResponse(false); | |
| 899 } | |
| 900 | |
| 901 | |
| 902 void ExecuteTasksFileBrowserFunction::ExecuteFileActionsOnUIThread( | |
| 903 const std::string& task_id, | |
| 904 const std::string& file_system_name, | |
| 905 const GURL& file_system_root, | |
| 906 const FileDefinitionList& file_list) { | |
| 907 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 908 ExtensionService* service = profile_->GetExtensionService(); | |
| 909 if (!service) | |
| 910 return; | |
| 911 // Get task details. | |
| 912 std::string handler_extension_id; | |
| 913 std::string action_id; | |
| 914 if (!CrackTaskIdentifier(task_id, &handler_extension_id, | |
| 915 &action_id)) { | |
| 916 LOG(WARNING) << "Invalid task " << task_id; | |
| 917 SendResponse(false); | |
| 918 return; | |
| 919 } | |
| 920 | |
| 921 const Extension* extension = service->GetExtensionById(handler_extension_id, | |
| 922 false); | |
| 923 if (!extension) { | |
| 924 SendResponse(false); | |
| 925 return; | |
| 926 } | |
| 927 | |
| 928 ExtensionEventRouter* event_router = profile_->GetExtensionEventRouter(); | |
| 929 if (!event_router) { | |
| 930 SendResponse(false); | |
| 931 return; | |
| 932 } | |
| 933 | |
| 934 scoped_ptr<ListValue> event_args(new ListValue()); | |
| 935 event_args->Append(Value::CreateStringValue(action_id)); | |
| 936 DictionaryValue* details = new DictionaryValue(); | |
| 937 event_args->Append(details); | |
| 938 // Get file definitions. These will be replaced with Entry instances by | |
| 939 // chromeHidden.Event.dispatchJSON() method from even_binding.js. | |
| 940 ListValue* files_urls = new ListValue(); | |
| 941 details->Set("entries", files_urls); | |
| 942 for (FileDefinitionList::const_iterator iter = file_list.begin(); | |
| 943 iter != file_list.end(); | |
| 944 ++iter) { | |
| 945 DictionaryValue* file_def = new DictionaryValue(); | |
| 946 files_urls->Append(file_def); | |
| 947 file_def->SetString("fileSystemName", file_system_name); | |
| 948 file_def->SetString("fileSystemRoot", file_system_root.spec()); | |
| 949 file_def->SetString("fileFullPath", iter->virtual_path.value()); | |
| 950 file_def->SetBoolean("fileIsDirectory", iter->is_directory); | |
| 951 } | |
| 952 // Get tab id. | |
| 953 Browser* browser = GetCurrentBrowser(); | |
| 954 if (browser) { | |
| 955 TabContents* contents = browser->GetSelectedTabContents(); | |
| 956 if (contents) | |
| 957 details->SetInteger("tab_id", ExtensionTabUtil::GetTabId(contents)); | |
| 958 } | |
| 959 | |
| 960 UpdateFileHandlerUsageStats(profile_, task_id); | |
| 961 | |
| 962 std::string json_args; | |
| 963 base::JSONWriter::Write(event_args.get(), false, &json_args); | |
| 964 event_router->DispatchEventToExtension( | |
| 965 handler_extension_id, std::string("fileBrowserHandler.onExecute"), | |
| 966 json_args, profile_, | |
| 967 GURL()); | |
| 968 SendResponse(true); | |
| 969 } | |
| 970 | |
| 971 FileBrowserFunction::FileBrowserFunction() { | |
| 972 } | |
| 973 | |
| 974 FileBrowserFunction::~FileBrowserFunction() { | |
| 975 } | |
| 976 | |
| 977 int32 FileBrowserFunction::GetTabId() const { | |
| 978 if (!dispatcher()) { | |
| 979 LOG(WARNING) << "No dispatcher"; | |
| 980 return 0; | |
| 981 } | |
| 982 if (!dispatcher()->delegate()) { | |
| 983 LOG(WARNING) << "No delegate"; | |
| 984 return 0; | |
| 985 } | |
| 986 TabContents* tab_contents = | |
| 987 dispatcher()->delegate()->GetAssociatedTabContents(); | |
| 988 if (!tab_contents) { | |
| 989 LOG(WARNING) << "No associated tab contents"; | |
| 990 return 0; | |
| 991 } | |
| 992 return ExtensionTabUtil::GetTabId(tab_contents); | |
| 993 } | |
| 994 | |
| 995 void FileBrowserFunction::GetLocalPathsOnFileThreadAndRunCallbackOnUIThread( | |
| 996 const UrlList& file_urls, | |
| 997 GetLocalPathsCallback callback) { | |
| 998 BrowserThread::PostTask( | |
| 999 BrowserThread::FILE, FROM_HERE, | |
| 1000 base::Bind( | |
| 1001 &FileBrowserFunction::GetLocalPathsOnFileThread, | |
| 1002 this, | |
| 1003 file_urls, callback)); | |
| 1004 } | |
| 1005 | |
| 1006 // GetFileSystemRootPathOnFileThread can only be called from the file thread, | |
| 1007 // so here we are. This function takes a vector of virtual paths, converts | |
| 1008 // them to local paths and calls |callback| with the result vector, on the UI | |
| 1009 // thread. | |
| 1010 void FileBrowserFunction::GetLocalPathsOnFileThread( | |
| 1011 const UrlList& file_urls, | |
| 1012 GetLocalPathsCallback callback) { | |
| 1013 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); | |
| 1014 FilePathList selected_files; | |
| 1015 | |
| 1016 // FilePath(virtual_path) doesn't work on win, so limit this to ChromeOS. | |
| 1017 #if defined(OS_CHROMEOS) | |
| 1018 GURL origin_url = source_url().GetOrigin(); | |
| 1019 fileapi::FileSystemPathManager* path_manager = | |
| 1020 profile()->GetFileSystemContext()->path_manager(); | |
| 1021 | |
| 1022 size_t len = file_urls.size(); | |
| 1023 selected_files.reserve(len); | |
| 1024 for (size_t i = 0; i < len; ++i) { | |
| 1025 const GURL& file_url = file_urls[i]; | |
| 1026 GURL file_origin_url; | |
| 1027 FilePath virtual_path; | |
| 1028 fileapi::FileSystemType type; | |
| 1029 if (!CrackFileSystemURL(file_url, &file_origin_url, &type, | |
| 1030 &virtual_path)) { | |
| 1031 continue; | |
| 1032 } | |
| 1033 if (type != fileapi::kFileSystemTypeExternal) { | |
| 1034 NOTREACHED(); | |
| 1035 continue; | |
| 1036 } | |
| 1037 FilePath root = path_manager->ValidateFileSystemRootAndGetPathOnFileThread( | |
| 1038 origin_url, | |
| 1039 fileapi::kFileSystemTypeExternal, | |
| 1040 FilePath(virtual_path), | |
| 1041 false); | |
| 1042 if (!root.empty()) { | |
| 1043 selected_files.push_back(root.Append(virtual_path)); | |
| 1044 } else { | |
| 1045 LOG(WARNING) << "GetLocalPathsOnFileThread failed " | |
| 1046 << file_url.spec(); | |
| 1047 } | |
| 1048 } | |
| 1049 #endif | |
| 1050 | |
| 1051 BrowserThread::PostTask(BrowserThread::UI, FROM_HERE, | |
| 1052 base::Bind(callback, selected_files)); | |
| 1053 } | |
| 1054 | |
| 1055 bool SelectFileFunction::RunImpl() { | |
| 1056 if (args_->GetSize() != 2) { | |
| 1057 return false; | |
| 1058 } | |
| 1059 std::string file_url; | |
| 1060 args_->GetString(0, &file_url); | |
| 1061 UrlList file_paths; | |
| 1062 file_paths.push_back(GURL(file_url)); | |
| 1063 | |
| 1064 GetLocalPathsOnFileThreadAndRunCallbackOnUIThread( | |
| 1065 file_paths, | |
| 1066 base::Bind(&SelectFileFunction::GetLocalPathsResponseOnUIThread, this)); | |
| 1067 return true; | |
| 1068 } | |
| 1069 | |
| 1070 void SelectFileFunction::GetLocalPathsResponseOnUIThread( | |
| 1071 const FilePathList& files) { | |
| 1072 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 1073 if (files.size() != 1) { | |
| 1074 SendResponse(false); | |
| 1075 return; | |
| 1076 } | |
| 1077 int index; | |
| 1078 args_->GetInteger(1, &index); | |
| 1079 int32 tab_id = GetTabId(); | |
| 1080 SelectFileDialogExtension::OnFileSelected(tab_id, files[0], index); | |
| 1081 SendResponse(true); | |
| 1082 } | |
| 1083 | |
| 1084 | |
| 1085 ViewFilesFunction::ViewFilesFunction() { | |
| 1086 } | |
| 1087 | |
| 1088 ViewFilesFunction::~ViewFilesFunction() { | |
| 1089 } | |
| 1090 | |
| 1091 bool ViewFilesFunction::RunImpl() { | |
| 1092 if (args_->GetSize() < 1) { | |
| 1093 return false; | |
| 1094 } | |
| 1095 | |
| 1096 ListValue* path_list = NULL; | |
| 1097 args_->GetList(0, &path_list); | |
| 1098 DCHECK(path_list); | |
| 1099 | |
| 1100 std::string internal_task_id; | |
| 1101 args_->GetString(1, &internal_task_id); | |
| 1102 | |
| 1103 std::string virtual_path; | |
| 1104 size_t len = path_list->GetSize(); | |
| 1105 UrlList file_urls; | |
| 1106 file_urls.reserve(len); | |
| 1107 for (size_t i = 0; i < len; ++i) { | |
| 1108 path_list->GetString(i, &virtual_path); | |
| 1109 file_urls.push_back(GURL(virtual_path)); | |
| 1110 } | |
| 1111 | |
| 1112 GetLocalPathsOnFileThreadAndRunCallbackOnUIThread( | |
| 1113 file_urls, | |
| 1114 base::Bind(&ViewFilesFunction::GetLocalPathsResponseOnUIThread, | |
| 1115 this, | |
| 1116 internal_task_id)); | |
| 1117 return true; | |
| 1118 } | |
| 1119 | |
| 1120 void ViewFilesFunction::GetLocalPathsResponseOnUIThread( | |
| 1121 const std::string& internal_task_id, | |
| 1122 const FilePathList& files) { | |
| 1123 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 1124 for (FilePathList::const_iterator iter = files.begin(); | |
| 1125 iter != files.end(); | |
| 1126 ++iter) { | |
| 1127 file_manager_util::ViewItem(*iter, | |
| 1128 internal_task_id == kEnqueueTaskId || | |
| 1129 // Start the first one, enqueue others. | |
| 1130 iter != files.begin()); | |
| 1131 } | |
| 1132 SendResponse(true); | |
| 1133 } | |
| 1134 | |
| 1135 SelectFilesFunction::SelectFilesFunction() { | |
| 1136 } | |
| 1137 | |
| 1138 SelectFilesFunction::~SelectFilesFunction() { | |
| 1139 } | |
| 1140 | |
| 1141 bool SelectFilesFunction::RunImpl() { | |
| 1142 if (args_->GetSize() != 1) { | |
| 1143 return false; | |
| 1144 } | |
| 1145 | |
| 1146 ListValue* path_list = NULL; | |
| 1147 args_->GetList(0, &path_list); | |
| 1148 DCHECK(path_list); | |
| 1149 | |
| 1150 std::string virtual_path; | |
| 1151 size_t len = path_list->GetSize(); | |
| 1152 UrlList file_urls; | |
| 1153 file_urls.reserve(len); | |
| 1154 for (size_t i = 0; i < len; ++i) { | |
| 1155 path_list->GetString(i, &virtual_path); | |
| 1156 file_urls.push_back(GURL(virtual_path)); | |
| 1157 } | |
| 1158 | |
| 1159 GetLocalPathsOnFileThreadAndRunCallbackOnUIThread( | |
| 1160 file_urls, | |
| 1161 base::Bind(&SelectFilesFunction::GetLocalPathsResponseOnUIThread, this)); | |
| 1162 return true; | |
| 1163 } | |
| 1164 | |
| 1165 void SelectFilesFunction::GetLocalPathsResponseOnUIThread( | |
| 1166 const FilePathList& files) { | |
| 1167 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 1168 int32 tab_id = GetTabId(); | |
| 1169 SelectFileDialogExtension::OnMultiFilesSelected(tab_id, files); | |
| 1170 SendResponse(true); | |
| 1171 } | |
| 1172 | |
| 1173 bool CancelFileDialogFunction::RunImpl() { | |
| 1174 int32 tab_id = GetTabId(); | |
| 1175 SelectFileDialogExtension::OnFileSelectionCanceled(tab_id); | |
| 1176 SendResponse(true); | |
| 1177 return true; | |
| 1178 } | |
| 1179 | |
| 1180 AddMountFunction::AddMountFunction() { | |
| 1181 } | |
| 1182 | |
| 1183 AddMountFunction::~AddMountFunction() { | |
| 1184 } | |
| 1185 | |
| 1186 bool AddMountFunction::RunImpl() { | |
| 1187 // The third argument is simply ignored. | |
| 1188 if (args_->GetSize() != 2 && args_->GetSize() != 3) { | |
| 1189 error_ = "Invalid argument count"; | |
| 1190 return false; | |
| 1191 } | |
| 1192 | |
| 1193 std::string file_url; | |
| 1194 if (!args_->GetString(0, &file_url)) { | |
| 1195 return false; | |
| 1196 } | |
| 1197 | |
| 1198 std::string mount_type_str; | |
| 1199 if (!args_->GetString(1, &mount_type_str)) { | |
| 1200 return false; | |
| 1201 } | |
| 1202 | |
| 1203 UrlList file_paths; | |
| 1204 file_paths.push_back(GURL(file_url)); | |
| 1205 | |
| 1206 #if defined(OS_CHROMEOS) | |
| 1207 GetLocalPathsOnFileThreadAndRunCallbackOnUIThread( | |
| 1208 file_paths, | |
| 1209 base::Bind(&AddMountFunction::GetLocalPathsResponseOnUIThread, | |
| 1210 this, | |
| 1211 mount_type_str)); | |
| 1212 #endif // OS_CHROMEOS | |
| 1213 | |
| 1214 return true; | |
| 1215 } | |
| 1216 | |
| 1217 void AddMountFunction::GetLocalPathsResponseOnUIThread( | |
| 1218 const std::string& mount_type_str, | |
| 1219 const FilePathList& files) { | |
| 1220 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 1221 | |
| 1222 if (!files.size()) { | |
| 1223 SendResponse(false); | |
| 1224 return; | |
| 1225 } | |
| 1226 | |
| 1227 #ifdef OS_CHROMEOS | |
| 1228 FilePath::StringType source_file = files[0].value(); | |
| 1229 | |
| 1230 chromeos::disks::DiskMountManager* disk_mount_manager = | |
| 1231 chromeos::disks::DiskMountManager::GetInstance(); | |
| 1232 | |
| 1233 chromeos::MountType mount_type = | |
| 1234 disk_mount_manager->MountTypeFromString(mount_type_str); | |
| 1235 if (mount_type == chromeos::MOUNT_TYPE_INVALID) { | |
| 1236 error_ = "Invalid mount type"; | |
| 1237 SendResponse(false); | |
| 1238 return; | |
| 1239 } | |
| 1240 | |
| 1241 disk_mount_manager->MountPath(source_file.data(), mount_type); | |
| 1242 #endif | |
| 1243 | |
| 1244 SendResponse(true); | |
| 1245 } | |
| 1246 | |
| 1247 RemoveMountFunction::RemoveMountFunction() { | |
| 1248 } | |
| 1249 | |
| 1250 RemoveMountFunction::~RemoveMountFunction() { | |
| 1251 } | |
| 1252 | |
| 1253 bool RemoveMountFunction::RunImpl() { | |
| 1254 if (args_->GetSize() != 1) { | |
| 1255 return false; | |
| 1256 } | |
| 1257 | |
| 1258 std::string mount_path; | |
| 1259 if (!args_->GetString(0, &mount_path)) { | |
| 1260 return false; | |
| 1261 } | |
| 1262 | |
| 1263 UrlList file_paths; | |
| 1264 file_paths.push_back(GURL(mount_path)); | |
| 1265 GetLocalPathsOnFileThreadAndRunCallbackOnUIThread( | |
| 1266 file_paths, | |
| 1267 base::Bind(&RemoveMountFunction::GetLocalPathsResponseOnUIThread, this)); | |
| 1268 return true; | |
| 1269 } | |
| 1270 | |
| 1271 void RemoveMountFunction::GetLocalPathsResponseOnUIThread( | |
| 1272 const FilePathList& files) { | |
| 1273 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 1274 | |
| 1275 if (files.size() != 1) { | |
| 1276 SendResponse(false); | |
| 1277 return; | |
| 1278 } | |
| 1279 #ifdef OS_CHROMEOS | |
| 1280 chromeos::disks::DiskMountManager::GetInstance()->UnmountPath( | |
| 1281 files[0].value()); | |
| 1282 #endif | |
| 1283 | |
| 1284 SendResponse(true); | |
| 1285 } | |
| 1286 | |
| 1287 GetMountPointsFunction::GetMountPointsFunction() { | |
| 1288 } | |
| 1289 | |
| 1290 GetMountPointsFunction::~GetMountPointsFunction() { | |
| 1291 } | |
| 1292 | |
| 1293 bool GetMountPointsFunction::RunImpl() { | |
| 1294 if (args_->GetSize()) | |
| 1295 return false; | |
| 1296 | |
| 1297 base::ListValue *mounts = new base::ListValue(); | |
| 1298 result_.reset(mounts); | |
| 1299 | |
| 1300 #ifdef OS_CHROMEOS | |
| 1301 chromeos::disks::DiskMountManager* disk_mount_manager = | |
| 1302 chromeos::disks::DiskMountManager::GetInstance(); | |
| 1303 chromeos::disks::DiskMountManager::MountPointMap mount_points = | |
| 1304 disk_mount_manager->mount_points(); | |
| 1305 | |
| 1306 for (chromeos::disks::DiskMountManager::MountPointMap::const_iterator it = | |
| 1307 mount_points.begin(); | |
| 1308 it != mount_points.end(); | |
| 1309 ++it) { | |
| 1310 mounts->Append(MountPointToValue(profile_, it->second)); | |
| 1311 } | |
| 1312 #endif | |
| 1313 | |
| 1314 SendResponse(true); | |
| 1315 return true; | |
| 1316 } | |
| 1317 | |
| 1318 GetSizeStatsFunction::GetSizeStatsFunction() { | |
| 1319 } | |
| 1320 | |
| 1321 GetSizeStatsFunction::~GetSizeStatsFunction() { | |
| 1322 } | |
| 1323 | |
| 1324 bool GetSizeStatsFunction::RunImpl() { | |
| 1325 if (args_->GetSize() != 1) { | |
| 1326 return false; | |
| 1327 } | |
| 1328 | |
| 1329 std::string mount_url; | |
| 1330 if (!args_->GetString(0, &mount_url)) | |
| 1331 return false; | |
| 1332 | |
| 1333 UrlList mount_paths; | |
| 1334 mount_paths.push_back(GURL(mount_url)); | |
| 1335 | |
| 1336 GetLocalPathsOnFileThreadAndRunCallbackOnUIThread( | |
| 1337 mount_paths, | |
| 1338 base::Bind(&GetSizeStatsFunction::GetLocalPathsResponseOnUIThread, this)); | |
| 1339 return true; | |
| 1340 } | |
| 1341 | |
| 1342 void GetSizeStatsFunction::GetLocalPathsResponseOnUIThread( | |
| 1343 const FilePathList& files) { | |
| 1344 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 1345 | |
| 1346 if (files.size() != 1) { | |
| 1347 SendResponse(false); | |
| 1348 return; | |
| 1349 } | |
| 1350 | |
| 1351 BrowserThread::PostTask( | |
| 1352 BrowserThread::FILE, FROM_HERE, | |
| 1353 base::Bind( | |
| 1354 &GetSizeStatsFunction::CallGetSizeStatsOnFileThread, | |
| 1355 this, | |
| 1356 files[0].value())); | |
| 1357 } | |
| 1358 | |
| 1359 void GetSizeStatsFunction::CallGetSizeStatsOnFileThread( | |
| 1360 const std::string& mount_path) { | |
| 1361 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE)); | |
| 1362 | |
| 1363 size_t total_size_kb = 0; | |
| 1364 size_t remaining_size_kb = 0; | |
| 1365 #ifdef OS_CHROMEOS | |
| 1366 chromeos::disks::DiskMountManager::GetInstance()-> | |
| 1367 GetSizeStatsOnFileThread(mount_path, &total_size_kb, &remaining_size_kb); | |
| 1368 #endif | |
| 1369 | |
| 1370 BrowserThread::PostTask( | |
| 1371 BrowserThread::UI, FROM_HERE, | |
| 1372 base::Bind( | |
| 1373 &GetSizeStatsFunction::GetSizeStatsCallbackOnUIThread, | |
| 1374 this, | |
| 1375 mount_path, total_size_kb, remaining_size_kb)); | |
| 1376 } | |
| 1377 | |
| 1378 void GetSizeStatsFunction::GetSizeStatsCallbackOnUIThread( | |
| 1379 const std::string& mount_path, | |
| 1380 size_t total_size_kb, | |
| 1381 size_t remaining_size_kb) { | |
| 1382 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 1383 | |
| 1384 base::DictionaryValue* sizes = new base::DictionaryValue(); | |
| 1385 result_.reset(sizes); | |
| 1386 | |
| 1387 sizes->SetInteger("totalSizeKB", total_size_kb); | |
| 1388 sizes->SetInteger("remainingSizeKB", remaining_size_kb); | |
| 1389 | |
| 1390 SendResponse(true); | |
| 1391 } | |
| 1392 | |
| 1393 FormatDeviceFunction::FormatDeviceFunction() { | |
| 1394 } | |
| 1395 | |
| 1396 FormatDeviceFunction::~FormatDeviceFunction() { | |
| 1397 } | |
| 1398 | |
| 1399 bool FormatDeviceFunction::RunImpl() { | |
| 1400 if (args_->GetSize() != 1) { | |
| 1401 return false; | |
| 1402 } | |
| 1403 | |
| 1404 std::string volume_file_url; | |
| 1405 if (!args_->GetString(0, &volume_file_url)) { | |
| 1406 NOTREACHED(); | |
| 1407 return false; | |
| 1408 } | |
| 1409 | |
| 1410 UrlList file_paths; | |
| 1411 file_paths.push_back(GURL(volume_file_url)); | |
| 1412 | |
| 1413 GetLocalPathsOnFileThreadAndRunCallbackOnUIThread( | |
| 1414 file_paths, | |
| 1415 base::Bind(&FormatDeviceFunction::GetLocalPathsResponseOnUIThread, this)); | |
| 1416 return true; | |
| 1417 } | |
| 1418 | |
| 1419 void FormatDeviceFunction::GetLocalPathsResponseOnUIThread( | |
| 1420 const FilePathList& files) { | |
| 1421 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI)); | |
| 1422 | |
| 1423 if (files.size() != 1) { | |
| 1424 SendResponse(false); | |
| 1425 return; | |
| 1426 } | |
| 1427 | |
| 1428 #ifdef OS_CHROMEOS | |
| 1429 chromeos::disks::DiskMountManager::GetInstance()->FormatMountedDevice( | |
| 1430 files[0].value()); | |
| 1431 #endif | |
| 1432 | |
| 1433 SendResponse(true); | |
| 1434 } | |
| 1435 | |
| 1436 GetVolumeMetadataFunction::GetVolumeMetadataFunction() { | |
| 1437 } | |
| 1438 | |
| 1439 GetVolumeMetadataFunction::~GetVolumeMetadataFunction() { | |
| 1440 } | |
| 1441 | |
| 1442 bool GetVolumeMetadataFunction::RunImpl() { | |
| 1443 if (args_->GetSize() != 1) { | |
| 1444 error_ = "Invalid argument count"; | |
| 1445 return false; | |
| 1446 } | |
| 1447 | |
| 1448 std::string volume_device_path; | |
| 1449 if (!args_->GetString(0, &volume_device_path)) { | |
| 1450 NOTREACHED(); | |
| 1451 } | |
| 1452 | |
| 1453 #ifdef OS_CHROMEOS | |
| 1454 chromeos::disks::DiskMountManager* disk_mount_manager = | |
| 1455 chromeos::disks::DiskMountManager::GetInstance(); | |
| 1456 chromeos::disks::DiskMountManager::DiskMap::const_iterator volume_it = | |
| 1457 disk_mount_manager->disks().find(volume_device_path); | |
| 1458 | |
| 1459 if (volume_it != disk_mount_manager->disks().end() && | |
| 1460 !volume_it->second->is_hidden()) { | |
| 1461 chromeos::disks::DiskMountManager::Disk* volume = volume_it->second; | |
| 1462 DictionaryValue* volume_info = new DictionaryValue(); | |
| 1463 result_.reset(volume_info); | |
| 1464 // Localising mount path. | |
| 1465 std::string mount_path; | |
| 1466 if (!volume->mount_path().empty()) { | |
| 1467 FilePath relative_mount_path; | |
| 1468 file_manager_util::ConvertFileToRelativeFileSystemPath(profile_, | |
| 1469 FilePath(volume->mount_path()), &relative_mount_path); | |
| 1470 mount_path = relative_mount_path.value(); | |
| 1471 } | |
| 1472 volume_info->SetString("devicePath", volume->device_path()); | |
| 1473 volume_info->SetString("mountPath", mount_path); | |
| 1474 volume_info->SetString("systemPath", volume->system_path()); | |
| 1475 volume_info->SetString("filePath", volume->file_path()); | |
| 1476 volume_info->SetString("deviceLabel", volume->device_label()); | |
| 1477 volume_info->SetString("driveLabel", volume->drive_label()); | |
| 1478 volume_info->SetString("deviceType", | |
| 1479 DeviceTypeToString(volume->device_type())); | |
| 1480 volume_info->SetInteger("totalSize", volume->total_size_in_bytes()); | |
| 1481 volume_info->SetBoolean("isParent", volume->is_parent()); | |
| 1482 volume_info->SetBoolean("isReadOnly", volume->is_read_only()); | |
| 1483 volume_info->SetBoolean("hasMedia", volume->has_media()); | |
| 1484 volume_info->SetBoolean("isOnBootDevice", volume->on_boot_device()); | |
| 1485 | |
| 1486 return true; | |
| 1487 } | |
| 1488 #endif | |
| 1489 error_ = kVolumeDevicePathNotFound; | |
| 1490 return false; | |
| 1491 } | |
| 1492 | |
| 1493 #ifdef OS_CHROMEOS | |
| 1494 std::string GetVolumeMetadataFunction::DeviceTypeToString( | |
| 1495 chromeos::DeviceType type) { | |
| 1496 switch (type) { | |
| 1497 case chromeos::FLASH: | |
| 1498 return kVolumeTypeFlash; | |
| 1499 case chromeos::OPTICAL: | |
| 1500 return kVolumeTypeOptical; | |
| 1501 case chromeos::HDD: | |
| 1502 return kVolumeTypeHardDrive; | |
| 1503 default: | |
| 1504 break; | |
| 1505 } | |
| 1506 return kVolumeTypeUnknown; | |
| 1507 } | |
| 1508 #endif | |
| 1509 | |
| 1510 bool FileDialogStringsFunction::RunImpl() { | |
| 1511 result_.reset(new DictionaryValue()); | |
| 1512 DictionaryValue* dict = reinterpret_cast<DictionaryValue*>(result_.get()); | |
| 1513 | |
| 1514 #define SET_STRING(ns, id) \ | |
| 1515 dict->SetString(#id, l10n_util::GetStringUTF16(ns##_##id)) | |
| 1516 | |
| 1517 SET_STRING(IDS, WEB_FONT_FAMILY); | |
| 1518 SET_STRING(IDS, WEB_FONT_SIZE); | |
| 1519 | |
| 1520 SET_STRING(IDS_FILE_BROWSER, ROOT_DIRECTORY_LABEL); | |
| 1521 SET_STRING(IDS_FILE_BROWSER, DOWNLOADS_DIRECTORY_LABEL); | |
| 1522 SET_STRING(IDS_FILE_BROWSER, ARCHIVE_DIRECTORY_LABEL); | |
| 1523 SET_STRING(IDS_FILE_BROWSER, REMOVABLE_DIRECTORY_LABEL); | |
| 1524 SET_STRING(IDS_FILE_BROWSER, NAME_COLUMN_LABEL); | |
| 1525 SET_STRING(IDS_FILE_BROWSER, SIZE_COLUMN_LABEL); | |
| 1526 SET_STRING(IDS_FILE_BROWSER, TYPE_COLUMN_LABEL); | |
| 1527 SET_STRING(IDS_FILE_BROWSER, DATE_COLUMN_LABEL); | |
| 1528 SET_STRING(IDS_FILE_BROWSER, PREVIEW_COLUMN_LABEL); | |
| 1529 | |
| 1530 SET_STRING(IDS_FILE_BROWSER, DOWNLOADS_DIRECTORY_WARNING); | |
| 1531 | |
| 1532 SET_STRING(IDS_FILE_BROWSER, ERROR_CREATING_FOLDER); | |
| 1533 SET_STRING(IDS_FILE_BROWSER, ERROR_INVALID_CHARACTER); | |
| 1534 SET_STRING(IDS_FILE_BROWSER, ERROR_RESERVED_NAME); | |
| 1535 SET_STRING(IDS_FILE_BROWSER, ERROR_HIDDEN_NAME); | |
| 1536 SET_STRING(IDS_FILE_BROWSER, ERROR_WHITESPACE_NAME); | |
| 1537 SET_STRING(IDS_FILE_BROWSER, NEW_FOLDER_PROMPT); | |
| 1538 SET_STRING(IDS_FILE_BROWSER, ERROR_NEW_FOLDER_EMPTY_NAME); | |
| 1539 SET_STRING(IDS_FILE_BROWSER, NEW_FOLDER_BUTTON_LABEL); | |
| 1540 SET_STRING(IDS_FILE_BROWSER, FILENAME_LABEL); | |
| 1541 | |
| 1542 SET_STRING(IDS_FILE_BROWSER, DIMENSIONS_LABEL); | |
| 1543 SET_STRING(IDS_FILE_BROWSER, DIMENSIONS_FORMAT); | |
| 1544 | |
| 1545 SET_STRING(IDS_FILE_BROWSER, EJECT_BUTTON); | |
| 1546 SET_STRING(IDS_FILE_BROWSER, IMAGE_DIMENSIONS); | |
| 1547 SET_STRING(IDS_FILE_BROWSER, VOLUME_LABEL); | |
| 1548 SET_STRING(IDS_FILE_BROWSER, READ_ONLY); | |
| 1549 | |
| 1550 SET_STRING(IDS_FILE_BROWSER, ARCHIVE_MOUNT_FAILED); | |
| 1551 SET_STRING(IDS_FILE_BROWSER, MOUNT_ARCHIVE); | |
| 1552 SET_STRING(IDS_FILE_BROWSER, UNMOUNT_ARCHIVE); | |
| 1553 SET_STRING(IDS_FILE_BROWSER, FORMAT_DEVICE); | |
| 1554 | |
| 1555 SET_STRING(IDS_FILE_BROWSER, GALLERY); | |
| 1556 SET_STRING(IDS_FILE_BROWSER, GALLERY_EDIT); | |
| 1557 SET_STRING(IDS_FILE_BROWSER, GALLERY_SHARE); | |
| 1558 SET_STRING(IDS_FILE_BROWSER, GALLERY_ENTER_WHEN_DONE); | |
| 1559 SET_STRING(IDS_FILE_BROWSER, GALLERY_AUTOFIX); | |
| 1560 SET_STRING(IDS_FILE_BROWSER, GALLERY_FIXED); | |
| 1561 SET_STRING(IDS_FILE_BROWSER, GALLERY_CROP); | |
| 1562 SET_STRING(IDS_FILE_BROWSER, GALLERY_EXPOSURE); | |
| 1563 SET_STRING(IDS_FILE_BROWSER, GALLERY_BRIGHTNESS); | |
| 1564 SET_STRING(IDS_FILE_BROWSER, GALLERY_CONTRAST); | |
| 1565 SET_STRING(IDS_FILE_BROWSER, GALLERY_ROTATE_LEFT); | |
| 1566 SET_STRING(IDS_FILE_BROWSER, GALLERY_ROTATE_RIGHT); | |
| 1567 SET_STRING(IDS_FILE_BROWSER, GALLERY_UNDO); | |
| 1568 SET_STRING(IDS_FILE_BROWSER, GALLERY_REDO); | |
| 1569 | |
| 1570 SET_STRING(IDS_FILE_BROWSER, CONFIRM_OVERWRITE_FILE); | |
| 1571 SET_STRING(IDS_FILE_BROWSER, FILE_ALREADY_EXISTS); | |
| 1572 SET_STRING(IDS_FILE_BROWSER, DIRECTORY_ALREADY_EXISTS); | |
| 1573 SET_STRING(IDS_FILE_BROWSER, ERROR_RENAMING); | |
| 1574 SET_STRING(IDS_FILE_BROWSER, RENAME_PROMPT); | |
| 1575 SET_STRING(IDS_FILE_BROWSER, RENAME_BUTTON_LABEL); | |
| 1576 | |
| 1577 SET_STRING(IDS_FILE_BROWSER, ERROR_DELETING); | |
| 1578 SET_STRING(IDS_FILE_BROWSER, DELETE_BUTTON_LABEL); | |
| 1579 | |
| 1580 SET_STRING(IDS_FILE_BROWSER, ERROR_MOVING); | |
| 1581 SET_STRING(IDS_FILE_BROWSER, MOVE_BUTTON_LABEL); | |
| 1582 | |
| 1583 SET_STRING(IDS_FILE_BROWSER, ERROR_PASTING); | |
| 1584 SET_STRING(IDS_FILE_BROWSER, PASTE_BUTTON_LABEL); | |
| 1585 | |
| 1586 SET_STRING(IDS_FILE_BROWSER, COPY_BUTTON_LABEL); | |
| 1587 SET_STRING(IDS_FILE_BROWSER, CUT_BUTTON_LABEL); | |
| 1588 | |
| 1589 SET_STRING(IDS_FILE_BROWSER, SELECTION_COPIED); | |
| 1590 SET_STRING(IDS_FILE_BROWSER, SELECTION_CUT); | |
| 1591 SET_STRING(IDS_FILE_BROWSER, PASTE_STARTED); | |
| 1592 SET_STRING(IDS_FILE_BROWSER, PASTE_SOME_PROGRESS); | |
| 1593 SET_STRING(IDS_FILE_BROWSER, PASTE_COMPLETE); | |
| 1594 SET_STRING(IDS_FILE_BROWSER, PASTE_CANCELLED); | |
| 1595 SET_STRING(IDS_FILE_BROWSER, PASTE_TARGET_EXISTS_ERROR); | |
| 1596 SET_STRING(IDS_FILE_BROWSER, PASTE_FILESYSTEM_ERROR); | |
| 1597 SET_STRING(IDS_FILE_BROWSER, PASTE_UNEXPECTED_ERROR); | |
| 1598 | |
| 1599 SET_STRING(IDS_FILE_BROWSER, DEVICE_TYPE_FLASH); | |
| 1600 SET_STRING(IDS_FILE_BROWSER, DEVICE_TYPE_HDD); | |
| 1601 SET_STRING(IDS_FILE_BROWSER, DEVICE_TYPE_OPTICAL); | |
| 1602 SET_STRING(IDS_FILE_BROWSER, DEVICE_TYPE_UNDEFINED); | |
| 1603 | |
| 1604 SET_STRING(IDS_FILE_BROWSER, CANCEL_LABEL); | |
| 1605 SET_STRING(IDS_FILE_BROWSER, OPEN_LABEL); | |
| 1606 SET_STRING(IDS_FILE_BROWSER, SAVE_LABEL); | |
| 1607 SET_STRING(IDS_FILE_BROWSER, OK_LABEL); | |
| 1608 | |
| 1609 SET_STRING(IDS_FILE_BROWSER, DEFAULT_NEW_FOLDER_NAME); | |
| 1610 SET_STRING(IDS_FILE_BROWSER, MORE_FILES); | |
| 1611 | |
| 1612 SET_STRING(IDS_FILE_BROWSER, CONFIRM_DELETE_ONE); | |
| 1613 SET_STRING(IDS_FILE_BROWSER, CONFIRM_DELETE_SOME); | |
| 1614 | |
| 1615 SET_STRING(IDS_FILE_BROWSER, UNKNOWN_FILESYSTEM_WARNING); | |
| 1616 SET_STRING(IDS_FILE_BROWSER, UNSUPPORTED_FILESYSTEM_WARNING); | |
| 1617 SET_STRING(IDS_FILE_BROWSER, FORMATTING_WARNING); | |
| 1618 | |
| 1619 SET_STRING(IDS_FILE_BROWSER, SELECT_FOLDER_TITLE); | |
| 1620 SET_STRING(IDS_FILE_BROWSER, SELECT_OPEN_FILE_TITLE); | |
| 1621 SET_STRING(IDS_FILE_BROWSER, SELECT_OPEN_MULTI_FILE_TITLE); | |
| 1622 SET_STRING(IDS_FILE_BROWSER, SELECT_SAVEAS_FILE_TITLE); | |
| 1623 | |
| 1624 SET_STRING(IDS_FILE_BROWSER, COMPUTING_SELECTION); | |
| 1625 SET_STRING(IDS_FILE_BROWSER, NOTHING_SELECTED); | |
| 1626 SET_STRING(IDS_FILE_BROWSER, ONE_FILE_SELECTED); | |
| 1627 SET_STRING(IDS_FILE_BROWSER, ONE_DIRECTORY_SELECTED); | |
| 1628 SET_STRING(IDS_FILE_BROWSER, MANY_FILES_SELECTED); | |
| 1629 SET_STRING(IDS_FILE_BROWSER, MANY_DIRECTORIES_SELECTED); | |
| 1630 SET_STRING(IDS_FILE_BROWSER, MANY_ENTRIES_SELECTED); | |
| 1631 | |
| 1632 SET_STRING(IDS_FILE_BROWSER, PLAYBACK_ERROR); | |
| 1633 | |
| 1634 // MP3 metadata extractor plugin | |
| 1635 SET_STRING(IDS_FILE_BROWSER, ID3_ALBUM); // TALB | |
| 1636 SET_STRING(IDS_FILE_BROWSER, ID3_BPM); // TBPM | |
| 1637 SET_STRING(IDS_FILE_BROWSER, ID3_COMPOSER); // TCOM | |
| 1638 SET_STRING(IDS_FILE_BROWSER, ID3_COPYRIGHT_MESSAGE); // TCOP | |
| 1639 SET_STRING(IDS_FILE_BROWSER, ID3_DATE); // TDAT | |
| 1640 SET_STRING(IDS_FILE_BROWSER, ID3_PLAYLIST_DELAY); // TDLY | |
| 1641 SET_STRING(IDS_FILE_BROWSER, ID3_ENCODED_BY); // TENC | |
| 1642 SET_STRING(IDS_FILE_BROWSER, ID3_LYRICIST); // TEXT | |
| 1643 SET_STRING(IDS_FILE_BROWSER, ID3_FILE_TYPE); // TFLT | |
| 1644 SET_STRING(IDS_FILE_BROWSER, ID3_TIME); // TIME | |
| 1645 SET_STRING(IDS_FILE_BROWSER, ID3_TITLE); // TIT2 | |
| 1646 SET_STRING(IDS_FILE_BROWSER, ID3_LENGTH); // TLEN | |
| 1647 SET_STRING(IDS_FILE_BROWSER, ID3_FILE_OWNER); // TOWN | |
| 1648 SET_STRING(IDS_FILE_BROWSER, ID3_LEAD_PERFORMER); // TPE1 | |
| 1649 SET_STRING(IDS_FILE_BROWSER, ID3_BAND); // TPE2 | |
| 1650 SET_STRING(IDS_FILE_BROWSER, ID3_TRACK_NUMBER); // TRCK | |
| 1651 SET_STRING(IDS_FILE_BROWSER, ID3_YEAR); // TYER | |
| 1652 SET_STRING(IDS_FILE_BROWSER, ID3_COPYRIGHT); // WCOP | |
| 1653 SET_STRING(IDS_FILE_BROWSER, ID3_OFFICIAL_AUDIO_FILE_WEBPAGE); // WOAF | |
| 1654 SET_STRING(IDS_FILE_BROWSER, ID3_OFFICIAL_ARTIST); // WOAR | |
| 1655 SET_STRING(IDS_FILE_BROWSER, ID3_OFFICIAL_AUDIO_SOURCE_WEBPAGE); // WOAS | |
| 1656 SET_STRING(IDS_FILE_BROWSER, ID3_PUBLISHERS_OFFICIAL_WEBPAGE); // WPUB | |
| 1657 SET_STRING(IDS_FILE_BROWSER, ID3_USER_DEFINED_URL_LINK_FRAME); // WXXX | |
| 1658 | |
| 1659 // File types | |
| 1660 SET_STRING(IDS_FILE_BROWSER, FOLDER); | |
| 1661 SET_STRING(IDS_FILE_BROWSER, DEVICE); | |
| 1662 SET_STRING(IDS_FILE_BROWSER, IMAGE_FILE_TYPE); | |
| 1663 SET_STRING(IDS_FILE_BROWSER, VIDEO_FILE_TYPE); | |
| 1664 SET_STRING(IDS_FILE_BROWSER, AUDIO_FILE_TYPE); | |
| 1665 SET_STRING(IDS_FILE_BROWSER, HTML_DOCUMENT_FILE_TYPE); | |
| 1666 SET_STRING(IDS_FILE_BROWSER, ZIP_ARCHIVE_FILE_TYPE); | |
| 1667 SET_STRING(IDS_FILE_BROWSER, PLAIN_TEXT_FILE_TYPE); | |
| 1668 SET_STRING(IDS_FILE_BROWSER, PDF_DOCUMENT_FILE_TYPE); | |
| 1669 | |
| 1670 SET_STRING(IDS_FILEBROWSER, ENQUEUE); | |
| 1671 #undef SET_STRING | |
| 1672 | |
| 1673 // TODO(serya): Create a new string in .grd file for this one in M13. | |
| 1674 dict->SetString("PREVIEW_IMAGE", | |
| 1675 l10n_util::GetStringUTF16(IDS_CERT_MANAGER_VIEW_CERT_BUTTON)); | |
| 1676 dict->SetString("PLAY_MEDIA", | |
| 1677 l10n_util::GetStringUTF16(IDS_CONTENT_CONTEXT_PLAY)); | |
| 1678 #if defined(OS_CHROMEOS) | |
| 1679 if (CommandLine::ForCurrentProcess()->HasSwitch(switches::kEnablePhotoEditor)) | |
| 1680 dict->SetString("ENABLE_PHOTO_EDITOR", "true"); | |
| 1681 #endif | |
| 1682 | |
| 1683 return true; | |
| 1684 } | |
| OLD | NEW |