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

Side by Side Diff: extensions/common/manifest_handlers/options_page_info.cc

Issue 518653002: Add the "options_ui" extension manifest field. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Address comments Created 6 years, 3 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
OLDNEW
(Empty)
1 // Copyright 2014 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 "extensions/common/manifest_handlers/options_page_info.h"
6
7 #include "base/file_util.h"
8 #include "base/memory/scoped_ptr.h"
9 #include "base/strings/utf_string_conversions.h"
10 #include "extensions/common/api/extensions_manifest_types.h"
11 #include "extensions/common/error_utils.h"
12 #include "extensions/common/feature_switch.h"
13 #include "extensions/common/file_util.h"
14 #include "extensions/common/manifest_constants.h"
15 #include "extensions/strings/grit/extensions_strings.h"
16 #include "ui/base/l10n/l10n_util.h"
17
18 using base::ASCIIToUTF16;
19 using base::DictionaryValue;
20
21 namespace extensions {
22
23 namespace keys = manifest_keys;
24 namespace values = manifest_values;
Devlin 2014/09/04 21:16:04 Is this used?
ericzeng 2014/09/04 22:22:07 No, must have left it there when refactoring in th
25 namespace errors = manifest_errors;
26
27 using core_api::extensions_manifest_types::OptionsUI;
28
29 namespace {
30
31 OptionsPageInfo* GetOptionsPageInfo(const Extension* extension) {
32 return static_cast<OptionsPageInfo*>(
33 extension->GetManifestData(keys::kOptionsUI));
34 }
35
36 // Parses |url_string| into a GURL |result| if it is a valid options page for
37 // this app/extension. If not, it returns the reason in |error|. Because this
38 // handles URLs for both "options_page" and "options_ui.page", the name of the
39 // manifest field must be provided in |manifest_field_name|.
40 bool ParseOptionsUrl(Extension* extension,
41 const std::string& url_string,
42 const std::string& manifest_field_name,
43 base::string16* error,
44 GURL* result) {
45 if (extension->is_hosted_app()) {
46 // Hosted apps require an absolute URL.
47 GURL options_url(url_string);
48 if (!options_url.is_valid() || !options_url.SchemeIsHTTPOrHTTPS()) {
49 *error = base::ASCIIToUTF16(errors::kInvalidOptionsPageInHostedApp);
50 return false;
51 }
52 *result = options_url;
53 return true;
54 }
55
56 // Otherwise the options URL should be inside the extension.
57 if (GURL(url_string).is_valid()) {
58 *error = base::ASCIIToUTF16(errors::kInvalidOptionsPageExpectUrlInPackage);
59 return false;
60 }
61
62 GURL resource_url = extension->GetResourceURL(url_string);
63 if (!resource_url.is_valid()) {
64 *error = ErrorUtils::FormatErrorMessageUTF16(errors::kInvalidOptionsPage,
65 manifest_field_name);
66 return false;
67 }
68 *result = resource_url;
69 return true;
70 }
71
72 } // namespace
73
74 OptionsPageInfo::OptionsPageInfo(const GURL& options_page,
75 bool chrome_styles,
76 bool open_in_tab)
77 : options_page_(options_page),
78 chrome_styles_(chrome_styles),
79 open_in_tab_(open_in_tab) {
80 }
81
82 OptionsPageInfo::~OptionsPageInfo() {
83 }
84
85 // static
86 const GURL& OptionsPageInfo::GetOptionsPage(const Extension* extension) {
87 OptionsPageInfo* info = GetOptionsPageInfo(extension);
88 if (!info)
Yoyo Zhou 2014/09/04 21:15:09 Use ternary operator as Devlin suggests.
Devlin 2014/09/04 21:16:04 inline: info ? info->options_page_ : GURL::EmptyGU
ericzeng 2014/09/04 22:22:07 Done.
89 return GURL::EmptyGURL();
90 return info->options_page_;
91 }
92
93 // static
94 bool OptionsPageInfo::HasOptionsPage(const Extension* extension) {
95 return !OptionsPageInfo::GetOptionsPage(extension).is_empty();
96 }
97
98 // static
99 bool OptionsPageInfo::ShouldUseChromeStyle(const Extension* extension) {
100 OptionsPageInfo* info = GetOptionsPageInfo(extension);
101 return info && info->chrome_styles_;
102 }
103
104 // static
105 bool OptionsPageInfo::ShouldOpenInTab(const Extension* extension) {
106 OptionsPageInfo* info = GetOptionsPageInfo(extension);
107 return info && info->open_in_tab_;
108 }
109
110 scoped_ptr<OptionsPageInfo> OptionsPageInfo::Create(
111 Extension* extension,
112 const base::Value* options_ui_value,
113 const std::string& options_page_string,
114 std::vector<InstallWarning>* install_warnings,
115 base::string16* error) {
116 GURL options_page;
117 bool chrome_style = false;
118 bool open_in_tab = !FeatureSwitch::embedded_extension_options()->IsEnabled();
Devlin 2014/09/04 21:16:04 This looks wrong. We default to opening in a tab
ericzeng 2014/09/04 22:22:07 So if the embedded extension options flag is turne
Devlin 2014/09/04 22:45:32 Apparently I'm not thinking straight today - the w
119
120 // Parse the options_ui object.
121 if (options_ui_value &&
122 FeatureSwitch::embedded_extension_options()->IsEnabled()) {
123 base::string16 options_ui_error;
124
125 scoped_ptr<OptionsUI> options_ui =
126 OptionsUI::FromValue(*options_ui_value, &options_ui_error);
127 if (!options_ui_error.empty()) {
128 // OptionsUI::FromValue populates |error| both when there are
129 // errors (in which case |options_ui| will be NULL) and warnings
130 // (in which case |options_ui| will be valid). Either way, show it
131 // as an install warning.
132 install_warnings->push_back(
133 InstallWarning(base::UTF16ToASCII(options_ui_error)));
134 }
135
136 if (options_ui) {
137 base::string16 options_parse_error;
138 if (!ParseOptionsUrl(extension,
139 options_ui->page,
140 keys::kOptionsUI,
141 &options_parse_error,
142 &options_page)) {
143 install_warnings->push_back(
144 InstallWarning(base::UTF16ToASCII(options_parse_error)));
145 }
146 chrome_style =
147 options_ui->chrome_style.get() && *options_ui->chrome_style;
148 open_in_tab = options_ui->open_in_tab.get() && *options_ui->open_in_tab;
Devlin 2014/09/04 21:16:04 This will override your default set above, because
ericzeng 2014/09/04 22:22:07 I think that is the intended behavior, since we wa
Devlin 2014/09/04 22:45:32 Right, see above.
149 }
150 }
151
152 // Parse the legacy options_page entry if there was no entry for
153 // options_ui.page.
154 if (!options_page_string.empty() && !options_page.is_valid()) {
155 if (!ParseOptionsUrl(extension,
156 options_page_string,
157 keys::kOptionsPage,
158 error,
159 &options_page)) {
160 return scoped_ptr<OptionsPageInfo>();
161 }
162 }
163
164 return make_scoped_ptr(
165 new OptionsPageInfo(options_page, chrome_style, open_in_tab));
166 }
167
168 OptionsPageManifestHandler::OptionsPageManifestHandler() {
169 }
170
171 OptionsPageManifestHandler::~OptionsPageManifestHandler() {
172 }
173
174 bool OptionsPageManifestHandler::Parse(Extension* extension,
175 base::string16* error) {
176 std::vector<InstallWarning> install_warnings;
177 const Manifest* manifest = extension->manifest();
178
179 std::string options_page_string;
180 if (manifest->HasPath(keys::kOptionsPage) &&
181 !manifest->GetString(keys::kOptionsPage, &options_page_string)) {
182 *error = ErrorUtils::FormatErrorMessageUTF16(errors::kInvalidOptionsPage,
183 keys::kOptionsPage);
184 return false;
185 }
186
187 const base::Value* options_ui_value = NULL;
188 ignore_result(manifest->Get(keys::kOptionsUI, &options_ui_value));
189
190 scoped_ptr<OptionsPageInfo> info =
191 OptionsPageInfo::Create(extension,
192 options_ui_value,
193 options_page_string,
194 &install_warnings,
195 error);
196 if (!info)
197 return false;
198
199 extension->AddInstallWarnings(install_warnings);
Devlin 2014/09/04 21:16:04 Do we want an install warning if the extension spe
ericzeng 2014/09/04 22:22:07 Maybe not? An extension might want to keep the old
200 extension->SetManifestData(keys::kOptionsUI, info.release());
201 return true;
202 }
203
204 bool OptionsPageManifestHandler::Validate(
205 const Extension* extension,
206 std::string* error,
207 std::vector<InstallWarning>* warnings) const {
208 // Validate path to the options page. Don't check the URL for hosted apps,
209 // because they are expected to refer to an external URL.
210 if (!OptionsPageInfo::HasOptionsPage(extension) || extension->is_hosted_app())
211 return true;
212
213 base::FilePath options_path = file_util::ExtensionURLToRelativeFilePath(
214 OptionsPageInfo::GetOptionsPage(extension));
215 base::FilePath path = extension->GetResource(options_path).GetFilePath();
216 if (path.empty() || !base::PathExists(path)) {
217 *error = l10n_util::GetStringFUTF8(IDS_EXTENSION_LOAD_OPTIONS_PAGE_FAILED,
218 options_path.LossyDisplayName());
219 return false;
220 }
221 return true;
222 }
223
224 const std::vector<std::string> OptionsPageManifestHandler::Keys() const {
225 static const char* keys[] = {keys::kOptionsPage, keys::kOptionsUI};
226 return std::vector<std::string>(keys, keys + arraysize(keys));
227 }
228
229 } // namespace extensions
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698