// Copyright (c) 2010 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "chrome/browser/gtk/options/content_page_gtk.h" #include #include "app/gtk_util.h" #include "app/l10n_util.h" #include "app/resource_bundle.h" #include "base/command_line.h" #include "chrome/browser/browser.h" #include "chrome/browser/browser_list.h" #include "chrome/browser/browser_window.h" #include "chrome/browser/defaults.h" #include "chrome/browser/gtk/gtk_chrome_link_button.h" #include "chrome/browser/gtk/gtk_theme_provider.h" #include "chrome/browser/gtk/gtk_util.h" #include "chrome/browser/gtk/import_dialog_gtk.h" #include "chrome/browser/gtk/options/customize_sync_window_gtk.h" #include "chrome/browser/gtk/options/options_layout_gtk.h" #include "chrome/browser/gtk/options/passwords_exceptions_window_gtk.h" #include "chrome/browser/importer/importer_data_types.h" #include "chrome/browser/pref_service.h" #include "chrome/browser/sync/sync_ui_util.h" #include "chrome/common/chrome_switches.h" #include "chrome/common/notification_service.h" #include "chrome/common/pref_names.h" #include "chrome/common/url_constants.h" #include "gfx/gtk_util.h" #include "grit/app_resources.h" #include "grit/chromium_strings.h" #include "grit/generated_resources.h" #include "grit/locale_settings.h" #if defined(OS_CHROMEOS) #include "chrome/browser/chromeos/options/options_window_view.h" #endif // defined(OS_CHROMEOS) namespace { // Background color for the status label when it's showing an error. static const GdkColor kSyncLabelErrorBgColor = GDK_COLOR_RGB(0xff, 0x9a, 0x9a); // Helper for WrapLabelAtAllocationHack. void OnLabelAllocate(GtkWidget* label, GtkAllocation* allocation) { gtk_widget_set_size_request(label, allocation->width, -1); // Disconnect ourselves. Repeatedly resizing based on allocation causes // the dialog to become unshrinkable. g_signal_handlers_disconnect_by_func( label, reinterpret_cast(OnLabelAllocate), NULL); } // Set the label to use a request size equal to its initial allocation // size. This causes the label to wrap at the width of the container // it is in, instead of at the default width. This is called a hack // because GTK doesn't really work when a widget to make its size // request depend on its allocation. It does, however, have the // intended effect of wrapping the label at the proper width. void WrapLabelAtAllocationHack(GtkWidget* label) { g_signal_connect(label, "size-allocate", G_CALLBACK(OnLabelAllocate), NULL); } } // anonymous namespace /////////////////////////////////////////////////////////////////////////////// // ContentPageGtk, public: ContentPageGtk::ContentPageGtk(Profile* profile) : OptionsPageBase(profile), sync_status_label_background_(NULL), sync_status_label_(NULL), #if !defined(OS_CHROMEOS) sync_action_link_background_(NULL), sync_action_link_(NULL), sync_start_stop_button_(NULL), #endif sync_customize_button_(NULL), initializing_(true), sync_service_(NULL) { if (profile->GetProfileSyncService()) { sync_service_ = profile->GetProfileSyncService(); sync_service_->AddObserver(this); } // Prepare the group options layout. scoped_ptr options_builder(OptionsLayoutBuilderGtk::CreateOptionallyCompactLayout()); if (sync_service_) { options_builder->AddOptionGroup( l10n_util::GetStringUTF8(IDS_SYNC_OPTIONS_GROUP_NAME), InitSyncGroup(), false); UpdateSyncControls(); } options_builder->AddOptionGroup( l10n_util::GetStringUTF8(IDS_OPTIONS_PASSWORDS_GROUP_NAME), InitPasswordSavingGroup(), false); options_builder->AddOptionGroup( l10n_util::GetStringUTF8(IDS_AUTOFILL_SETTING_WINDOWS_GROUP_NAME), InitFormAutoFillGroup(), false); #if !defined(OS_CHROMEOS) options_builder->AddOptionGroup( l10n_util::GetStringUTF8(IDS_OPTIONS_BROWSING_DATA_GROUP_NAME), InitBrowsingDataGroup(), false); #endif options_builder->AddOptionGroup( l10n_util::GetStringUTF8(IDS_APPEARANCE_GROUP_NAME), InitThemesGroup(), false); page_ = options_builder->get_page_widget(); // Add preferences observers. ask_to_save_passwords_.Init(prefs::kPasswordManagerEnabled, profile->GetPrefs(), this); enable_form_autofill_.Init(prefs::kAutoFillEnabled, profile->GetPrefs(), this); if (browser_defaults::kCanToggleSystemTitleBar) { use_custom_chrome_frame_.Init(prefs::kUseCustomChromeFrame, profile->GetPrefs(), this); } // Load initial values. NotifyPrefChanged(NULL); registrar_.Add(this, NotificationType::BROWSER_THEME_CHANGED, NotificationService::AllSources()); ObserveThemeChanged(); } ContentPageGtk::~ContentPageGtk() { if (sync_service_) sync_service_->RemoveObserver(this); } /////////////////////////////////////////////////////////////////////////////// // ContentsPageView, ProfileSyncServiceObserver implementation: void ContentPageGtk::OnStateChanged() { // If the UI controls are not yet initialized, then don't do anything. This // can happen if the Options dialog is up, but the Content tab is not yet // clicked. if (!initializing_) UpdateSyncControls(); } /////////////////////////////////////////////////////////////////////////////// // ContentPageGtk, private: // If |pref_name| is NULL, set the state of all the widgets. (This is used // in ContentPageGtk() above to initialize the dialog.) Otherwise, reset the // state of the widget for the given preference name, as it has changed. void ContentPageGtk::NotifyPrefChanged(const std::wstring* pref_name) { initializing_ = true; if (!pref_name || *pref_name == prefs::kPasswordManagerEnabled) { if (ask_to_save_passwords_.GetValue()) { gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(passwords_asktosave_radio_), TRUE); } else { gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(passwords_neversave_radio_), TRUE); } } if (!pref_name || *pref_name == prefs::kAutoFillEnabled) { if (enable_form_autofill_.GetValue()) { gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(form_autofill_enable_radio_), TRUE); gtk_widget_set_sensitive(autofill_button_, TRUE); } else { gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(form_autofill_disable_radio_), TRUE); gtk_widget_set_sensitive(autofill_button_, FALSE); } } if (browser_defaults::kCanToggleSystemTitleBar && (!pref_name || *pref_name == prefs::kUseCustomChromeFrame)) { if (use_custom_chrome_frame_.GetValue()) { gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(system_title_bar_hide_radio_), TRUE); } else { gtk_toggle_button_set_active( GTK_TOGGLE_BUTTON(system_title_bar_show_radio_), TRUE); } } initializing_ = false; } void ContentPageGtk::Observe(NotificationType type, const NotificationSource& source, const NotificationDetails& details) { if (type == NotificationType::BROWSER_THEME_CHANGED) ObserveThemeChanged(); else OptionsPageBase::Observe(type, source, details); } void ContentPageGtk::ObserveThemeChanged() { #if defined(TOOLKIT_GTK) GtkThemeProvider* provider = GtkThemeProvider::GetFrom(profile()); bool is_gtk_theme = provider->UseGtkTheme(); gtk_widget_set_sensitive(gtk_theme_button_, !is_gtk_theme); #else BrowserThemeProvider* provider = reinterpret_cast(profile()->GetThemeProvider()); bool is_gtk_theme = false; #endif bool is_classic_theme = !is_gtk_theme && provider->GetThemeID().empty(); gtk_widget_set_sensitive(themes_reset_button_, !is_classic_theme); } GtkWidget* ContentPageGtk::InitPasswordSavingGroup() { GtkWidget* vbox = gtk_vbox_new(FALSE, gtk_util::kControlSpacing); // Ask to save radio button. passwords_asktosave_radio_ = gtk_radio_button_new_with_label(NULL, l10n_util::GetStringUTF8(IDS_OPTIONS_PASSWORDS_ASKTOSAVE).c_str()); g_signal_connect(passwords_asktosave_radio_, "toggled", G_CALLBACK(OnPasswordRadioToggledThunk), this); gtk_box_pack_start(GTK_BOX(vbox), passwords_asktosave_radio_, FALSE, FALSE, 0); // Never save radio button. passwords_neversave_radio_ = gtk_radio_button_new_with_label_from_widget( GTK_RADIO_BUTTON(passwords_asktosave_radio_), l10n_util::GetStringUTF8(IDS_OPTIONS_PASSWORDS_NEVERSAVE).c_str()); g_signal_connect(passwords_neversave_radio_, "toggled", G_CALLBACK(OnPasswordRadioToggledThunk), this); gtk_box_pack_start(GTK_BOX(vbox), passwords_neversave_radio_, FALSE, FALSE, 0); // Add the show passwords button into its own horizontal box so it does not // depend on the spacing above. GtkWidget* button_hbox = gtk_hbox_new(FALSE, gtk_util::kLabelSpacing); gtk_container_add(GTK_CONTAINER(vbox), button_hbox); GtkWidget* show_passwords_button = gtk_button_new_with_label( l10n_util::GetStringUTF8(IDS_OPTIONS_PASSWORDS_SHOWPASSWORDS).c_str()); g_signal_connect(show_passwords_button, "clicked", G_CALLBACK(OnShowPasswordsButtonClickedThunk), this); gtk_box_pack_start(GTK_BOX(button_hbox), show_passwords_button, FALSE, FALSE, 0); return vbox; } GtkWidget* ContentPageGtk::InitFormAutoFillGroup() { GtkWidget* vbox = gtk_vbox_new(FALSE, gtk_util::kControlSpacing); // Enable radio button. form_autofill_enable_radio_ = gtk_radio_button_new_with_label(NULL, l10n_util::GetStringUTF8(IDS_OPTIONS_AUTOFILL_ENABLE).c_str()); g_signal_connect(G_OBJECT(form_autofill_enable_radio_), "toggled", G_CALLBACK(OnAutoFillRadioToggledThunk), this); gtk_box_pack_start(GTK_BOX(vbox), form_autofill_enable_radio_, FALSE, FALSE, 0); // Disable radio button. form_autofill_disable_radio_ = gtk_radio_button_new_with_label_from_widget( GTK_RADIO_BUTTON(form_autofill_enable_radio_), l10n_util::GetStringUTF8(IDS_OPTIONS_AUTOFILL_DISABLE).c_str()); g_signal_connect(G_OBJECT(form_autofill_disable_radio_), "toggled", G_CALLBACK(OnAutoFillRadioToggledThunk), this); gtk_box_pack_start(GTK_BOX(vbox), form_autofill_disable_radio_, FALSE, FALSE, 0); GtkWidget* button_hbox = gtk_hbox_new(FALSE, gtk_util::kControlSpacing); gtk_container_add(GTK_CONTAINER(vbox), button_hbox); // AutoFill button. autofill_button_ = gtk_button_new_with_label( l10n_util::GetStringUTF8(IDS_AUTOFILL_OPTIONS).c_str()); if (!profile()->GetPrefs()->GetBoolean(prefs::kAutoFillEnabled)) gtk_widget_set_sensitive(autofill_button_, FALSE); g_signal_connect(G_OBJECT(autofill_button_), "clicked", G_CALLBACK(OnAutoFillButtonClickedThunk), this); gtk_box_pack_start(GTK_BOX(button_hbox), autofill_button_, FALSE, FALSE, 0); return vbox; } GtkWidget* ContentPageGtk::InitBrowsingDataGroup() { GtkWidget* button_box = gtk_hbox_new(FALSE, gtk_util::kControlSpacing); // Import button. GtkWidget* import_button = gtk_button_new_with_label( l10n_util::GetStringUTF8(IDS_OPTIONS_IMPORT_DATA_BUTTON).c_str()); g_signal_connect(import_button, "clicked", G_CALLBACK(OnImportButtonClickedThunk), this); gtk_box_pack_start(GTK_BOX(button_box), import_button, FALSE, FALSE, 0); return button_box; } GtkWidget* ContentPageGtk::InitThemesGroup() { GtkWidget* vbox = gtk_vbox_new(FALSE, gtk_util::kControlSpacing); GtkWidget* hbox = gtk_hbox_new(FALSE, gtk_util::kControlSpacing); #if defined(TOOLKIT_GTK) // GTK theme button. gtk_theme_button_ = gtk_button_new_with_label( l10n_util::GetStringUTF8(IDS_THEMES_GTK_BUTTON).c_str()); g_signal_connect(gtk_theme_button_, "clicked", G_CALLBACK(OnGtkThemeButtonClickedThunk), this); gtk_box_pack_start(GTK_BOX(hbox), gtk_theme_button_, FALSE, FALSE, 0); #endif // Reset theme button. themes_reset_button_ = gtk_button_new_with_label( l10n_util::GetStringUTF8(IDS_THEMES_SET_CLASSIC).c_str()); g_signal_connect(themes_reset_button_, "clicked", G_CALLBACK(OnResetDefaultThemeButtonClickedThunk), this); gtk_box_pack_start(GTK_BOX(hbox), themes_reset_button_, FALSE, FALSE, 0); // Get themes button. GtkWidget* themes_gallery_button = gtk_chrome_link_button_new( l10n_util::GetStringUTF8(IDS_THEMES_GALLERY_BUTTON).c_str()); g_signal_connect(themes_gallery_button, "clicked", G_CALLBACK(OnGetThemesButtonClickedThunk), this); gtk_box_pack_start(GTK_BOX(hbox), themes_gallery_button, FALSE, FALSE, 0); gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 0); // "Use system title bar and borders" radio buttons. if (browser_defaults::kCanToggleSystemTitleBar) { // Use system title bar and borders system_title_bar_show_radio_ = gtk_radio_button_new_with_label(NULL, l10n_util::GetStringUTF8(IDS_SHOW_WINDOW_DECORATIONS_RADIO).c_str()); g_signal_connect(system_title_bar_show_radio_, "toggled", G_CALLBACK(OnSystemTitleBarRadioToggledThunk), this); gtk_box_pack_start(GTK_BOX(vbox), system_title_bar_show_radio_, FALSE, FALSE, 0); // Hide system title bar and use custom borders system_title_bar_hide_radio_ = gtk_radio_button_new_with_label_from_widget( GTK_RADIO_BUTTON(system_title_bar_show_radio_), l10n_util::GetStringUTF8(IDS_HIDE_WINDOW_DECORATIONS_RADIO).c_str()); g_signal_connect(system_title_bar_hide_radio_, "toggled", G_CALLBACK(OnSystemTitleBarRadioToggledThunk), this); gtk_box_pack_start(GTK_BOX(vbox), system_title_bar_hide_radio_, FALSE, FALSE, 0); } return vbox; } GtkWidget* ContentPageGtk::InitSyncGroup() { GtkWidget* vbox = gtk_vbox_new(FALSE, gtk_util::kControlSpacing); // Sync label. sync_status_label_background_ = gtk_event_box_new(); sync_status_label_ = gtk_label_new(""); WrapLabelAtAllocationHack(sync_status_label_); gtk_label_set_line_wrap(GTK_LABEL(sync_status_label_), TRUE); gtk_misc_set_alignment(GTK_MISC(sync_status_label_), 0, 0); gtk_box_pack_start(GTK_BOX(vbox), sync_status_label_background_, FALSE, FALSE, 0); gtk_container_add(GTK_CONTAINER(sync_status_label_background_), sync_status_label_); #if !defined(OS_CHROMEOS) // Sync action link. GtkWidget* link_hbox = gtk_hbox_new(FALSE, gtk_util::kLabelSpacing); sync_action_link_background_ = gtk_event_box_new(); sync_action_link_ = gtk_chrome_link_button_new(""); g_signal_connect(sync_action_link_, "clicked", G_CALLBACK(OnSyncActionLinkClickedThunk), this); gtk_box_pack_start(GTK_BOX(vbox), link_hbox, FALSE, FALSE, 0); gtk_box_pack_start(GTK_BOX(link_hbox), sync_action_link_background_, FALSE, FALSE, 0); gtk_container_add(GTK_CONTAINER(sync_action_link_background_), sync_action_link_); gtk_widget_hide(sync_action_link_background_); #endif // Add the sync button into its own horizontal box so it does not // depend on the spacing above. GtkWidget* button_hbox = gtk_hbox_new(FALSE, gtk_util::kLabelSpacing); gtk_container_add(GTK_CONTAINER(vbox), button_hbox); #if !defined(OS_CHROMEOS) sync_start_stop_button_ = gtk_button_new_with_label(""); g_signal_connect(sync_start_stop_button_, "clicked", G_CALLBACK(OnSyncStartStopButtonClickedThunk), this); gtk_box_pack_start(GTK_BOX(button_hbox), sync_start_stop_button_, FALSE, FALSE, 0); #endif sync_customize_button_ = gtk_button_new_with_label(""); g_signal_connect(sync_customize_button_, "clicked", G_CALLBACK(OnSyncCustomizeButtonClickedThunk), this); gtk_box_pack_start(GTK_BOX(button_hbox), sync_customize_button_, FALSE, FALSE, 0); return vbox; } void ContentPageGtk::UpdateSyncControls() { DCHECK(sync_service_); string16 status_label; string16 link_label; std::string customize_button_label; std::string button_label; bool sync_setup_completed = sync_service_->HasSyncSetupCompleted(); bool status_has_error = sync_ui_util::GetStatusLabels(sync_service_, &status_label, &link_label) == sync_ui_util::SYNC_ERROR; customize_button_label = l10n_util::GetStringUTF8(IDS_SYNC_CUSTOMIZE_BUTTON_LABEL); if (sync_setup_completed) { button_label = l10n_util::GetStringUTF8(IDS_SYNC_STOP_SYNCING_BUTTON_LABEL); } else if (sync_service_->SetupInProgress()) { button_label = l10n_util::GetStringUTF8(IDS_SYNC_NTP_SETUP_IN_PROGRESS); } else { button_label = l10n_util::GetStringUTF8(IDS_SYNC_START_SYNC_BUTTON_LABEL); } gtk_label_set_label(GTK_LABEL(sync_status_label_), UTF16ToUTF8(status_label).c_str()); #if !defined(OS_CHROMEOS) gtk_widget_set_sensitive(sync_start_stop_button_, !sync_service_->WizardIsVisible()); gtk_button_set_label(GTK_BUTTON(sync_start_stop_button_), button_label.c_str()); #endif gtk_widget_set_child_visible(sync_customize_button_, sync_setup_completed && !status_has_error); gtk_button_set_label(GTK_BUTTON(sync_customize_button_), customize_button_label.c_str()); #if !defined(OS_CHROMEOS) gtk_chrome_link_button_set_label(GTK_CHROME_LINK_BUTTON(sync_action_link_), UTF16ToUTF8(link_label).c_str()); if (link_label.empty()) { gtk_widget_set_no_show_all(sync_action_link_background_, TRUE); gtk_widget_hide(sync_action_link_background_); } else { gtk_widget_set_no_show_all(sync_action_link_background_, FALSE); gtk_widget_show(sync_action_link_background_); } #endif if (status_has_error) { gtk_widget_modify_bg(sync_status_label_background_, GTK_STATE_NORMAL, &kSyncLabelErrorBgColor); #if !defined(OS_CHROMEOS) gtk_widget_modify_bg(sync_action_link_background_, GTK_STATE_NORMAL, &kSyncLabelErrorBgColor); #endif } else { gtk_widget_modify_bg(sync_status_label_background_, GTK_STATE_NORMAL, NULL); #if !defined(OS_CHROMEOS) gtk_widget_modify_bg(sync_action_link_background_, GTK_STATE_NORMAL, NULL); #endif } } void ContentPageGtk::OnAutoFillButtonClicked(GtkWidget* widget) { ShowAutoFillDialog( NULL, profile()->GetPersonalDataManager(), profile(), NULL, NULL); } void ContentPageGtk::OnImportButtonClicked(GtkWidget* widget) { ImportDialogGtk::Show( GTK_WINDOW(gtk_widget_get_toplevel(widget)), profile(), importer::ALL); } void ContentPageGtk::OnGtkThemeButtonClicked(GtkWidget* widget) { UserMetricsRecordAction(UserMetricsAction("Options_GtkThemeSet"), profile()->GetPrefs()); profile()->SetNativeTheme(); } void ContentPageGtk::OnResetDefaultThemeButtonClicked(GtkWidget* widget) { UserMetricsRecordAction(UserMetricsAction("Options_ThemesReset"), profile()->GetPrefs()); profile()->ClearTheme(); } void ContentPageGtk::OnGetThemesButtonClicked(GtkWidget* widget) { UserMetricsRecordAction(UserMetricsAction("Options_ThemesGallery"), profile()->GetPrefs()); #if defined(OS_CHROMEOS) // Close options dialog for ChromeOS becuase it is always stacked on top // of browser window and blocks user's view. chromeos::CloseOptionsWindow(); #endif // defined(OS_CHROMEOS) BrowserList::GetLastActive()->OpenThemeGalleryTabAndActivate(); } void ContentPageGtk::OnSystemTitleBarRadioToggled(GtkWidget* widget) { DCHECK(browser_defaults::kCanToggleSystemTitleBar); if (initializing_) return; // We get two signals when selecting a radio button, one for the old radio // being toggled off and one for the new one being toggled on. Ignore the // signal for the toggling off the old button. if (!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget))) return; bool use_custom = gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON(system_title_bar_hide_radio_)); if (use_custom) { UserMetricsRecordAction(UserMetricsAction("Options_CustomFrame_Enable"), profile()->GetPrefs()); } else { UserMetricsRecordAction(UserMetricsAction("Options_CustomFrame_Disable"), profile()->GetPrefs()); } use_custom_chrome_frame_.SetValue(use_custom); } void ContentPageGtk::OnShowPasswordsButtonClicked(GtkWidget* widget) { ShowPasswordsExceptionsWindow(profile()); } void ContentPageGtk::OnPasswordRadioToggled(GtkWidget* widget) { if (initializing_) return; // We get two signals when selecting a radio button, one for the old radio // being toggled off and one for the new one being toggled on. Ignore the // signal for the toggling off the old button. if (!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget))) return; bool enabled = gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON(passwords_asktosave_radio_)); if (enabled) { UserMetricsRecordAction(UserMetricsAction("Options_PasswordManager_Enable"), profile()->GetPrefs()); } else { UserMetricsRecordAction( UserMetricsAction("Options_PasswordManager_Disable"), profile()->GetPrefs()); } ask_to_save_passwords_.SetValue(enabled); } void ContentPageGtk::OnAutoFillRadioToggled(GtkWidget* widget) { if (initializing_) return; // We get two signals when selecting a radio button, one for the old radio // being toggled off and one for the new one being toggled on. Ignore the // signal for the toggling off the old button. if (!gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(widget))) return; bool enabled = gtk_toggle_button_get_active( GTK_TOGGLE_BUTTON(form_autofill_enable_radio_)); if (enabled) { UserMetricsRecordAction(UserMetricsAction("Options_FormAutofill_Enable"), profile()->GetPrefs()); gtk_widget_set_sensitive(autofill_button_, TRUE); } else { UserMetricsRecordAction(UserMetricsAction("Options_FormAutofill_Disable"), profile()->GetPrefs()); gtk_widget_set_sensitive(autofill_button_, FALSE); } enable_form_autofill_.SetValue(enabled); } void ContentPageGtk::OnSyncStartStopButtonClicked(GtkWidget* widget) { DCHECK(sync_service_); if (sync_service_->HasSyncSetupCompleted()) { GtkWidget* dialog = gtk_message_dialog_new( GTK_WINDOW(gtk_widget_get_toplevel(widget)), static_cast(GTK_DIALOG_MODAL), GTK_MESSAGE_WARNING, GTK_BUTTONS_NONE, "%s", l10n_util::GetStringFUTF8( IDS_SYNC_STOP_SYNCING_EXPLANATION_LABEL, l10n_util::GetStringUTF16(IDS_PRODUCT_NAME)).c_str()); gtk_util::ApplyMessageDialogQuirks(dialog); gtk_window_set_title(GTK_WINDOW(dialog), l10n_util::GetStringUTF8( IDS_SYNC_STOP_SYNCING_DIALOG_TITLE).c_str()); gtk_dialog_add_buttons( GTK_DIALOG(dialog), l10n_util::GetStringUTF8(IDS_CANCEL).c_str(), GTK_RESPONSE_REJECT, l10n_util::GetStringUTF8( IDS_SYNC_STOP_SYNCING_CONFIRM_BUTTON_LABEL).c_str(), GTK_RESPONSE_ACCEPT, NULL); g_signal_connect(dialog, "response", G_CALLBACK(OnStopSyncDialogResponseThunk), this); gtk_util::ShowDialog(dialog); return; } else { sync_service_->EnableForUser(); ProfileSyncService::SyncEvent(ProfileSyncService::START_FROM_OPTIONS); } } void ContentPageGtk::OnSyncCustomizeButtonClicked(GtkWidget* widget) { // sync_customize_button_ should be invisible if sync is not yet set up. DCHECK(sync_service_->HasSyncSetupCompleted()); ShowCustomizeSyncWindow(profile()); } void ContentPageGtk::OnSyncActionLinkClicked(GtkWidget* widget) { DCHECK(sync_service_); sync_service_->ShowLoginDialog(); } void ContentPageGtk::OnStopSyncDialogResponse(GtkWidget* widget, int response) { if (response == GTK_RESPONSE_ACCEPT) { sync_service_->DisableForUser(); ProfileSyncService::SyncEvent(ProfileSyncService::STOP_FROM_OPTIONS); } gtk_widget_destroy(widget); }