Refactor management of overview window copy lifetime into a separate class.
[chromium-blink-merge.git] / content / shell / browser / shell_download_manager_delegate.cc
blob1dbbea7626b70e01dc119f9fc1462173c5b00f39
1 // Copyright 2013 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.
5 #include "content/shell/browser/shell_download_manager_delegate.h"
7 #if defined(TOOLKIT_GTK)
8 #include <gtk/gtk.h>
9 #endif
11 #if defined(OS_WIN)
12 #include <windows.h>
13 #include <commdlg.h>
14 #endif
16 #include "base/bind.h"
17 #include "base/command_line.h"
18 #include "base/file_util.h"
19 #include "base/logging.h"
20 #include "base/strings/string_util.h"
21 #include "base/strings/utf_string_conversions.h"
22 #include "content/public/browser/browser_context.h"
23 #include "content/public/browser/browser_thread.h"
24 #include "content/public/browser/download_manager.h"
25 #include "content/public/browser/web_contents.h"
26 #include "content/public/browser/web_contents_view.h"
27 #include "content/shell/browser/webkit_test_controller.h"
28 #include "content/shell/common/shell_switches.h"
29 #include "net/base/net_util.h"
31 namespace content {
33 ShellDownloadManagerDelegate::ShellDownloadManagerDelegate()
34 : download_manager_(NULL),
35 suppress_prompting_(false) {
36 // Balanced in Shutdown();
37 AddRef();
40 ShellDownloadManagerDelegate::~ShellDownloadManagerDelegate(){
44 void ShellDownloadManagerDelegate::SetDownloadManager(
45 DownloadManager* download_manager) {
46 download_manager_ = download_manager;
49 void ShellDownloadManagerDelegate::Shutdown() {
50 Release();
53 bool ShellDownloadManagerDelegate::DetermineDownloadTarget(
54 DownloadItem* download,
55 const DownloadTargetCallback& callback) {
56 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
57 // This assignment needs to be here because even at the call to
58 // SetDownloadManager, the system is not fully initialized.
59 if (default_download_path_.empty()) {
60 default_download_path_ = download_manager_->GetBrowserContext()->GetPath().
61 Append(FILE_PATH_LITERAL("Downloads"));
64 if (!download->GetForcedFilePath().empty()) {
65 callback.Run(download->GetForcedFilePath(),
66 DownloadItem::TARGET_DISPOSITION_OVERWRITE,
67 DOWNLOAD_DANGER_TYPE_NOT_DANGEROUS,
68 download->GetForcedFilePath());
69 return true;
72 base::FilePath generated_name = net::GenerateFileName(
73 download->GetURL(),
74 download->GetContentDisposition(),
75 EmptyString(),
76 download->GetSuggestedFilename(),
77 download->GetMimeType(),
78 "download");
80 BrowserThread::PostTask(
81 BrowserThread::FILE,
82 FROM_HERE,
83 base::Bind(
84 &ShellDownloadManagerDelegate::GenerateFilename,
85 this, download->GetId(), callback, generated_name,
86 default_download_path_));
87 return true;
90 bool ShellDownloadManagerDelegate::ShouldOpenDownload(
91 DownloadItem* item,
92 const DownloadOpenDelayedCallback& callback) {
93 if (CommandLine::ForCurrentProcess()->HasSwitch(switches::kDumpRenderTree) &&
94 WebKitTestController::Get()->IsMainWindow(item->GetWebContents()) &&
95 item->GetMimeType() == "text/html") {
96 WebKitTestController::Get()->OpenURL(
97 net::FilePathToFileURL(item->GetFullPath()));
99 return true;
102 void ShellDownloadManagerDelegate::GetNextId(
103 const DownloadIdCallback& callback) {
104 static uint32 next_id = DownloadItem::kInvalidId + 1;
105 callback.Run(next_id++);
108 void ShellDownloadManagerDelegate::GenerateFilename(
109 uint32 download_id,
110 const DownloadTargetCallback& callback,
111 const base::FilePath& generated_name,
112 const base::FilePath& suggested_directory) {
113 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::FILE));
114 if (!base::PathExists(suggested_directory))
115 file_util::CreateDirectory(suggested_directory);
117 base::FilePath suggested_path(suggested_directory.Append(generated_name));
118 BrowserThread::PostTask(
119 BrowserThread::UI,
120 FROM_HERE,
121 base::Bind(
122 &ShellDownloadManagerDelegate::OnDownloadPathGenerated,
123 this, download_id, callback, suggested_path));
126 void ShellDownloadManagerDelegate::OnDownloadPathGenerated(
127 uint32 download_id,
128 const DownloadTargetCallback& callback,
129 const base::FilePath& suggested_path) {
130 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
131 if (suppress_prompting_) {
132 // Testing exit.
133 callback.Run(suggested_path, DownloadItem::TARGET_DISPOSITION_OVERWRITE,
134 DOWNLOAD_DANGER_TYPE_NOT_DANGEROUS,
135 suggested_path.AddExtension(FILE_PATH_LITERAL(".crdownload")));
136 return;
139 ChooseDownloadPath(download_id, callback, suggested_path);
142 void ShellDownloadManagerDelegate::ChooseDownloadPath(
143 uint32 download_id,
144 const DownloadTargetCallback& callback,
145 const base::FilePath& suggested_path) {
146 DCHECK(BrowserThread::CurrentlyOn(BrowserThread::UI));
147 DownloadItem* item = download_manager_->GetDownload(download_id);
148 if (!item || (item->GetState() != DownloadItem::IN_PROGRESS))
149 return;
151 base::FilePath result;
152 #if defined(OS_WIN) && !defined(USE_AURA)
153 std::wstring file_part = base::FilePath(suggested_path).BaseName().value();
154 wchar_t file_name[MAX_PATH];
155 base::wcslcpy(file_name, file_part.c_str(), arraysize(file_name));
156 OPENFILENAME save_as;
157 ZeroMemory(&save_as, sizeof(save_as));
158 save_as.lStructSize = sizeof(OPENFILENAME);
159 save_as.hwndOwner = item->GetWebContents()->GetView()->GetNativeView();
160 save_as.lpstrFile = file_name;
161 save_as.nMaxFile = arraysize(file_name);
163 std::wstring directory;
164 if (!suggested_path.empty())
165 directory = suggested_path.DirName().value();
167 save_as.lpstrInitialDir = directory.c_str();
168 save_as.Flags = OFN_OVERWRITEPROMPT | OFN_EXPLORER | OFN_ENABLESIZING |
169 OFN_NOCHANGEDIR | OFN_PATHMUSTEXIST;
171 if (GetSaveFileName(&save_as))
172 result = base::FilePath(std::wstring(save_as.lpstrFile));
173 #elif defined(TOOLKIT_GTK)
174 GtkWidget *dialog;
175 gfx::NativeWindow parent_window;
176 std::string base_name = base::FilePath(suggested_path).BaseName().value();
178 parent_window = item->GetWebContents()->GetView()->GetTopLevelNativeWindow();
179 dialog = gtk_file_chooser_dialog_new("Save File",
180 parent_window,
181 GTK_FILE_CHOOSER_ACTION_SAVE,
182 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
183 GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT,
184 NULL);
185 gtk_file_chooser_set_do_overwrite_confirmation(GTK_FILE_CHOOSER(dialog),
186 TRUE);
187 gtk_file_chooser_set_current_name(GTK_FILE_CHOOSER(dialog),
188 base_name.c_str());
190 if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT) {
191 char *filename;
192 filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
193 result = base::FilePath(filename);
194 g_free(filename);
196 gtk_widget_destroy(dialog);
197 #else
198 NOTIMPLEMENTED();
199 #endif
201 callback.Run(result, DownloadItem::TARGET_DISPOSITION_PROMPT,
202 DOWNLOAD_DANGER_TYPE_NOT_DANGEROUS, result);
205 void ShellDownloadManagerDelegate::SetDownloadBehaviorForTesting(
206 const base::FilePath& default_download_path) {
207 default_download_path_ = default_download_path;
208 suppress_prompting_ = true;
211 } // namespace content