Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(78)

Side by Side Diff: chrome/browser/gtk/dialogs_gtk.cc

Issue 6251001: Move chrome/browser/gtk/ to chrome/browser/ui/gtk/... (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src/
Patch Set: '' Created 9 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « chrome/browser/gtk/custom_drag.cc ('k') | chrome/browser/gtk/download_in_progress_dialog_gtk.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(Empty)
1 // Copyright (c) 2010 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 <gtk/gtk.h>
6 #include <map>
7 #include <set>
8
9 #include "app/gtk_signal.h"
10 #include "app/l10n_util.h"
11 #include "base/file_util.h"
12 #include "base/logging.h"
13 #include "base/message_loop.h"
14 #include "base/mime_util.h"
15 #include "base/sys_string_conversions.h"
16 #include "base/threading/thread.h"
17 #include "base/threading/thread_restrictions.h"
18 #include "base/utf_string_conversions.h"
19 #include "chrome/browser/browser_thread.h"
20 #include "chrome/browser/shell_dialogs.h"
21 #include "grit/generated_resources.h"
22
23 // The size of the preview we display for selected image files. We set height
24 // larger than width because generally there is more free space vertically
25 // than horiztonally (setting the preview image will alway expand the width of
26 // the dialog, but usually not the height). The image's aspect ratio will always
27 // be preserved.
28 static const int kPreviewWidth = 256;
29 static const int kPreviewHeight = 512;
30
31 // Implementation of SelectFileDialog that shows a Gtk common dialog for
32 // choosing a file or folder. This acts as a modal dialog.
33 class SelectFileDialogImpl : public SelectFileDialog {
34 public:
35 explicit SelectFileDialogImpl(Listener* listener);
36
37 // BaseShellDialog implementation.
38 virtual bool IsRunning(gfx::NativeWindow parent_window) const;
39 virtual void ListenerDestroyed();
40
41 // SelectFileDialog implementation.
42 // |params| is user data we pass back via the Listener interface.
43 virtual void SelectFile(Type type,
44 const string16& title,
45 const FilePath& default_path,
46 const FileTypeInfo* file_types,
47 int file_type_index,
48 const FilePath::StringType& default_extension,
49 gfx::NativeWindow owning_window,
50 void* params);
51
52 private:
53 virtual ~SelectFileDialogImpl();
54
55 // Add the filters from |file_types_| to |chooser|.
56 void AddFilters(GtkFileChooser* chooser);
57
58 // Notifies the listener that a single file was chosen.
59 void FileSelected(GtkWidget* dialog, const FilePath& path);
60
61 // Notifies the listener that multiple files were chosen.
62 void MultiFilesSelected(GtkWidget* dialog,
63 const std::vector<FilePath>& files);
64
65 // Notifies the listener that no file was chosen (the action was canceled).
66 // Dialog is passed so we can find that |params| pointer that was passed to
67 // us when we were told to show the dialog.
68 void FileNotSelected(GtkWidget* dialog);
69
70 GtkWidget* CreateSelectFolderDialog(const std::string& title,
71 const FilePath& default_path, gfx::NativeWindow parent);
72
73 GtkWidget* CreateFileOpenDialog(const std::string& title,
74 const FilePath& default_path, gfx::NativeWindow parent);
75
76 GtkWidget* CreateMultiFileOpenDialog(const std::string& title,
77 const FilePath& default_path, gfx::NativeWindow parent);
78
79 GtkWidget* CreateSaveAsDialog(const std::string& title,
80 const FilePath& default_path, gfx::NativeWindow parent);
81
82 // Removes and returns the |params| associated with |dialog| from
83 // |params_map_|.
84 void* PopParamsForDialog(GtkWidget* dialog);
85
86 // Take care of internal data structures when a file dialog is destroyed.
87 void FileDialogDestroyed(GtkWidget* dialog);
88
89 // Check whether response_id corresponds to the user cancelling/closing the
90 // dialog. Used as a helper for the below callbacks.
91 bool IsCancelResponse(gint response_id);
92
93 // Common function for OnSelectSingleFileDialogResponse and
94 // OnSelectSingleFolderDialogResponse.
95 void SelectSingleFileHelper(GtkWidget* dialog,
96 gint response_id,
97 bool allow_folder);
98
99 // Common function for CreateFileOpenDialog and CreateMultiFileOpenDialog.
100 GtkWidget* CreateFileOpenHelper(const std::string& title,
101 const FilePath& default_path,
102 gfx::NativeWindow parent);
103
104 // Wrapper for file_util::DirectoryExists() that allow access on the UI
105 // thread. Use this only in the file dialog functions, where it's ok
106 // because the file dialog has to do many stats anyway. One more won't
107 // hurt too badly and it's likely already cached.
108 bool CallDirectoryExistsOnUIThread(const FilePath& path);
109
110 // Callback for when the user responds to a Save As or Open File dialog.
111 CHROMEGTK_CALLBACK_1(SelectFileDialogImpl, void,
112 OnSelectSingleFileDialogResponse, gint);
113
114 // Callback for when the user responds to a Select Folder dialog.
115 CHROMEGTK_CALLBACK_1(SelectFileDialogImpl, void,
116 OnSelectSingleFolderDialogResponse, gint);
117
118 // Callback for when the user responds to a Open Multiple Files dialog.
119 CHROMEGTK_CALLBACK_1(SelectFileDialogImpl, void,
120 OnSelectMultiFileDialogResponse, gint);
121
122 // Callback for when the file chooser gets destroyed.
123 CHROMEGTK_CALLBACK_0(SelectFileDialogImpl, void, OnFileChooserDestroy);
124
125 // Callback for when we update the preview for the selection.
126 CHROMEGTK_CALLBACK_0(SelectFileDialogImpl, void, OnUpdatePreview);
127
128 // The listener to be notified of selection completion.
129 Listener* listener_;
130
131 // A map from dialog windows to the |params| user data associated with them.
132 std::map<GtkWidget*, void*> params_map_;
133
134 // The file filters.
135 FileTypeInfo file_types_;
136
137 // The index of the default selected file filter.
138 // Note: This starts from 1, not 0.
139 size_t file_type_index_;
140
141 // The set of all parent windows for which we are currently running dialogs.
142 std::set<GtkWindow*> parents_;
143
144 // The type of dialog we are showing the user.
145 Type type_;
146
147 // These two variables track where the user last saved a file or opened a
148 // file so that we can display future dialogs with the same starting path.
149 static FilePath* last_saved_path_;
150 static FilePath* last_opened_path_;
151
152 // The GtkImage widget for showing previews of selected images.
153 GtkWidget* preview_;
154
155 // All our dialogs.
156 std::set<GtkWidget*> dialogs_;
157
158 DISALLOW_COPY_AND_ASSIGN(SelectFileDialogImpl);
159 };
160
161 FilePath* SelectFileDialogImpl::last_saved_path_ = NULL;
162 FilePath* SelectFileDialogImpl::last_opened_path_ = NULL;
163
164 // static
165 SelectFileDialog* SelectFileDialog::Create(Listener* listener) {
166 DCHECK(!BrowserThread::CurrentlyOn(BrowserThread::IO));
167 DCHECK(!BrowserThread::CurrentlyOn(BrowserThread::FILE));
168 return new SelectFileDialogImpl(listener);
169 }
170
171 SelectFileDialogImpl::SelectFileDialogImpl(Listener* listener)
172 : listener_(listener),
173 file_type_index_(0),
174 type_(SELECT_NONE),
175 preview_(NULL) {
176 if (!last_saved_path_) {
177 last_saved_path_ = new FilePath();
178 last_opened_path_ = new FilePath();
179 }
180 }
181
182 SelectFileDialogImpl::~SelectFileDialogImpl() {
183 while (dialogs_.begin() != dialogs_.end()) {
184 gtk_widget_destroy(*(dialogs_.begin()));
185 }
186 }
187
188 bool SelectFileDialogImpl::IsRunning(gfx::NativeWindow parent_window) const {
189 return parents_.find(parent_window) != parents_.end();
190 }
191
192 void SelectFileDialogImpl::ListenerDestroyed() {
193 listener_ = NULL;
194 }
195
196 // We ignore |default_extension|.
197 void SelectFileDialogImpl::SelectFile(
198 Type type,
199 const string16& title,
200 const FilePath& default_path,
201 const FileTypeInfo* file_types,
202 int file_type_index,
203 const FilePath::StringType& default_extension,
204 gfx::NativeWindow owning_window,
205 void* params) {
206 type_ = type;
207 // |owning_window| can be null when user right-clicks on a downloadable item
208 // and chooses 'Open Link in New Tab' when 'Ask where to save each file
209 // before downloading.' preference is turned on. (http://crbug.com/29213)
210 if (owning_window)
211 parents_.insert(owning_window);
212
213 std::string title_string = UTF16ToUTF8(title);
214
215 file_type_index_ = file_type_index;
216 if (file_types)
217 file_types_ = *file_types;
218 else
219 file_types_.include_all_files = true;
220
221 GtkWidget* dialog = NULL;
222 switch (type) {
223 case SELECT_FOLDER:
224 dialog = CreateSelectFolderDialog(title_string, default_path,
225 owning_window);
226 break;
227 case SELECT_OPEN_FILE:
228 dialog = CreateFileOpenDialog(title_string, default_path, owning_window);
229 break;
230 case SELECT_OPEN_MULTI_FILE:
231 dialog = CreateMultiFileOpenDialog(title_string, default_path,
232 owning_window);
233 break;
234 case SELECT_SAVEAS_FILE:
235 dialog = CreateSaveAsDialog(title_string, default_path, owning_window);
236 break;
237 default:
238 NOTREACHED();
239 return;
240 }
241 dialogs_.insert(dialog);
242
243 preview_ = gtk_image_new();
244 g_signal_connect(dialog, "destroy",
245 G_CALLBACK(OnFileChooserDestroyThunk), this);
246 g_signal_connect(dialog, "update-preview",
247 G_CALLBACK(OnUpdatePreviewThunk), this);
248 gtk_file_chooser_set_preview_widget(GTK_FILE_CHOOSER(dialog), preview_);
249
250 params_map_[dialog] = params;
251
252 // Set window-to-parent modality by adding the dialog to the same window
253 // group as the parent.
254 gtk_window_group_add_window(gtk_window_get_group(owning_window),
255 GTK_WINDOW(dialog));
256 gtk_window_set_modal(GTK_WINDOW(dialog), TRUE);
257
258 gtk_widget_show_all(dialog);
259 }
260
261 void SelectFileDialogImpl::AddFilters(GtkFileChooser* chooser) {
262 for (size_t i = 0; i < file_types_.extensions.size(); ++i) {
263 GtkFileFilter* filter = NULL;
264 for (size_t j = 0; j < file_types_.extensions[i].size(); ++j) {
265 if (!file_types_.extensions[i][j].empty()) {
266 if (!filter)
267 filter = gtk_file_filter_new();
268 std::string mime_type = mime_util::GetFileMimeType(
269 FilePath("name.").Append(file_types_.extensions[i][j]));
270 gtk_file_filter_add_mime_type(filter, mime_type.c_str());
271 }
272 }
273 // We didn't find any non-empty extensions to filter on.
274 if (!filter)
275 continue;
276
277 // The description vector may be blank, in which case we are supposed to
278 // use some sort of default description based on the filter.
279 if (i < file_types_.extension_description_overrides.size()) {
280 gtk_file_filter_set_name(filter, UTF16ToUTF8(
281 file_types_.extension_description_overrides[i]).c_str());
282 } else {
283 // There is no system default filter description so we use
284 // the MIME type itself if the description is blank.
285 std::string mime_type = mime_util::GetFileMimeType(
286 FilePath("name.").Append(file_types_.extensions[i][0]));
287 gtk_file_filter_set_name(filter, mime_type.c_str());
288 }
289
290 gtk_file_chooser_add_filter(chooser, filter);
291 if (i == file_type_index_ - 1)
292 gtk_file_chooser_set_filter(chooser, filter);
293 }
294
295 // Add the *.* filter, but only if we have added other filters (otherwise it
296 // is implied).
297 if (file_types_.include_all_files && file_types_.extensions.size() > 0) {
298 GtkFileFilter* filter = gtk_file_filter_new();
299 gtk_file_filter_add_pattern(filter, "*");
300 gtk_file_filter_set_name(filter,
301 l10n_util::GetStringUTF8(IDS_SAVEAS_ALL_FILES).c_str());
302 gtk_file_chooser_add_filter(chooser, filter);
303 }
304 }
305
306 void SelectFileDialogImpl::FileSelected(GtkWidget* dialog,
307 const FilePath& path) {
308 if (type_ == SELECT_SAVEAS_FILE)
309 *last_saved_path_ = path.DirName();
310 else if (type_ == SELECT_OPEN_FILE)
311 *last_opened_path_ = path.DirName();
312 else if (type_ == SELECT_FOLDER)
313 *last_opened_path_ = path.DirName().DirName();
314 else
315 NOTREACHED();
316
317 if (listener_) {
318 GtkFileFilter* selected_filter =
319 gtk_file_chooser_get_filter(GTK_FILE_CHOOSER(dialog));
320 GSList* filters = gtk_file_chooser_list_filters(GTK_FILE_CHOOSER(dialog));
321 int idx = g_slist_index(filters, selected_filter);
322 g_slist_free(filters);
323 listener_->FileSelected(path, idx + 1, PopParamsForDialog(dialog));
324 }
325 gtk_widget_destroy(dialog);
326 }
327
328 void SelectFileDialogImpl::MultiFilesSelected(GtkWidget* dialog,
329 const std::vector<FilePath>& files) {
330 *last_opened_path_ = files[0].DirName();
331
332 if (listener_)
333 listener_->MultiFilesSelected(files, PopParamsForDialog(dialog));
334 gtk_widget_destroy(dialog);
335 }
336
337 void SelectFileDialogImpl::FileNotSelected(GtkWidget* dialog) {
338 void* params = PopParamsForDialog(dialog);
339 if (listener_)
340 listener_->FileSelectionCanceled(params);
341 gtk_widget_destroy(dialog);
342 }
343
344 bool SelectFileDialogImpl::CallDirectoryExistsOnUIThread(const FilePath& path) {
345 base::ThreadRestrictions::ScopedAllowIO allow_io;
346 return file_util::DirectoryExists(path);
347 }
348
349 GtkWidget* SelectFileDialogImpl::CreateFileOpenHelper(
350 const std::string& title,
351 const FilePath& default_path,
352 gfx::NativeWindow parent) {
353 GtkWidget* dialog =
354 gtk_file_chooser_dialog_new(title.c_str(), parent,
355 GTK_FILE_CHOOSER_ACTION_OPEN,
356 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
357 GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT,
358 NULL);
359 AddFilters(GTK_FILE_CHOOSER(dialog));
360
361 if (!default_path.empty()) {
362 if (CallDirectoryExistsOnUIThread(default_path)) {
363 gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(dialog),
364 default_path.value().c_str());
365 } else {
366 // If the file doesn't exist, this will just switch to the correct
367 // directory. That's good enough.
368 gtk_file_chooser_set_filename(GTK_FILE_CHOOSER(dialog),
369 default_path.value().c_str());
370 }
371 } else if (!last_opened_path_->empty()) {
372 gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(dialog),
373 last_opened_path_->value().c_str());
374 }
375 return dialog;
376 }
377
378 GtkWidget* SelectFileDialogImpl::CreateSelectFolderDialog(
379 const std::string& title,
380 const FilePath& default_path,
381 gfx::NativeWindow parent) {
382 std::string title_string = !title.empty() ? title :
383 l10n_util::GetStringUTF8(IDS_SELECT_FOLDER_DIALOG_TITLE);
384
385 GtkWidget* dialog =
386 gtk_file_chooser_dialog_new(title_string.c_str(), parent,
387 GTK_FILE_CHOOSER_ACTION_SELECT_FOLDER,
388 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
389 GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT,
390 NULL);
391
392 if (!default_path.empty()) {
393 gtk_file_chooser_set_filename(GTK_FILE_CHOOSER(dialog),
394 default_path.value().c_str());
395 } else if (!last_opened_path_->empty()) {
396 gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(dialog),
397 last_opened_path_->value().c_str());
398 }
399 gtk_file_chooser_set_select_multiple(GTK_FILE_CHOOSER(dialog), FALSE);
400 g_signal_connect(dialog, "response",
401 G_CALLBACK(OnSelectSingleFolderDialogResponseThunk), this);
402 return dialog;
403 }
404
405 GtkWidget* SelectFileDialogImpl::CreateFileOpenDialog(
406 const std::string& title,
407 const FilePath& default_path,
408 gfx::NativeWindow parent) {
409 std::string title_string = !title.empty() ? title :
410 l10n_util::GetStringUTF8(IDS_OPEN_FILE_DIALOG_TITLE);
411 GtkWidget* dialog = CreateFileOpenHelper(title_string, default_path, parent);
412 gtk_file_chooser_set_select_multiple(GTK_FILE_CHOOSER(dialog), FALSE);
413 g_signal_connect(dialog, "response",
414 G_CALLBACK(OnSelectSingleFileDialogResponseThunk), this);
415 return dialog;
416 }
417
418 GtkWidget* SelectFileDialogImpl::CreateMultiFileOpenDialog(
419 const std::string& title,
420 const FilePath& default_path,
421 gfx::NativeWindow parent) {
422 std::string title_string = !title.empty() ? title :
423 l10n_util::GetStringUTF8(IDS_OPEN_FILES_DIALOG_TITLE);
424 GtkWidget* dialog = CreateFileOpenHelper(title_string, default_path, parent);
425 gtk_file_chooser_set_select_multiple(GTK_FILE_CHOOSER(dialog), TRUE);
426 g_signal_connect(dialog, "response",
427 G_CALLBACK(OnSelectMultiFileDialogResponseThunk), this);
428 return dialog;
429 }
430
431 GtkWidget* SelectFileDialogImpl::CreateSaveAsDialog(const std::string& title,
432 const FilePath& default_path, gfx::NativeWindow parent) {
433 std::string title_string = !title.empty() ? title :
434 l10n_util::GetStringUTF8(IDS_SAVE_AS_DIALOG_TITLE);
435
436 GtkWidget* dialog =
437 gtk_file_chooser_dialog_new(title_string.c_str(), parent,
438 GTK_FILE_CHOOSER_ACTION_SAVE,
439 GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
440 GTK_STOCK_SAVE, GTK_RESPONSE_ACCEPT,
441 NULL);
442
443 AddFilters(GTK_FILE_CHOOSER(dialog));
444 if (!default_path.empty()) {
445 // Since the file may not already exist, we use
446 // set_current_folder() followed by set_current_name(), as per the
447 // recommendation of the GTK docs.
448 gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(dialog),
449 default_path.DirName().value().c_str());
450 gtk_file_chooser_set_current_name(GTK_FILE_CHOOSER(dialog),
451 default_path.BaseName().value().c_str());
452 } else if (!last_saved_path_->empty()) {
453 gtk_file_chooser_set_current_folder(GTK_FILE_CHOOSER(dialog),
454 last_saved_path_->value().c_str());
455 }
456 gtk_file_chooser_set_select_multiple(GTK_FILE_CHOOSER(dialog), FALSE);
457 gtk_file_chooser_set_do_overwrite_confirmation(GTK_FILE_CHOOSER(dialog),
458 TRUE);
459 g_signal_connect(dialog, "response",
460 G_CALLBACK(OnSelectSingleFileDialogResponseThunk), this);
461 return dialog;
462 }
463
464 void* SelectFileDialogImpl::PopParamsForDialog(GtkWidget* dialog) {
465 std::map<GtkWidget*, void*>::iterator iter = params_map_.find(dialog);
466 DCHECK(iter != params_map_.end());
467 void* params = iter->second;
468 params_map_.erase(iter);
469 return params;
470 }
471
472 void SelectFileDialogImpl::FileDialogDestroyed(GtkWidget* dialog) {
473 dialogs_.erase(dialog);
474
475 // Parent may be NULL in a few cases: 1) on shutdown when
476 // AllBrowsersClosed() trigger this handler after all the browser
477 // windows got destroyed, or 2) when the parent tab has been opened by
478 // 'Open Link in New Tab' context menu on a downloadable item and
479 // the tab has no content (see the comment in SelectFile as well).
480 GtkWindow* parent = gtk_window_get_transient_for(GTK_WINDOW(dialog));
481 if (!parent)
482 return;
483 std::set<GtkWindow*>::iterator iter = parents_.find(parent);
484 if (iter != parents_.end())
485 parents_.erase(iter);
486 else
487 NOTREACHED();
488 }
489
490 bool SelectFileDialogImpl::IsCancelResponse(gint response_id) {
491 bool is_cancel = response_id == GTK_RESPONSE_CANCEL ||
492 response_id == GTK_RESPONSE_DELETE_EVENT;
493 if (is_cancel)
494 return true;
495
496 DCHECK(response_id == GTK_RESPONSE_ACCEPT);
497 return false;
498 }
499
500 void SelectFileDialogImpl::SelectSingleFileHelper(GtkWidget* dialog,
501 gint response_id,
502 bool allow_folder) {
503 if (IsCancelResponse(response_id)) {
504 FileNotSelected(dialog);
505 return;
506 }
507
508 gchar* filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(dialog));
509 if (!filename) {
510 FileNotSelected(dialog);
511 return;
512 }
513
514 FilePath path(filename);
515 g_free(filename);
516
517 if (allow_folder) {
518 FileSelected(dialog, path);
519 return;
520 }
521
522 if (CallDirectoryExistsOnUIThread(path))
523 FileNotSelected(dialog);
524 else
525 FileSelected(dialog, path);
526 }
527
528 void SelectFileDialogImpl::OnSelectSingleFileDialogResponse(
529 GtkWidget* dialog, gint response_id) {
530 return SelectSingleFileHelper(dialog, response_id, false);
531 }
532
533 void SelectFileDialogImpl::OnSelectSingleFolderDialogResponse(
534 GtkWidget* dialog, gint response_id) {
535 return SelectSingleFileHelper(dialog, response_id, true);
536 }
537
538 void SelectFileDialogImpl::OnSelectMultiFileDialogResponse(
539 GtkWidget* dialog, gint response_id) {
540 if (IsCancelResponse(response_id)) {
541 FileNotSelected(dialog);
542 return;
543 }
544
545 GSList* filenames = gtk_file_chooser_get_filenames(GTK_FILE_CHOOSER(dialog));
546 if (!filenames) {
547 FileNotSelected(dialog);
548 return;
549 }
550
551 std::vector<FilePath> filenames_fp;
552 for (GSList* iter = filenames; iter != NULL; iter = g_slist_next(iter)) {
553 FilePath path(static_cast<char*>(iter->data));
554 g_free(iter->data);
555 if (CallDirectoryExistsOnUIThread(path))
556 continue;
557 filenames_fp.push_back(path);
558 }
559 g_slist_free(filenames);
560
561 if (filenames_fp.empty()) {
562 FileNotSelected(dialog);
563 return;
564 }
565 MultiFilesSelected(dialog, filenames_fp);
566 }
567
568 void SelectFileDialogImpl::OnFileChooserDestroy(GtkWidget* dialog) {
569 FileDialogDestroyed(dialog);
570 }
571
572 void SelectFileDialogImpl::OnUpdatePreview(GtkWidget* chooser) {
573 gchar* filename = gtk_file_chooser_get_preview_filename(
574 GTK_FILE_CHOOSER(chooser));
575 if (!filename)
576 return;
577 // This will preserve the image's aspect ratio.
578 GdkPixbuf* pixbuf = gdk_pixbuf_new_from_file_at_size(filename, kPreviewWidth,
579 kPreviewHeight, NULL);
580 g_free(filename);
581 if (pixbuf) {
582 gtk_image_set_from_pixbuf(GTK_IMAGE(preview_), pixbuf);
583 g_object_unref(pixbuf);
584 }
585 gtk_file_chooser_set_preview_widget_active(GTK_FILE_CHOOSER(chooser),
586 pixbuf ? TRUE : FALSE);
587 }
OLDNEW
« no previous file with comments | « chrome/browser/gtk/custom_drag.cc ('k') | chrome/browser/gtk/download_in_progress_dialog_gtk.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698