fix: update the FileSelectHelper to support the new promise API (#18288)
* fix: update the FileSelectHelper to support the new promise API Fixes #18254 So it turns out we've successfully introduced a way to write non-typesafe C++. This fixes two things: * Uses the object the promise resolves * Ensures we attach the Then handler before moving the promise * fix: also fix misuse of Promise::Then in the download manager
This commit is contained in:
		
					parent
					
						
							
								d027be05a6
							
						
					
				
			
			
				commit
				
					
						fde3137b90
					
				
			
		
					 4 changed files with 78 additions and 79 deletions
				
			
		|  | @ -127,8 +127,8 @@ void AtomDownloadManagerDelegate::OnDownloadPathGenerated( | ||||||
|         base::BindOnce(&AtomDownloadManagerDelegate::OnDownloadSaveDialogDone, |         base::BindOnce(&AtomDownloadManagerDelegate::OnDownloadSaveDialogDone, | ||||||
|                        base::Unretained(this), download_id, callback); |                        base::Unretained(this), download_id, callback); | ||||||
| 
 | 
 | ||||||
|     file_dialog::ShowSaveDialog(settings, std::move(dialog_promise)); |  | ||||||
|     ignore_result(dialog_promise.Then(std::move(dialog_callback))); |     ignore_result(dialog_promise.Then(std::move(dialog_callback))); | ||||||
|  |     file_dialog::ShowSaveDialog(settings, std::move(dialog_promise)); | ||||||
|   } else { |   } else { | ||||||
|     callback.Run(path, download::DownloadItem::TARGET_DISPOSITION_PROMPT, |     callback.Run(path, download::DownloadItem::TARGET_DISPOSITION_PROMPT, | ||||||
|                  download::DOWNLOAD_DANGER_TYPE_NOT_DANGEROUS, path, |                  download::DOWNLOAD_DANGER_TYPE_NOT_DANGEROUS, path, | ||||||
|  | @ -136,54 +136,48 @@ void AtomDownloadManagerDelegate::OnDownloadPathGenerated( | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #if defined(MAS_BUILD) |  | ||||||
| void AtomDownloadManagerDelegate::OnDownloadSaveDialogDone( | void AtomDownloadManagerDelegate::OnDownloadSaveDialogDone( | ||||||
|     uint32_t download_id, |     uint32_t download_id, | ||||||
|     const content::DownloadTargetCallback& download_callback, |     const content::DownloadTargetCallback& download_callback, | ||||||
|     bool result, |     mate::Dictionary result) { | ||||||
|     const base::FilePath& path, |  | ||||||
|     const std::string& bookmark) |  | ||||||
| #else |  | ||||||
| void AtomDownloadManagerDelegate::OnDownloadSaveDialogDone( |  | ||||||
|     uint32_t download_id, |  | ||||||
|     const content::DownloadTargetCallback& download_callback, |  | ||||||
|     bool result, |  | ||||||
|     const base::FilePath& path) |  | ||||||
| #endif |  | ||||||
| { |  | ||||||
|   DCHECK_CURRENTLY_ON(content::BrowserThread::UI); |   DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | ||||||
| 
 | 
 | ||||||
|   auto* item = download_manager_->GetDownload(download_id); |   auto* item = download_manager_->GetDownload(download_id); | ||||||
|   if (!item) |   if (!item) | ||||||
|     return; |     return; | ||||||
| 
 | 
 | ||||||
|   if (result) { |   bool canceled = true; | ||||||
|     // Remember the last selected download directory.
 |   result.Get("canceled", &canceled); | ||||||
|     AtomBrowserContext* browser_context = static_cast<AtomBrowserContext*>( |  | ||||||
|         download_manager_->GetBrowserContext()); |  | ||||||
|     browser_context->prefs()->SetFilePath(prefs::kDownloadDefaultDirectory, |  | ||||||
|                                           path.DirName()); |  | ||||||
| 
 | 
 | ||||||
|     v8::Isolate* isolate = v8::Isolate::GetCurrent(); |   base::FilePath path; | ||||||
|     v8::Locker locker(isolate); | 
 | ||||||
|     v8::HandleScope handle_scope(isolate); |   if (!canceled) { | ||||||
|     api::DownloadItem* download_item = |     if (result.Get("filePath", &path)) { | ||||||
|         api::DownloadItem::FromWrappedClass(isolate, item); |       // Remember the last selected download directory.
 | ||||||
|     if (download_item) |       AtomBrowserContext* browser_context = static_cast<AtomBrowserContext*>( | ||||||
|       download_item->SetSavePath(path); |           download_manager_->GetBrowserContext()); | ||||||
|  |       browser_context->prefs()->SetFilePath(prefs::kDownloadDefaultDirectory, | ||||||
|  |                                             path.DirName()); | ||||||
|  | 
 | ||||||
|  |       v8::Isolate* isolate = v8::Isolate::GetCurrent(); | ||||||
|  |       v8::Locker locker(isolate); | ||||||
|  |       v8::HandleScope handle_scope(isolate); | ||||||
|  |       api::DownloadItem* download_item = | ||||||
|  |           api::DownloadItem::FromWrappedClass(isolate, item); | ||||||
|  |       if (download_item) | ||||||
|  |         download_item->SetSavePath(path); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // Running the DownloadTargetCallback with an empty FilePath signals that the
 |   // Running the DownloadTargetCallback with an empty FilePath signals that the
 | ||||||
|   // download should be cancelled. If user cancels the file save dialog, run
 |   // download should be cancelled. If user cancels the file save dialog, run
 | ||||||
|   // the callback with empty FilePath.
 |   // the callback with empty FilePath.
 | ||||||
|   const base::FilePath download_path = result ? path : base::FilePath(); |  | ||||||
|   const auto interrupt_reason = |   const auto interrupt_reason = | ||||||
|       download_path.empty() ? download::DOWNLOAD_INTERRUPT_REASON_USER_CANCELED |       path.empty() ? download::DOWNLOAD_INTERRUPT_REASON_USER_CANCELED | ||||||
|                             : download::DOWNLOAD_INTERRUPT_REASON_NONE; |                    : download::DOWNLOAD_INTERRUPT_REASON_NONE; | ||||||
|   download_callback.Run(download_path, |   download_callback.Run(path, download::DownloadItem::TARGET_DISPOSITION_PROMPT, | ||||||
|                         download::DownloadItem::TARGET_DISPOSITION_PROMPT, |                         download::DOWNLOAD_DANGER_TYPE_NOT_DANGEROUS, path, | ||||||
|                         download::DOWNLOAD_DANGER_TYPE_NOT_DANGEROUS, |                         interrupt_reason); | ||||||
|                         download_path, interrupt_reason); |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void AtomDownloadManagerDelegate::Shutdown() { | void AtomDownloadManagerDelegate::Shutdown() { | ||||||
|  |  | ||||||
|  | @ -45,20 +45,10 @@ class AtomDownloadManagerDelegate : public content::DownloadManagerDelegate { | ||||||
|                                const content::DownloadTargetCallback& callback, |                                const content::DownloadTargetCallback& callback, | ||||||
|                                const base::FilePath& default_path); |                                const base::FilePath& default_path); | ||||||
| 
 | 
 | ||||||
| #if defined(MAS_BUILD) |  | ||||||
|   void OnDownloadSaveDialogDone( |   void OnDownloadSaveDialogDone( | ||||||
|       uint32_t download_id, |       uint32_t download_id, | ||||||
|       const content::DownloadTargetCallback& download_callback, |       const content::DownloadTargetCallback& download_callback, | ||||||
|       bool result, |       mate::Dictionary result); | ||||||
|       const base::FilePath& path, |  | ||||||
|       const std::string& bookmark); |  | ||||||
| #else |  | ||||||
|   void OnDownloadSaveDialogDone( |  | ||||||
|       uint32_t download_id, |  | ||||||
|       const content::DownloadTargetCallback& download_callback, |  | ||||||
|       bool result, |  | ||||||
|       const base::FilePath& path); |  | ||||||
| #endif |  | ||||||
| 
 | 
 | ||||||
|   content::DownloadManager* download_manager_; |   content::DownloadManager* download_manager_; | ||||||
|   base::WeakPtrFactory<AtomDownloadManagerDelegate> weak_ptr_factory_; |   base::WeakPtrFactory<AtomDownloadManagerDelegate> weak_ptr_factory_; | ||||||
|  |  | ||||||
|  | @ -24,6 +24,7 @@ | ||||||
| #include "content/public/browser/render_view_host.h" | #include "content/public/browser/render_view_host.h" | ||||||
| #include "content/public/browser/web_contents.h" | #include "content/public/browser/web_contents.h" | ||||||
| #include "content/public/browser/web_contents_observer.h" | #include "content/public/browser/web_contents_observer.h" | ||||||
|  | #include "native_mate/dictionary.h" | ||||||
| #include "net/base/mime_util.h" | #include "net/base/mime_util.h" | ||||||
| #include "ui/shell_dialogs/selected_file_info.h" | #include "ui/shell_dialogs/selected_file_info.h" | ||||||
| 
 | 
 | ||||||
|  | @ -51,9 +52,10 @@ class FileSelectHelper : public base::RefCounted<FileSelectHelper>, | ||||||
|     v8::Isolate* isolate = v8::Isolate::GetCurrent(); |     v8::Isolate* isolate = v8::Isolate::GetCurrent(); | ||||||
|     atom::util::Promise promise(isolate); |     atom::util::Promise promise(isolate); | ||||||
| 
 | 
 | ||||||
|     file_dialog::ShowOpenDialog(settings, std::move(promise)); |  | ||||||
|     auto callback = base::BindOnce(&FileSelectHelper::OnOpenDialogDone, this); |     auto callback = base::BindOnce(&FileSelectHelper::OnOpenDialogDone, this); | ||||||
|     ignore_result(promise.Then(std::move(callback))); |     ignore_result(promise.Then(std::move(callback))); | ||||||
|  | 
 | ||||||
|  |     file_dialog::ShowOpenDialog(settings, std::move(promise)); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   void ShowSaveDialog(const file_dialog::DialogSettings& settings) { |   void ShowSaveDialog(const file_dialog::DialogSettings& settings) { | ||||||
|  | @ -74,45 +76,43 @@ class FileSelectHelper : public base::RefCounted<FileSelectHelper>, | ||||||
| 
 | 
 | ||||||
|   ~FileSelectHelper() override {} |   ~FileSelectHelper() override {} | ||||||
| 
 | 
 | ||||||
| #if defined(MAS_BUILD) |   void OnOpenDialogDone(mate::Dictionary result) { | ||||||
|   void OnOpenDialogDone(bool result, |  | ||||||
|                         const std::vector<base::FilePath>& paths, |  | ||||||
|                         const std::vector<std::string>& bookmarks) |  | ||||||
| #else |  | ||||||
|   void OnOpenDialogDone(bool result, const std::vector<base::FilePath>& paths) |  | ||||||
| #endif |  | ||||||
|   { |  | ||||||
|     std::vector<FileChooserFileInfoPtr> file_info; |     std::vector<FileChooserFileInfoPtr> file_info; | ||||||
|     if (result) { |     bool canceled = true; | ||||||
|       for (auto& path : paths) { |     result.Get("canceled", &canceled); | ||||||
|         file_info.push_back(FileChooserFileInfo::NewNativeFile( |  | ||||||
|             blink::mojom::NativeFileInfo::New( |  | ||||||
|                 path, path.BaseName().AsUTF16Unsafe()))); |  | ||||||
|       } |  | ||||||
| 
 | 
 | ||||||
|       if (render_frame_host_ && !paths.empty()) { |     if (!canceled) { | ||||||
|         auto* browser_context = static_cast<atom::AtomBrowserContext*>( |       std::vector<base::FilePath> paths; | ||||||
|             render_frame_host_->GetProcess()->GetBrowserContext()); |       if (result.Get("filePaths", &paths)) { | ||||||
|         browser_context->prefs()->SetFilePath(prefs::kSelectFileLastDirectory, |         for (auto& path : paths) { | ||||||
|                                               paths[0].DirName()); |           file_info.push_back(FileChooserFileInfo::NewNativeFile( | ||||||
|  |               blink::mojom::NativeFileInfo::New( | ||||||
|  |                   path, path.BaseName().AsUTF16Unsafe()))); | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         if (render_frame_host_ && !paths.empty()) { | ||||||
|  |           auto* browser_context = static_cast<atom::AtomBrowserContext*>( | ||||||
|  |               render_frame_host_->GetProcess()->GetBrowserContext()); | ||||||
|  |           browser_context->prefs()->SetFilePath(prefs::kSelectFileLastDirectory, | ||||||
|  |                                                 paths[0].DirName()); | ||||||
|  |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     OnFilesSelected(std::move(file_info)); |     OnFilesSelected(std::move(file_info)); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| #if defined(MAS_BUILD) |   void OnSaveDialogDone(mate::Dictionary result) { | ||||||
|   void OnSaveDialogDone(bool result, |  | ||||||
|                         const base::FilePath& path, |  | ||||||
|                         const std::string& bookmark) |  | ||||||
| #else |  | ||||||
|   void OnSaveDialogDone(bool result, const base::FilePath& path) |  | ||||||
| #endif |  | ||||||
|   { |  | ||||||
|     std::vector<FileChooserFileInfoPtr> file_info; |     std::vector<FileChooserFileInfoPtr> file_info; | ||||||
|     if (result) { |     bool canceled = true; | ||||||
|       file_info.push_back( |     result.Get("canceled", &canceled); | ||||||
|           FileChooserFileInfo::NewNativeFile(blink::mojom::NativeFileInfo::New( | 
 | ||||||
|               path, path.BaseName().AsUTF16Unsafe()))); |     if (!canceled) { | ||||||
|  |       base::FilePath path; | ||||||
|  |       if (result.Get("filePath", &path)) { | ||||||
|  |         file_info.push_back(FileChooserFileInfo::NewNativeFile( | ||||||
|  |             blink::mojom::NativeFileInfo::New( | ||||||
|  |                 path, path.BaseName().AsUTF16Unsafe()))); | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|     OnFilesSelected(std::move(file_info)); |     OnFilesSelected(std::move(file_info)); | ||||||
|   } |   } | ||||||
|  | @ -123,7 +123,6 @@ class FileSelectHelper : public base::RefCounted<FileSelectHelper>, | ||||||
|       listener_.reset(); |       listener_.reset(); | ||||||
|     } |     } | ||||||
|     render_frame_host_ = nullptr; |     render_frame_host_ = nullptr; | ||||||
|     Release(); |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // content::WebContentsObserver:
 |   // content::WebContentsObserver:
 | ||||||
|  |  | ||||||
|  | @ -104,9 +104,25 @@ class Promise { | ||||||
|     return GetInner()->Reject(GetContext(), v8::Undefined(isolate())); |     return GetInner()->Reject(GetContext(), v8::Undefined(isolate())); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   template <typename ReturnType, typename... ArgTypes> |   // Please note that using Then is effectively the same as calling .then
 | ||||||
|  |   // in javascript.  This means (a) it is not type safe and (b) please note
 | ||||||
|  |   // it is NOT type safe.
 | ||||||
|  |   // If the base::Callback you provide here is of type void(boolean) and you
 | ||||||
|  |   // resolve the promise with a string, Electron will compile successfully and
 | ||||||
|  |   // then that promise will be rejected as soon as you try to use it as the
 | ||||||
|  |   // mate converters doing work behind the scenes will throw an error for you.
 | ||||||
|  |   // This can be really hard to trace so until either
 | ||||||
|  |   //   * This helper becomes typesafe (by templating the class instead of each
 | ||||||
|  |   //   method)
 | ||||||
|  |   //   * or the world goes mad
 | ||||||
|  |   // Please try your hardest not to use this method
 | ||||||
|  |   // The world thanks you
 | ||||||
|  |   template <typename... ResolveType> | ||||||
|   v8::MaybeLocal<v8::Promise> Then( |   v8::MaybeLocal<v8::Promise> Then( | ||||||
|       base::OnceCallback<ReturnType(ArgTypes...)> cb) { |       base::OnceCallback<void(ResolveType...)> cb) { | ||||||
|  |     static_assert(sizeof...(ResolveType) <= 1, | ||||||
|  |                   "A promise's 'Then' callback should only receive at most one " | ||||||
|  |                   "parameter"); | ||||||
|     v8::HandleScope handle_scope(isolate()); |     v8::HandleScope handle_scope(isolate()); | ||||||
|     v8::Context::Scope context_scope( |     v8::Context::Scope context_scope( | ||||||
|         v8::Local<v8::Context>::New(isolate(), GetContext())); |         v8::Local<v8::Context>::New(isolate(), GetContext())); | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Samuel Attard
				Samuel Attard