| OLD | NEW |
| (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 "chrome/browser/gtk/first_run_bubble.h" | |
| 6 | |
| 7 #include <gtk/gtk.h> | |
| 8 | |
| 9 #include "app/l10n_util.h" | |
| 10 #include "base/command_line.h" | |
| 11 #include "base/i18n/rtl.h" | |
| 12 #include "base/utf_string_conversions.h" | |
| 13 #include "chrome/browser/browser_list.h" | |
| 14 #include "chrome/browser/gtk/gtk_theme_provider.h" | |
| 15 #include "chrome/browser/gtk/gtk_util.h" | |
| 16 #include "chrome/browser/search_engines/util.h" | |
| 17 #include "chrome/browser/ui/browser.h" | |
| 18 #include "chrome/common/notification_service.h" | |
| 19 #include "grit/chromium_strings.h" | |
| 20 #include "grit/generated_resources.h" | |
| 21 #include "grit/locale_settings.h" | |
| 22 | |
| 23 namespace { | |
| 24 // Markup for the text of the Omnibox search label | |
| 25 const char kSearchLabelMarkup[] = "<big><b>%s</b></big>"; | |
| 26 | |
| 27 // Padding for the buttons on first run bubble. | |
| 28 const int kButtonPadding = 4; | |
| 29 | |
| 30 // Padding between content and edge of info bubble. | |
| 31 const int kContentBorder = 7; | |
| 32 | |
| 33 // Vertical spacing between labels. | |
| 34 const int kInterLineSpacing = 5; | |
| 35 | |
| 36 } // namespace | |
| 37 | |
| 38 // static | |
| 39 void FirstRunBubble::Show(Profile* profile, | |
| 40 GtkWidget* anchor, | |
| 41 const gfx::Rect& rect, | |
| 42 FirstRun::BubbleType bubble_type) { | |
| 43 new FirstRunBubble(profile, anchor, rect, bubble_type); | |
| 44 } | |
| 45 | |
| 46 void FirstRunBubble::InfoBubbleClosing(InfoBubbleGtk* info_bubble, | |
| 47 bool closed_by_escape) { | |
| 48 // TODO(port): Enable parent window | |
| 49 } | |
| 50 | |
| 51 bool FirstRunBubble::CloseOnEscape() { | |
| 52 return true; | |
| 53 } | |
| 54 | |
| 55 void FirstRunBubble::Observe(NotificationType type, | |
| 56 const NotificationSource& source, | |
| 57 const NotificationDetails& details) { | |
| 58 DCHECK(type == NotificationType::BROWSER_THEME_CHANGED); | |
| 59 | |
| 60 if (theme_provider_->UseGtkTheme()) { | |
| 61 for (std::vector<GtkWidget*>::iterator it = labels_.begin(); | |
| 62 it != labels_.end(); ++it) { | |
| 63 gtk_widget_modify_fg(*it, GTK_STATE_NORMAL, NULL); | |
| 64 } | |
| 65 } else { | |
| 66 for (std::vector<GtkWidget*>::iterator it = labels_.begin(); | |
| 67 it != labels_.end(); ++it) { | |
| 68 gtk_widget_modify_fg(*it, GTK_STATE_NORMAL, >k_util::kGdkBlack); | |
| 69 } | |
| 70 } | |
| 71 } | |
| 72 | |
| 73 FirstRunBubble::FirstRunBubble(Profile* profile, | |
| 74 GtkWidget* anchor, | |
| 75 const gfx::Rect& rect, | |
| 76 FirstRun::BubbleType bubble_type) | |
| 77 : profile_(profile), | |
| 78 theme_provider_(GtkThemeProvider::GetFrom(profile_)), | |
| 79 anchor_(anchor), | |
| 80 content_(NULL), | |
| 81 bubble_(NULL) { | |
| 82 content_ = gtk_vbox_new(FALSE, kInterLineSpacing); | |
| 83 gtk_container_set_border_width(GTK_CONTAINER(content_), kContentBorder); | |
| 84 g_signal_connect(content_, "destroy", | |
| 85 G_CALLBACK(&HandleDestroyThunk), this); | |
| 86 | |
| 87 int width_resource = 0; | |
| 88 if (bubble_type == FirstRun::LARGE_BUBBLE) { | |
| 89 width_resource = IDS_FIRSTRUNBUBBLE_DIALOG_WIDTH_CHARS; | |
| 90 InitializeContentForLarge(); | |
| 91 } else if (bubble_type == FirstRun::OEM_BUBBLE) { | |
| 92 width_resource = IDS_FIRSTRUNOEMBUBBLE_DIALOG_WIDTH_CHARS; | |
| 93 InitializeContentForOEM(); | |
| 94 } else if (bubble_type == FirstRun::MINIMAL_BUBBLE) { | |
| 95 width_resource = IDS_FIRSTRUN_MINIMAL_BUBBLE_DIALOG_WIDTH_CHARS; | |
| 96 InitializeContentForMinimal(); | |
| 97 } else { | |
| 98 NOTREACHED(); | |
| 99 } | |
| 100 | |
| 101 InitializeLabels(width_resource); | |
| 102 | |
| 103 InfoBubbleGtk::ArrowLocationGtk arrow_location = | |
| 104 !base::i18n::IsRTL() ? | |
| 105 InfoBubbleGtk::ARROW_LOCATION_TOP_LEFT : | |
| 106 InfoBubbleGtk::ARROW_LOCATION_TOP_RIGHT; | |
| 107 bubble_ = InfoBubbleGtk::Show(anchor_, | |
| 108 &rect, | |
| 109 content_, | |
| 110 arrow_location, | |
| 111 true, // match_system_theme | |
| 112 true, // grab_input | |
| 113 theme_provider_, | |
| 114 this); // delegate | |
| 115 if (!bubble_) { | |
| 116 NOTREACHED(); | |
| 117 return; | |
| 118 } | |
| 119 | |
| 120 registrar_.Add(this, NotificationType::BROWSER_THEME_CHANGED, | |
| 121 NotificationService::AllSources()); | |
| 122 theme_provider_->InitThemesFor(this); | |
| 123 } | |
| 124 | |
| 125 FirstRunBubble::~FirstRunBubble() { | |
| 126 } | |
| 127 | |
| 128 void FirstRunBubble::InitializeContentForLarge() { | |
| 129 GtkWidget* label1 = gtk_label_new(NULL); | |
| 130 labels_.push_back(label1); | |
| 131 char* markup = g_markup_printf_escaped(kSearchLabelMarkup, | |
| 132 l10n_util::GetStringUTF8(IDS_FR_BUBBLE_TITLE).c_str()); | |
| 133 gtk_label_set_markup(GTK_LABEL(label1), markup); | |
| 134 g_free(markup); | |
| 135 | |
| 136 GtkWidget* label2 = gtk_label_new( | |
| 137 l10n_util::GetStringUTF8(IDS_FR_BUBBLE_SUBTEXT).c_str()); | |
| 138 labels_.push_back(label2); | |
| 139 | |
| 140 string16 search_engine = GetDefaultSearchEngineName(profile_); | |
| 141 GtkWidget* label3 = gtk_label_new( | |
| 142 l10n_util::GetStringFUTF8(IDS_FR_BUBBLE_QUESTION, search_engine).c_str()); | |
| 143 labels_.push_back(label3); | |
| 144 | |
| 145 GtkWidget* keep_button = gtk_button_new_with_label( | |
| 146 l10n_util::GetStringFUTF8(IDS_FR_BUBBLE_OK, search_engine).c_str()); | |
| 147 GtkWidget* change_button = gtk_button_new_with_label( | |
| 148 l10n_util::GetStringUTF8(IDS_FR_BUBBLE_CHANGE).c_str()); | |
| 149 | |
| 150 gtk_box_pack_start(GTK_BOX(content_), label1, FALSE, FALSE, 0); | |
| 151 gtk_box_pack_start(GTK_BOX(content_), label2, FALSE, FALSE, 0); | |
| 152 // Leave an empty line. | |
| 153 gtk_box_pack_start(GTK_BOX(content_), gtk_label_new(NULL), FALSE, FALSE, 0); | |
| 154 gtk_box_pack_start(GTK_BOX(content_), label3, FALSE, FALSE, 0); | |
| 155 | |
| 156 GtkWidget* bottom = gtk_hbox_new(FALSE, 0); | |
| 157 // We want the buttons on the right, so just use an expanding label to fill | |
| 158 // all of the extra space on the left. | |
| 159 gtk_box_pack_start(GTK_BOX(bottom), gtk_label_new(NULL), TRUE, TRUE, 0); | |
| 160 gtk_box_pack_start(GTK_BOX(bottom), keep_button, FALSE, FALSE, | |
| 161 kButtonPadding); | |
| 162 gtk_box_pack_start(GTK_BOX(bottom), change_button, FALSE, FALSE, 0); | |
| 163 | |
| 164 gtk_box_pack_start(GTK_BOX(content_), bottom, FALSE, FALSE, 0); | |
| 165 // We want the focus to start on the keep entry, not on the change button. | |
| 166 gtk_widget_grab_focus(keep_button); | |
| 167 | |
| 168 g_signal_connect(keep_button, "clicked", | |
| 169 G_CALLBACK(&HandleKeepButtonThunk), this); | |
| 170 g_signal_connect(change_button, "clicked", | |
| 171 G_CALLBACK(&HandleChangeButtonThunk), this); | |
| 172 } | |
| 173 | |
| 174 void FirstRunBubble::InitializeContentForOEM() { | |
| 175 NOTIMPLEMENTED() << "Falling back to minimal bubble"; | |
| 176 InitializeContentForMinimal(); | |
| 177 } | |
| 178 | |
| 179 void FirstRunBubble::InitializeContentForMinimal() { | |
| 180 GtkWidget* label1 = gtk_label_new(NULL); | |
| 181 labels_.push_back(label1); | |
| 182 char* markup = g_markup_printf_escaped(kSearchLabelMarkup, | |
| 183 l10n_util::GetStringFUTF8( | |
| 184 IDS_FR_SE_BUBBLE_TITLE, | |
| 185 GetDefaultSearchEngineName(profile_)).c_str()); | |
| 186 gtk_label_set_markup(GTK_LABEL(label1), markup); | |
| 187 g_free(markup); | |
| 188 | |
| 189 GtkWidget* label2 = | |
| 190 gtk_label_new(l10n_util::GetStringUTF8(IDS_FR_BUBBLE_SUBTEXT).c_str()); | |
| 191 labels_.push_back(label2); | |
| 192 | |
| 193 gtk_box_pack_start(GTK_BOX(content_), label1, FALSE, FALSE, 0); | |
| 194 gtk_box_pack_start(GTK_BOX(content_), label2, FALSE, FALSE, 0); | |
| 195 } | |
| 196 | |
| 197 void FirstRunBubble::InitializeLabels(int width_resource) { | |
| 198 int width = -1; | |
| 199 | |
| 200 gtk_util::GetWidgetSizeFromResources( | |
| 201 anchor_, width_resource, 0, &width, NULL); | |
| 202 | |
| 203 for (size_t i = 0; i < labels_.size(); ++i) { | |
| 204 // Resize the labels so that they don't wrap more than necessary. We leave | |
| 205 // |content_| unsized so that it'll expand as needed to hold the other | |
| 206 // widgets -- the buttons may be wider than |width| on high-DPI displays. | |
| 207 gtk_util::SetLabelWidth(labels_[i], width); | |
| 208 gtk_misc_set_alignment(GTK_MISC(labels_[i]), 0, 0.5); | |
| 209 } | |
| 210 } | |
| 211 | |
| 212 void FirstRunBubble::HandleDestroy(GtkWidget* sender) { | |
| 213 content_ = NULL; | |
| 214 delete this; | |
| 215 } | |
| 216 | |
| 217 void FirstRunBubble::HandleKeepButton(GtkWidget* sender) { | |
| 218 bubble_->Close(); | |
| 219 } | |
| 220 | |
| 221 void FirstRunBubble::HandleChangeButton(GtkWidget* sender) { | |
| 222 bubble_->Close(); | |
| 223 Browser* browser = BrowserList::GetLastActive(); | |
| 224 DCHECK(browser); | |
| 225 browser->OpenSearchEngineOptionsDialog(); | |
| 226 } | |
| OLD | NEW |