blob: ac9fe80d0e6bd12eebe127e4ac7e0affbc3d4aeb [file] [log] [blame]
sdefresne14900ee2015-11-27 14:43:211// Copyright 2012 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// Implementation of about_flags for iOS that sets flags based on experimental
6// settings.
7
8#include "ios/chrome/browser/about_flags.h"
9
avi571943672015-12-22 02:12:4910#include <stddef.h>
11#include <stdint.h>
sdefresne14900ee2015-11-27 14:43:2112#import <UIKit/UIKit.h>
13
Gregory Chatzinofff6523722017-11-21 01:33:5314#include "base/base_switches.h"
sdefresne14900ee2015-11-27 14:43:2115#include "base/bind.h"
16#include "base/command_line.h"
17#include "base/logging.h"
18#include "base/memory/singleton.h"
kokihoon189a4022018-10-13 02:43:4419#include "base/stl_util.h"
sdefresne14900ee2015-11-27 14:43:2120#include "base/strings/stringprintf.h"
21#include "base/strings/sys_string_conversions.h"
Sebastien Marchand75a7cdf2018-11-13 23:47:0322#include "base/system/sys_info.h"
Mohamad Ahmadic4df81f2017-12-20 04:41:5923#include "components/autofill/core/common/autofill_features.h"
Moe Ahmadid6d5d172018-06-20 17:20:2324#include "components/autofill/core/common/autofill_switches.h"
Olivier Robin3d60411622018-02-23 10:03:2225#include "components/autofill/ios/browser/autofill_switches.h"
sdefresne14900ee2015-11-27 14:43:2126#include "components/dom_distiller/core/dom_distiller_switches.h"
Tommy Nyquistc1d6dea12017-07-26 20:37:2327#include "components/feature_engagement/public/feature_constants.h"
28#include "components/feature_engagement/public/feature_list.h"
sdefresne14900ee2015-11-27 14:43:2129#include "components/flags_ui/feature_entry.h"
30#include "components/flags_ui/feature_entry_macros.h"
31#include "components/flags_ui/flags_storage.h"
32#include "components/flags_ui/flags_ui_switches.h"
Tanja Gornakb0985dd2018-10-11 17:24:4033#include "components/invalidation/impl/invalidation_switches.h"
noyau4cfb1332016-10-25 17:05:4234#include "components/ntp_tiles/switches.h"
Justin Donnelly33d712e2017-08-23 21:32:5135#include "components/omnibox/browser/omnibox_field_trial.h"
Ioana Pandeled14ce9e2017-11-28 14:41:4636#include "components/password_manager/core/common/password_manager_features.h"
mathp9b4c11d2017-07-06 20:24:1337#include "components/payments/core/features.h"
Marc Treib5fc99e1f2017-11-08 12:19:1838#include "components/search_provider_logos/switches.h"
Emily Starkbafa9062017-12-27 15:22:4639#include "components/security_state/core/features.h"
Valeriya Sinevicha79dc612018-09-25 14:49:0940#include "components/signin/core/browser/account_reconcilor.h"
David Rogera6c88122017-10-25 13:02:4641#include "components/signin/core/browser/signin_switches.h"
sdefresne36579782016-02-05 11:08:2542#include "components/strings/grit/components_strings.h"
Moe Ahmadid6d5d172018-06-20 17:20:2343#include "components/sync/driver/sync_driver_switches.h"
Moe Ahmadie4cce1f2018-12-18 23:48:0144#include "components/translate/core/browser/translate_prefs.h"
Mihai Sardarescu160ec662018-07-18 21:13:5145#include "components/unified_consent/feature.h"
mrefaat4539b032018-08-02 22:25:4546#include "ios/chrome/browser/app_launcher/app_launcher_flags.h"
Sylvain Defresneae6987e2018-03-01 13:21:4047#include "ios/chrome/browser/browsing_data/browsing_data_features.h"
sdefresne14900ee2015-11-27 14:43:2148#include "ios/chrome/browser/chrome_switches.h"
Olivier Robin2a898dc2018-10-31 14:37:0449#include "ios/chrome/browser/crash_report/crash_report_flags.h"
Moe Ahmadi2d4c8452018-11-30 04:44:2650#include "ios/chrome/browser/download/features.h"
Jean-François Geyelin30be5912017-08-25 12:05:3851#include "ios/chrome/browser/drag_and_drop/drag_and_drop_flag.h"
Eugene But66e917002019-01-07 23:38:5252#include "ios/chrome/browser/experimental_flags.h"
Chris Lu77905ac2018-12-06 20:46:5153#include "ios/chrome/browser/find_in_page/features.h"
vabr0215a8e2017-03-28 12:47:3454#include "ios/chrome/browser/ios_chrome_flag_descriptions.h"
mrefaat390f99c0d2018-05-10 19:44:2455#include "ios/chrome/browser/itunes_urls/itunes_urls_flag.h"
Yi Sua3d55f3d2018-09-24 17:49:4156#include "ios/chrome/browser/search_engines/feature_flags.h"
Jérôme Lebelec282832018-09-18 08:34:1857#include "ios/chrome/browser/signin/feature_flags.h"
Kurt Horimoto9165404a2018-11-13 23:48:1258#import "ios/chrome/browser/ui/dialogs/dialog_features.h"
Kurt Horimoto4da682b022018-04-12 07:43:3559#import "ios/chrome/browser/ui/fullscreen/fullscreen_features.h"
sczs80a6f462018-12-05 20:37:1860#import "ios/chrome/browser/ui/infobars/infobar_feature.h"
Eugene Butc61230ab2018-11-09 00:31:5861#include "ios/chrome/browser/ui/sad_tab/features.h"
Mark Cogan02ed6c772018-05-31 17:00:5162#import "ios/chrome/browser/ui/toolbar/public/features.h"
Kurt Horimoto79d590b2018-09-12 19:09:2863#import "ios/chrome/browser/ui/toolbar_container/toolbar_container_features.h"
Justin Cohen2d81c582018-01-22 14:46:4464#include "ios/chrome/browser/ui/ui_feature_flags.h"
Yi Su392b3032018-06-05 07:26:4765#include "ios/chrome/browser/web/features.h"
sdefresne14900ee2015-11-27 14:43:2166#include "ios/chrome/grit/ios_strings.h"
msardafc76f662017-02-24 12:46:2867#include "ios/public/provider/chrome/browser/chrome_browser_provider.h"
Danyao Wang34da6ff2017-10-30 15:16:5668#include "ios/web/public/features.h"
sdefresne14900ee2015-11-27 14:43:2169#include "ios/web/public/user_agent.h"
70#include "ios/web/public/web_view_creation_util.h"
71
72#if !defined(OFFICIAL_BUILD)
73#include "components/variations/variations_switches.h"
vitaliii489217aa2017-01-30 14:50:2274#endif
stkhapuginc1be1792016-12-13 14:30:5375
76#if !defined(__has_feature) || !__has_feature(objc_arc)
77#error "This file requires ARC support."
78#endif
sdefresne14900ee2015-11-27 14:43:2179
elawrence816f6790e2017-06-16 18:19:2880using flags_ui::FeatureEntry;
81
sdefresne14900ee2015-11-27 14:43:2182namespace {
Emily Starkbafa9062017-12-27 15:22:4683
84const FeatureEntry::FeatureParam kMarkHttpAsDangerous[] = {
85 {security_state::features::kMarkHttpAsFeatureParameterName,
86 security_state::features::kMarkHttpAsParameterDangerous}};
Emily Starkbafa9062017-12-27 15:22:4687const FeatureEntry::FeatureParam kMarkHttpAsWarningAndDangerousOnFormEdits[] = {
88 {security_state::features::kMarkHttpAsFeatureParameterName,
89 security_state::features::
90 kMarkHttpAsParameterWarningAndDangerousOnFormEdits}};
Emily Starkbafa9062017-12-27 15:22:4691
92const FeatureEntry::FeatureVariation kMarkHttpAsFeatureVariations[] = {
93 {"(mark as actively dangerous)", kMarkHttpAsDangerous,
kokihoon189a4022018-10-13 02:43:4494 base::size(kMarkHttpAsDangerous), nullptr},
Emily Starkbafa9062017-12-27 15:22:4695 {"(mark with a Not Secure warning and dangerous on form edits)",
96 kMarkHttpAsWarningAndDangerousOnFormEdits,
Joe DeBlasio20e01fd2018-12-20 19:59:3097 base::size(kMarkHttpAsWarningAndDangerousOnFormEdits), nullptr}};
elawrence816f6790e2017-06-16 18:19:2898
Marc Treib5fc99e1f2017-11-08 12:19:1899const FeatureEntry::Choice kUseDdljsonApiChoices[] = {
100 {flags_ui::kGenericExperimentChoiceDefault, "", ""},
101 {"(force test doodle 0)", search_provider_logos::switches::kGoogleDoodleUrl,
102 "https://www.gstatic.com/chrome/ntp/doodle_test/ddljson_ios0.json"},
103 {"(force test doodle 1)", search_provider_logos::switches::kGoogleDoodleUrl,
104 "https://www.gstatic.com/chrome/ntp/doodle_test/ddljson_ios1.json"},
105 {"(force test doodle 2)", search_provider_logos::switches::kGoogleDoodleUrl,
106 "https://www.gstatic.com/chrome/ntp/doodle_test/ddljson_ios2.json"},
107 {"(force test doodle 3)", search_provider_logos::switches::kGoogleDoodleUrl,
108 "https://www.gstatic.com/chrome/ntp/doodle_test/ddljson_ios3.json"},
109 {"(force test doodle 4)", search_provider_logos::switches::kGoogleDoodleUrl,
110 "https://www.gstatic.com/chrome/ntp/doodle_test/ddljson_ios4.json"},
111};
Marc Treib2752e8b2017-08-04 14:12:09112
Olivier Robin3d60411622018-02-23 10:03:22113const FeatureEntry::Choice kAutofillIOSDelayBetweenFieldsChoices[] = {
114 {flags_ui::kGenericExperimentChoiceDefault, "", ""},
115 {"0", autofill::switches::kAutofillIOSDelayBetweenFields, "0"},
116 {"10", autofill::switches::kAutofillIOSDelayBetweenFields, "10"},
117 {"20", autofill::switches::kAutofillIOSDelayBetweenFields, "20"},
118 {"50", autofill::switches::kAutofillIOSDelayBetweenFields, "50"},
119 {"100", autofill::switches::kAutofillIOSDelayBetweenFields, "100"},
120 {"200", autofill::switches::kAutofillIOSDelayBetweenFields, "200"},
121 {"500", autofill::switches::kAutofillIOSDelayBetweenFields, "500"},
122 {"1000", autofill::switches::kAutofillIOSDelayBetweenFields, "1000"},
123};
124
Gauthier Ambard2392f2a2018-10-24 16:33:27125const FeatureEntry::FeatureParam kIconForSearchButtonGrey[] = {
126 {kIconForSearchButtonFeatureParameterName,
127 kIconForSearchButtonParameterGrey}};
128const FeatureEntry::FeatureParam kIconForSearchButtonColorful[] = {
129 {kIconForSearchButtonFeatureParameterName,
130 kIconForSearchButtonParameterColorful}};
131const FeatureEntry::FeatureParam kIconForSearchButtonMagnifying[] = {
132 {kIconForSearchButtonFeatureParameterName,
133 kIconForSearchButtonParameterMagnifying}};
134
135const FeatureEntry::FeatureVariation kIconForSearchButtonVariations[] = {
136 {"Grey search engine logo", kIconForSearchButtonGrey,
137 base::size(kIconForSearchButtonGrey), nullptr},
138 {"Colorful search engine logo", kIconForSearchButtonColorful,
139 base::size(kIconForSearchButtonColorful), nullptr},
140 {"Magnifying glass", kIconForSearchButtonMagnifying,
141 base::size(kIconForSearchButtonMagnifying), nullptr}};
142
Olivier Robince9386ebd2018-11-29 09:07:38143const FeatureEntry::FeatureParam kDetectMainThreadFreezeTimeout3s[] = {
144 {crash_report::kDetectMainThreadFreezeParameterName,
145 crash_report::kDetectMainThreadFreezeParameter3s}};
146const FeatureEntry::FeatureParam kDetectMainThreadFreezeTimeout5s[] = {
147 {crash_report::kDetectMainThreadFreezeParameterName,
148 crash_report::kDetectMainThreadFreezeParameter5s}};
149const FeatureEntry::FeatureParam kDetectMainThreadFreezeTimeout7s[] = {
150 {crash_report::kDetectMainThreadFreezeParameterName,
151 crash_report::kDetectMainThreadFreezeParameter7s}};
152const FeatureEntry::FeatureParam kDetectMainThreadFreezeTimeout9s[] = {
153 {crash_report::kDetectMainThreadFreezeParameterName,
154 crash_report::kDetectMainThreadFreezeParameter9s}};
155
156const FeatureEntry::FeatureVariation kDetectMainThreadFreezeVariations[] = {
157 {"3s", kDetectMainThreadFreezeTimeout3s,
158 base::size(kDetectMainThreadFreezeTimeout3s), nullptr},
159 {"5s", kDetectMainThreadFreezeTimeout5s,
160 base::size(kDetectMainThreadFreezeTimeout5s), nullptr},
161 {"7s", kDetectMainThreadFreezeTimeout7s,
162 base::size(kDetectMainThreadFreezeTimeout7s), nullptr},
163 {"9s", kDetectMainThreadFreezeTimeout9s,
164 base::size(kDetectMainThreadFreezeTimeout9s), nullptr}};
165
Cooper Knaak1e026562017-07-26 05:22:28166// To add a new entry, add to the end of kFeatureEntries. There are four
sdefresne14900ee2015-11-27 14:43:21167// distinct types of entries:
Cooper Knaak1e026562017-07-26 05:22:28168// . ENABLE_DISABLE_VALUE: entry is either enabled, disabled, or uses the
169// default value for this feature. Use the ENABLE_DISABLE_VALUE_TYPE
sdefresne14900ee2015-11-27 14:43:21170// macro for this type supplying the command line to the macro.
171// . MULTI_VALUE: a list of choices, the first of which should correspond to a
172// deactivated state for this lab (i.e. no command line option). To specify
173// this type of entry use the macro MULTI_VALUE_TYPE supplying it the
174// array of choices.
Cooper Knaak1e026562017-07-26 05:22:28175// . FEATURE_VALUE: entry is associated with a base::Feature instance. Entry is
176// either enabled, disabled, or uses the default value of the associated
177// base::Feature instance. To specify this type of entry use the macro
178// FEATURE_VALUE_TYPE supplying it the base::Feature instance.
179// . FEATURE_WITH_PARAM_VALUES: a list of choices associated with a
180// base::Feature instance. Choices corresponding to the default state, a
181// universally enabled state, and a universally disabled state are
182// automatically included. To specify this type of entry use the macro
183// FEATURE_WITH_PARAMS_VALUE_TYPE supplying it the base::Feature instance and
184// the array of choices.
185//
sdefresne14900ee2015-11-27 14:43:21186// See the documentation of FeatureEntry for details on the fields.
187//
188// When adding a new choice, add it to the end of the list.
189const flags_ui::FeatureEntry kFeatureEntries[] = {
Emily Starkbafa9062017-12-27 15:22:46190 {"enable-mark-http-as", flag_descriptions::kMarkHttpAsName,
elawrence816f6790e2017-06-16 18:19:28191 flag_descriptions::kMarkHttpAsDescription, flags_ui::kOsIos,
Emily Starkbafa9062017-12-27 15:22:46192 FEATURE_WITH_PARAMS_VALUE_TYPE(
193 security_state::features::kMarkHttpAsFeature,
194 kMarkHttpAsFeatureVariations,
195 "MarkHttpAs")},
Cooper Knaak1e026562017-07-26 05:22:28196 {"in-product-help-demo-mode-choice",
197 flag_descriptions::kInProductHelpDemoModeName,
198 flag_descriptions::kInProductHelpDemoModeDescription, flags_ui::kOsIos,
199 FEATURE_WITH_PARAMS_VALUE_TYPE(
Tommy Nyquistc1d6dea12017-07-26 20:37:23200 feature_engagement::kIPHDemoMode,
201 feature_engagement::kIPHDemoModeChoiceVariations,
Marc Treib2752e8b2017-08-04 14:12:09202 "IPH_DemoMode")},
Moe Ahmadi2d4c8452018-11-30 04:44:26203 {"preview-usdz", flag_descriptions::kUsdzPreviewName,
204 flag_descriptions::kUsdzPreviewDescription, flags_ui::kOsIos,
205 FEATURE_VALUE_TYPE(download::kUsdzPreview)},
Marc Treib2752e8b2017-08-04 14:12:09206 {"use-ddljson-api", flag_descriptions::kUseDdljsonApiName,
207 flag_descriptions::kUseDdljsonApiDescription, flags_ui::kOsIos,
Marc Treib5fc99e1f2017-11-08 12:19:18208 MULTI_VALUE_TYPE(kUseDdljsonApiChoices)},
Jean-François Geyelin30be5912017-08-25 12:05:38209 {"drag_and_drop", flag_descriptions::kDragAndDropName,
210 flag_descriptions::kDragAndDropDescription, flags_ui::kOsIos,
Rohit Rao6a3a481d2017-09-25 19:31:41211 FEATURE_VALUE_TYPE(kDragAndDrop)},
Eugene But468b6d5f02018-10-18 21:17:45212 {"ignores-viewport-scale-limits",
213 flag_descriptions::kIgnoresViewportScaleLimitsName,
214 flag_descriptions::kIgnoresViewportScaleLimitsDescription,
215 flags_ui::kOsIos,
216 FEATURE_VALUE_TYPE(web::features::kIgnoresViewportScaleLimits)},
Danyao Wang34da6ff2017-10-30 15:16:56217 {"slim-navigation-manager", flag_descriptions::kSlimNavigationManagerName,
218 flag_descriptions::kSlimNavigationManagerDescription, flags_ui::kOsIos,
219 FEATURE_VALUE_TYPE(web::features::kSlimNavigationManager)},
sczsdc2e8e02018-01-09 18:39:32220 {"memex-tab-switcher", flag_descriptions::kMemexTabSwitcherName,
221 flag_descriptions::kMemexTabSwitcherDescription, flags_ui::kOsIos,
222 FEATURE_VALUE_TYPE(kMemexTabSwitcher)},
mrefaata6501d92017-12-11 19:50:50223 {"wk-http-system-cookie-store",
224 flag_descriptions::kWKHTTPSystemCookieStoreName,
225 flag_descriptions::kWKHTTPSystemCookieStoreName, flags_ui::kOsIos,
Mohamad Ahmadic4df81f2017-12-20 04:41:59226 FEATURE_VALUE_TYPE(web::features::kWKHTTPSystemCookieStore)},
Moe Ahmadic3fd7cd2018-05-11 21:40:22227 {"enable-autofill-credit-card-upload",
228 flag_descriptions::kAutofillCreditCardUploadName,
229 flag_descriptions::kAutofillCreditCardUploadDescription, flags_ui::kOsIos,
Anne Lim579b5732018-08-30 18:24:24230 FEATURE_VALUE_TYPE(autofill::features::kAutofillUpstream)},
Lakshmi Kumar Dabbiruee22004e2018-06-01 01:52:27231 {"enable-autofill-credit-card-downstream-google-pay-branding",
232 flag_descriptions::kAutofillDownstreamUseGooglePayBrandingOniOSName,
233 flag_descriptions::kAutofillDownstreamUseGooglePayBrandingOniOSDescription,
234 flags_ui::kOsIos,
235 FEATURE_VALUE_TYPE(
236 autofill::features::kAutofillDownstreamUseGooglePayBrandingOniOS)},
Jared Saul2e55e7f12018-10-26 20:44:43237 {"enable-autofill-save-credit-card-uses-strike-system",
238 flag_descriptions::kEnableAutofillSaveCreditCardUsesStrikeSystemName,
239 flag_descriptions::
240 kEnableAutofillSaveCreditCardUsesStrikeSystemDescription,
241 flags_ui::kOsIos,
242 FEATURE_VALUE_TYPE(
243 autofill::features::kAutofillSaveCreditCardUsesStrikeSystem)},
Anne Lim9d8f66a2019-01-07 23:49:21244 {"enable-autofill-save-credit-card-uses-strike-system-v2",
245 flag_descriptions::kEnableAutofillSaveCreditCardUsesStrikeSystemV2Name,
246 flag_descriptions::
247 kEnableAutofillSaveCreditCardUsesStrikeSystemV2Description,
248 flags_ui::kOsIos,
249 FEATURE_VALUE_TYPE(
250 autofill::features::kAutofillSaveCreditCardUsesStrikeSystemV2)},
Moe Ahmadid6d5d172018-06-20 17:20:23251 {"use-sync-sandbox", flag_descriptions::kSyncSandboxName,
252 flag_descriptions::kSyncSandboxDescription, flags_ui::kOsIos,
253 SINGLE_VALUE_TYPE_AND_VALUE(
254 switches::kSyncServiceURL,
255 "https://chrome-sync.sandbox.google.com/chrome-sync/alpha")},
256 {"wallet-service-use-sandbox",
257 flag_descriptions::kWalletServiceUseSandboxName,
258 flag_descriptions::kWalletServiceUseSandboxDescription, flags_ui::kOsIos,
259 ENABLE_DISABLE_VALUE_TYPE_AND_VALUE(
260 autofill::switches::kWalletServiceUseSandbox,
261 "1",
262 autofill::switches::kWalletServiceUseSandbox,
263 "0")},
Mohamad Ahmadic4df81f2017-12-20 04:41:59264 {"show-autofill-type-predictions",
265 flag_descriptions::kShowAutofillTypePredictionsName,
266 flag_descriptions::kShowAutofillTypePredictionsDescription,
267 flags_ui::kOsIos,
Gauthier Ambardcfd85182018-01-05 10:37:46268 FEATURE_VALUE_TYPE(autofill::features::kAutofillShowTypePredictions)},
Olivier Robin3d60411622018-02-23 10:03:22269 {"autofill-ios-delay-between-fields",
270 flag_descriptions::kAutofillIOSDelayBetweenFieldsName,
271 flag_descriptions::kAutofillIOSDelayBetweenFieldsDescription,
272 flags_ui::kOsIos, MULTI_VALUE_TYPE(kAutofillIOSDelayBetweenFieldsChoices)},
Sylvain Defresneae6987e2018-03-01 13:21:40273 {"new-clear-browsing-data-ui",
274 flag_descriptions::kNewClearBrowsingDataUIName,
275 flag_descriptions::kNewClearBrowsingDataUIDescription, flags_ui::kOsIos,
276 FEATURE_VALUE_TYPE(kNewClearBrowsingDataUI)},
mrefaat390f99c0d2018-05-10 19:44:24277 {"itunes-urls-store-kit-handling",
278 flag_descriptions::kITunesUrlsStoreKitHandlingName,
279 flag_descriptions::kITunesUrlsStoreKitHandlingDescription,
280 flags_ui::kOsIos, FEATURE_VALUE_TYPE(kITunesUrlsStoreKitHandling)},
Jérôme Lebelfa6c5ae2018-03-29 13:46:57281 {"unified-consent", flag_descriptions::kUnifiedConsentName,
282 flag_descriptions::kUnifiedConsentDescription, flags_ui::kOsIos,
Mihai Sardarescu160ec662018-07-18 21:13:51283 FEATURE_VALUE_TYPE(unified_consent::kUnifiedConsent)},
sebsg93dc01c2018-04-05 16:42:48284 {"autofill-dynamic-forms", flag_descriptions::kAutofillDynamicFormsName,
285 flag_descriptions::kAutofillDynamicFormsDescription, flags_ui::kOsIos,
286 FEATURE_VALUE_TYPE(autofill::features::kAutofillDynamicForms)},
Olivier Robin081912dd2018-06-11 09:14:26287 {"autofill-prefilled-fields",
288 flag_descriptions::kAutofillPrefilledFieldsName,
289 flag_descriptions::kAutofillPrefilledFieldsDescription, flags_ui::kOsIos,
290 FEATURE_VALUE_TYPE(autofill::features::kAutofillPrefilledFields)},
Olivier Robind81475b02018-06-20 21:30:11291 {"autofill-show-all-profiles-on-prefilled-forms",
292 flag_descriptions::kAutofillShowAllSuggestionsOnPrefilledFormsName,
293 flag_descriptions::kAutofillShowAllSuggestionsOnPrefilledFormsDescription,
294 flags_ui::kOsIos,
295 FEATURE_VALUE_TYPE(
296 autofill::features::kAutofillShowAllSuggestionsOnPrefilledForms)},
Moe Ahmadid3ddfcb2018-04-20 19:51:08297 {"autofill-restrict-formless-form-extraction",
298 flag_descriptions::kAutofillRestrictUnownedFieldsToFormlessCheckoutName,
299 flag_descriptions::
300 kAutofillRestrictUnownedFieldsToFormlessCheckoutDescription,
301 flags_ui::kOsIos,
302 FEATURE_VALUE_TYPE(
303 autofill::features::kAutofillRestrictUnownedFieldsToFormlessCheckout)},
Roger McFarlanee193ab122018-11-22 20:55:07304 {"autofill-rich-metadata-queries",
305 flag_descriptions::kAutofillRichMetadataQueriesName,
306 flag_descriptions::kAutofillRichMetadataQueriesDescription,
307 flags_ui::kOsIos,
308 FEATURE_VALUE_TYPE(autofill::features::kAutofillRichMetadataQueries)},
Kurt Horimotodc33af32018-05-01 01:39:14309 {"fullscreen-viewport-adjustment-experiment",
310 flag_descriptions::kFullscreenViewportAdjustmentExperimentName,
311 flag_descriptions::kFullscreenViewportAdjustmentExperimentDescription,
312 flags_ui::kOsIos,
313 MULTI_VALUE_TYPE(
314 fullscreen::features::kViewportAdjustmentExperimentChoices)},
Roger McFarlane1a37d7b2018-04-20 22:45:19315 {"autofill-enforce-min-required-fields-for-heuristics",
316 flag_descriptions::kAutofillEnforceMinRequiredFieldsForHeuristicsName,
317 flag_descriptions::
318 kAutofillEnforceMinRequiredFieldsForHeuristicsDescription,
319 flags_ui::kOsIos,
320 FEATURE_VALUE_TYPE(
321 autofill::features::kAutofillEnforceMinRequiredFieldsForHeuristics)},
322 {"autofill-enforce-min-required-fields-for-query",
323 flag_descriptions::kAutofillEnforceMinRequiredFieldsForQueryName,
324 flag_descriptions::kAutofillEnforceMinRequiredFieldsForQueryDescription,
325 flags_ui::kOsIos,
326 FEATURE_VALUE_TYPE(
327 autofill::features::kAutofillEnforceMinRequiredFieldsForQuery)},
328 {"autofill-enforce-min-required-fields-for-upload",
329 flag_descriptions::kAutofillEnforceMinRequiredFieldsForUploadName,
330 flag_descriptions::kAutofillEnforceMinRequiredFieldsForUploadDescription,
331 flags_ui::kOsIos,
332 FEATURE_VALUE_TYPE(
333 autofill::features::kAutofillEnforceMinRequiredFieldsForUpload)},
Gauthier Ambard4417f852018-04-26 14:24:32334 {"browser-container-fullscreen",
335 flag_descriptions::kBrowserContainerFullscreenName,
336 flag_descriptions::kBrowserContainerFullscreenDescription,
Gauthier Ambard5d7627282018-05-18 08:20:07337 flags_ui::kOsIos,
338 FEATURE_VALUE_TYPE(web::features::kBrowserContainerFullscreen)},
Roger McFarlane2eceba982018-05-18 01:37:11339 {"autofill-cache-query-responses",
340 flag_descriptions::kAutofillCacheQueryResponsesName,
341 flag_descriptions::kAutofillCacheQueryResponsesDescription,
342 flags_ui::kOsIos,
343 FEATURE_VALUE_TYPE(autofill::features::kAutofillCacheQueryResponses)},
Jeffrey Cohen05cdafa392018-08-15 20:49:17344 {"autofill-enable-company-name",
345 flag_descriptions::kAutofillEnableCompanyNameName,
346 flag_descriptions::kAutofillEnableCompanyNameDescription, flags_ui::kOsIos,
347 FEATURE_VALUE_TYPE(autofill::features::kAutofillEnableCompanyName)},
Javier Ernesto Flores Robles23c71a7d2018-06-01 09:51:17348 {"autofill-manual-fallback", flag_descriptions::kAutofillManualFallbackName,
349 flag_descriptions::kAutofillManualFallbackDescription, flags_ui::kOsIos,
350 FEATURE_VALUE_TYPE(autofill::features::kAutofillManualFallback)},
Yi Su392b3032018-06-05 07:26:47351 {"webpage-text-accessibility",
352 flag_descriptions::kWebPageTextAccessibilityName,
353 flag_descriptions::kWebPageTextAccessibilityDescription, flags_ui::kOsIos,
354 FEATURE_VALUE_TYPE(web::kWebPageTextAccessibility)},
Mike Dougherty39d36b02018-06-19 19:21:40355 {"web-frame-messaging", flag_descriptions::kWebFrameMessagingName,
356 flag_descriptions::kWebFrameMessagingDescription, flags_ui::kOsIos,
357 FEATURE_VALUE_TYPE(web::features::kWebFrameMessaging)},
Vadym Doroshenkoa9a914942018-07-19 08:23:30358 {"new-password-form-parsing",
359 flag_descriptions::kNewPasswordFormParsingName,
360 flag_descriptions::kNewPasswordFormParsingDescription, flags_ui::kOsIos,
361 FEATURE_VALUE_TYPE(password_manager::features::kNewPasswordFormParsing)},
mrefaat4539b032018-08-02 22:25:45362 {"app-launcher-refresh", flag_descriptions::kAppLauncherRefreshName,
363 flag_descriptions::kAppLauncherRefreshDescription, flags_ui::kOsIos,
364 FEATURE_VALUE_TYPE(kAppLauncherRefresh)},
Marc Treib6f0bbd22018-08-28 10:07:19365 {"sync-standalone-transport",
366 flag_descriptions::kSyncStandaloneTransportName,
367 flag_descriptions::kSyncStandaloneTransportDescription, flags_ui::kOsIos,
368 FEATURE_VALUE_TYPE(switches::kSyncStandaloneTransport)},
Marc Treib4bd95e72018-08-30 06:30:58369 {"sync-support-secondary-account",
370 flag_descriptions::kSyncSupportSecondaryAccountName,
371 flag_descriptions::kSyncSupportSecondaryAccountDescription,
372 flags_ui::kOsIos,
373 FEATURE_VALUE_TYPE(switches::kSyncSupportSecondaryAccount)},
Gauthier Ambard66de93d2018-08-30 15:58:19374 {"out-of-web-fullscreen", flag_descriptions::kOutOfWebFullscreenName,
375 flag_descriptions::kOutOfWebFullscreenDescription, flags_ui::kOsIos,
376 FEATURE_VALUE_TYPE(web::features::kOutOfWebFullscreen)},
Javier Ernesto Flores Robles8e66f732018-09-12 15:41:15377 {"autofill-manual-fallback-phase-two",
378 flag_descriptions::kAutofillManualFallbackPhaseTwoName,
379 flag_descriptions::kAutofillManualFallbackPhaseTwoDescription,
380 flags_ui::kOsIos,
381 FEATURE_VALUE_TYPE(autofill::features::kAutofillManualFallbackPhaseTwo)},
Kurt Horimoto79d590b2018-09-12 19:09:28382 {"toolbar-container", flag_descriptions::kToolbarContainerName,
383 flag_descriptions::kToolbarContainerDescription, flags_ui::kOsIos,
384 FEATURE_VALUE_TYPE(toolbar_container::kToolbarContainerEnabled)},
Eugene Butc61230ab2018-11-09 00:31:58385 {"present-sad-tab-in-view-controller",
386 flag_descriptions::kPresentSadTabInViewControllerName,
387 flag_descriptions::kPresentSadTabInViewControllerDescription,
388 flags_ui::kOsIos, FEATURE_VALUE_TYPE(kPresentSadTabInViewController)},
[email protected]5b899dc2018-10-02 09:20:01389 {"omnibox-popup-shortcuts",
390 flag_descriptions::kOmniboxPopupShortcutIconsInZeroStateName,
391 flag_descriptions::kOmniboxPopupShortcutIconsInZeroStateDescription,
392 flags_ui::kOsIos,
393 FEATURE_VALUE_TYPE(omnibox::kOmniboxPopupShortcutIconsInZeroState)},
Jérôme Lebelec282832018-09-18 08:34:18394 {"sso-with-wkwebview", flag_descriptions::kSSOWithWKWebViewName,
395 flag_descriptions::kSSOWithWKWebViewDescription, flags_ui::kOsIos,
396 FEATURE_VALUE_TYPE(kSSOWithWKWebView)},
Yi Sua3d55f3d2018-09-24 17:49:41397 {"custom-search-engines", flag_descriptions::kCustomSearchEnginesName,
398 flag_descriptions::kCustomSearchEnginesDescription, flags_ui::kOsIos,
399 FEATURE_VALUE_TYPE(kCustomSearchEngines)},
Valeriya Sinevicha79dc612018-09-25 14:49:09400 {"use-multilogin-endpoint", flag_descriptions::kUseMultiloginEndpointName,
401 flag_descriptions::kUseMultiloginEndpointDescription, flags_ui::kOsIos,
402 FEATURE_VALUE_TYPE(kUseMultiloginEndpoint)},
edchin18a3e602018-10-08 23:39:35403 {"closing-last-incognito-tab",
404 flag_descriptions::kClosingLastIncognitoTabName,
405 flag_descriptions::kClosingLastIncognitoTabDescription, flags_ui::kOsIos,
406 FEATURE_VALUE_TYPE(kClosingLastIncognitoTab)},
Gauthier Ambardce7fd582018-10-09 08:37:05407 {"omnibox-tab-switch-suggestions",
408 flag_descriptions::kOmniboxTabSwitchSuggestionsName,
409 flag_descriptions::kOmniboxTabSwitchSuggestionsDescription,
410 flags_ui::kOsIos,
411 FEATURE_VALUE_TYPE(omnibox::kOmniboxTabSwitchSuggestions)},
Tanja Gornakb0985dd2018-10-11 17:24:40412 {"fcm-invalidations", flag_descriptions::kFCMInvalidationsName,
413 flag_descriptions::kFCMInvalidationsDescription, flags_ui::kOsIos,
414 FEATURE_VALUE_TYPE(invalidation::switches::kFCMInvalidations)},
Justin Cohen9a5023212018-10-15 14:32:37415 {"browser-container-contains-ntp",
416 flag_descriptions::kBrowserContainerContainsNTPName,
417 flag_descriptions::kBrowserContainerContainsNTPDescription,
418 flags_ui::kOsIos, FEATURE_VALUE_TYPE(kBrowserContainerContainsNTP)},
Eugene But66e917002019-01-07 23:38:52419 {"external-files-loaded-in-web-state",
420 flag_descriptions::kExternalFilesLoadedInWebStateName,
421 flag_descriptions::kExternalFilesLoadedInWebStateDescription,
422 flags_ui::kOsIos,
423 FEATURE_VALUE_TYPE(experimental_flags::kExternalFilesLoadedInWebState)},
Gauthier Ambard2392f2a2018-10-24 16:33:27424 {"search-icon-toggle", flag_descriptions::kSearchIconToggleName,
425 flag_descriptions::kSearchIconToggleDescription, flags_ui::kOsIos,
426 FEATURE_WITH_PARAMS_VALUE_TYPE(kIconForSearchButtonFeature,
427 kIconForSearchButtonVariations,
428 "ToggleSearchButtonIcon")},
Olivier Robin2a898dc2018-10-31 14:37:04429 {"enable-breakpad-upload-no-delay",
430 flag_descriptions::kBreakpadNoDelayInitialUploadName,
431 flag_descriptions::kBreakpadNoDelayInitialUploadDescription,
432 flags_ui::kOsIos,
433 FEATURE_VALUE_TYPE(crash_report::kBreakpadNoDelayInitialUpload)},
Moe Ahmadie4cce1f2018-12-18 23:48:01434 {"enable-compact-translate-infobar",
435 flag_descriptions::kCompactTranslateInfobarName,
436 flag_descriptions::kCompactTranslateInfobarDescription, flags_ui::kOsIos,
437 FEATURE_VALUE_TYPE(translate::kCompactTranslateInfobarIOS)},
Kurt Horimoto9165404a2018-11-13 23:48:12438 {"non-modal-dialogs", flag_descriptions::kNonModalDialogsName,
439 flag_descriptions::kNonModalDialogsDescription, flags_ui::kOsIos,
440 FEATURE_VALUE_TYPE(dialogs::kNonModalDialogs)},
Mikel Astizdb43c5652018-11-22 16:52:05441 {"sync-pseudo-uss-favicons", flag_descriptions::kSyncPseudoUSSFaviconsName,
442 flag_descriptions::kSyncPseudoUSSFaviconsDescription, flags_ui::kOsIos,
443 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSFavicons)},
444 {"sync-pseudo-uss-history-delete-directives",
445 flag_descriptions::kSyncPseudoUSSHistoryDeleteDirectivesName,
446 flag_descriptions::kSyncPseudoUSSHistoryDeleteDirectivesDescription,
447 flags_ui::kOsIos,
448 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSHistoryDeleteDirectives)},
449 {"sync-pseudo-uss-passwords",
450 flag_descriptions::kSyncPseudoUSSPasswordsName,
451 flag_descriptions::kSyncPseudoUSSPasswordsDescription, flags_ui::kOsIos,
452 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSPasswords)},
453 {"sync-pseudo-uss-preferences",
454 flag_descriptions::kSyncPseudoUSSPreferencesName,
455 flag_descriptions::kSyncPseudoUSSPreferencesDescription, flags_ui::kOsIos,
456 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSPreferences)},
457 {"sync-pseudo-uss-priority-preferences",
458 flag_descriptions::kSyncPseudoUSSPriorityPreferencesName,
459 flag_descriptions::kSyncPseudoUSSPriorityPreferencesDescription,
460 flags_ui::kOsIos,
461 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSPriorityPreferences)},
Jérôme Lebel5717fa852018-12-19 12:11:13462 {"use-nsurlsession-for-signin",
463 flag_descriptions::kUseNSURLSessionForGaiaSigninRequestsName,
464 flag_descriptions::kUseNSURLSessionForGaiaSigninRequestsDescription,
465 flags_ui::kOsIos,
466 FEATURE_VALUE_TYPE(kUseNSURLSessionForGaiaSigninRequests)},
Mikel Astizdb43c5652018-11-22 16:52:05467 {"sync-pseudo-uss-supervised-users",
468 flag_descriptions::kSyncPseudoUSSSupervisedUsersName,
469 flag_descriptions::kSyncPseudoUSSSupervisedUsersDescription,
470 flags_ui::kOsIos,
471 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSSupervisedUsers)},
Olivier Robince9386ebd2018-11-29 09:07:38472 {"detect-main-thread-freeze",
473 flag_descriptions::kDetectMainThreadFreezeName,
474 flag_descriptions::kDetectMainThreadFreezeDescription, flags_ui::kOsIos,
475 FEATURE_WITH_PARAMS_VALUE_TYPE(crash_report::kDetectMainThreadFreeze,
476 kDetectMainThreadFreezeVariations,
477 "DetectMainThreadFreeze")},
Mohamed Amir Yosef9e8a1f02018-12-03 21:41:14478 {"enable-sync-uss-bookmarks",
479 flag_descriptions::kEnableSyncUSSBookmarksName,
480 flag_descriptions::kEnableSyncUSSBookmarksDescription, flags_ui::kOsIos,
481 FEATURE_VALUE_TYPE(switches::kSyncUSSBookmarks)},
sczs80a6f462018-12-05 20:37:18482 {"infobar-ui-reboot", flag_descriptions::kInfobarUIRebootName,
483 flag_descriptions::kInfobarUIRebootDescription, flags_ui::kOsIos,
484 FEATURE_VALUE_TYPE(kInfobarUIReboot)},
Chris Lu77905ac2018-12-06 20:46:51485 {"find-in-page-iframe", flag_descriptions::kFindInPageiFrameName,
486 flag_descriptions::kFindInPageiFrameDescription, flags_ui::kOsIos,
487 FEATURE_VALUE_TYPE(kFindInPageiFrame)},
Robbie Gibson80d732e2019-01-10 10:42:49488 {"enable-clipboard-provider-text-suggestions",
489 flag_descriptions::kEnableClipboardProviderTextSuggestionsName,
490 flag_descriptions::kEnableClipboardProviderTextSuggestionsDescription,
491 flags_ui::kOsIos,
492 FEATURE_VALUE_TYPE(omnibox::kEnableClipboardProviderTextSuggestions)},
493 {"enable-clipboard-provider-image-suggestions",
494 flag_descriptions::kEnableClipboardProviderImageSuggestionsName,
495 flag_descriptions::kEnableClipboardProviderImageSuggestionsDescription,
496 flags_ui::kOsIos,
497 FEATURE_VALUE_TYPE(omnibox::kEnableClipboardProviderImageSuggestions)},
498 {"copied-content-behavior", flag_descriptions::kCopiedContentBehaviorName,
499 flag_descriptions::kCopiedContentBehaviorName, flags_ui::kOsIos,
500 FEATURE_VALUE_TYPE(kCopiedContentBehavior)},
Vadym Doroshenkod178ff3f2019-01-02 14:59:23501 {"new-password-form-parsing-for-saving",
502 flag_descriptions::kNewPasswordFormParsingForSavingName,
503 flag_descriptions::kNewPasswordFormParsingForSavingDescription,
504 flags_ui::kOsIos,
505 FEATURE_VALUE_TYPE(
506 password_manager::features::kNewPasswordFormParsingForSaving)},
507 {"only-new-password-form-parsing",
508 flag_descriptions::kOnlyNewPasswordFormParsingName,
509 flag_descriptions::kOnlyNewPasswordFormParsingDescription,
510 flags_ui::kOsIos,
511 FEATURE_VALUE_TYPE(password_manager::features::kOnlyNewParser)},
edchin5ae0b4c2019-01-06 06:52:28512 {"snapshot-draw-view", flag_descriptions::kSnapshotDrawViewName,
513 flag_descriptions::kSnapshotDrawViewDescription, flags_ui::kOsIos,
514 FEATURE_VALUE_TYPE(kSnapshotDrawView)},
Sebastien Lalancette5daaf4a2019-01-11 22:09:25515 {"enable-autocomplete-data-retention-policy",
516 flag_descriptions::kEnableAutocompleteDataRetentionPolicyName,
517 flag_descriptions::kEnableAutocompleteDataRetentionPolicyDescription,
518 flags_ui::kOsIos,
519 FEATURE_VALUE_TYPE(
520 autofill::features::kAutocompleteRetentionPolicyEnabled)},
Gauthier Ambardcfd85182018-01-05 10:37:46521};
sdefresne14900ee2015-11-27 14:43:21522
523// Add all switches from experimental flags to |command_line|.
524void AppendSwitchesFromExperimentalSettings(base::CommandLine* command_line) {
525 NSUserDefaults* defaults = [NSUserDefaults standardUserDefaults];
526
sdefresne14900ee2015-11-27 14:43:21527 // Set the UA flag if UseMobileSafariUA is enabled.
528 if ([defaults boolForKey:@"UseMobileSafariUA"]) {
529 // Safari uses "Vesion/", followed by the OS version excluding bugfix, where
530 // Chrome puts its product token.
avi571943672015-12-22 02:12:49531 int32_t major = 0;
532 int32_t minor = 0;
533 int32_t bugfix = 0;
sdefresne14900ee2015-11-27 14:43:21534 base::SysInfo::OperatingSystemVersionNumbers(&major, &minor, &bugfix);
535 std::string product = base::StringPrintf("Version/%d.%d", major, minor);
536
537 command_line->AppendSwitchASCII(switches::kUserAgent,
538 web::BuildUserAgentFromProduct(product));
539 }
540
541 // Freeform commandline flags. These are added last, so that any flags added
542 // earlier in this function take precedence.
543 if ([defaults boolForKey:@"EnableFreeformCommandLineFlags"]) {
544 base::CommandLine::StringVector flags;
545 // Append an empty "program" argument.
546 flags.push_back("");
547
548 // The number of flags corresponds to the number of text fields in
549 // Experimental.plist.
550 const int kNumFreeformFlags = 5;
551 for (int i = 1; i <= kNumFreeformFlags; ++i) {
552 NSString* key =
553 [NSString stringWithFormat:@"FreeformCommandLineFlag%d", i];
554 NSString* flag = [defaults stringForKey:key];
555 if ([flag length]) {
556 flags.push_back(base::SysNSStringToUTF8(flag));
557 }
558 }
559
560 base::CommandLine temp_command_line(flags);
561 command_line->AppendArguments(temp_command_line, false);
562 }
msardafc76f662017-02-24 12:46:28563
justincohendacc85d2017-06-28 23:34:10564 // Populate command line flag for 3rd party keyboard omnibox workaround.
565 NSString* enableThirdPartyKeyboardWorkaround =
566 [defaults stringForKey:@"EnableThirdPartyKeyboardWorkaround"];
567 if ([enableThirdPartyKeyboardWorkaround isEqualToString:@"Enabled"]) {
568 command_line->AppendSwitch(switches::kEnableThirdPartyKeyboardWorkaround);
569 } else if ([enableThirdPartyKeyboardWorkaround isEqualToString:@"Disabled"]) {
570 command_line->AppendSwitch(switches::kDisableThirdPartyKeyboardWorkaround);
571 }
572
msardafc76f662017-02-24 12:46:28573 ios::GetChromeBrowserProvider()->AppendSwitchesFromExperimentalSettings(
574 defaults, command_line);
sdefresne14900ee2015-11-27 14:43:21575}
576
577bool SkipConditionalFeatureEntry(const flags_ui::FeatureEntry& entry) {
578 return false;
579}
580
581class FlagsStateSingleton {
582 public:
583 FlagsStateSingleton()
kokihoon189a4022018-10-13 02:43:44584 : flags_state_(kFeatureEntries, base::size(kFeatureEntries)) {}
sdefresne14900ee2015-11-27 14:43:21585 ~FlagsStateSingleton() {}
586
587 static FlagsStateSingleton* GetInstance() {
588 return base::Singleton<FlagsStateSingleton>::get();
589 }
590
591 static flags_ui::FlagsState* GetFlagsState() {
592 return &GetInstance()->flags_state_;
593 }
594
595 private:
596 flags_ui::FlagsState flags_state_;
597
598 DISALLOW_COPY_AND_ASSIGN(FlagsStateSingleton);
599};
600} // namespace
601
602void ConvertFlagsToSwitches(flags_ui::FlagsStorage* flags_storage,
603 base::CommandLine* command_line) {
Gregory Chatzinoff31305112017-12-20 21:56:02604 AppendSwitchesFromExperimentalSettings(command_line);
sdefresne14900ee2015-11-27 14:43:21605 FlagsStateSingleton::GetFlagsState()->ConvertFlagsToSwitches(
sdefresnec9763902015-12-02 10:30:11606 flags_storage, command_line, flags_ui::kAddSentinels,
Gregory Chatzinofff6523722017-11-21 01:33:53607 switches::kEnableFeatures, switches::kDisableFeatures);
sdefresne14900ee2015-11-27 14:43:21608}
609
jkrcalbf073372016-07-29 07:21:31610std::vector<std::string> RegisterAllFeatureVariationParameters(
611 flags_ui::FlagsStorage* flags_storage,
612 base::FeatureList* feature_list) {
613 return FlagsStateSingleton::GetFlagsState()
614 ->RegisterAllFeatureVariationParameters(flags_storage, feature_list);
615}
616
sdefresne14900ee2015-11-27 14:43:21617void GetFlagFeatureEntries(flags_ui::FlagsStorage* flags_storage,
618 flags_ui::FlagAccess access,
619 base::ListValue* supported_entries,
620 base::ListValue* unsupported_entries) {
621 FlagsStateSingleton::GetFlagsState()->GetFlagFeatureEntries(
622 flags_storage, access, supported_entries, unsupported_entries,
623 base::Bind(&SkipConditionalFeatureEntry));
624}
625
626void SetFeatureEntryEnabled(flags_ui::FlagsStorage* flags_storage,
627 const std::string& internal_name,
628 bool enable) {
629 FlagsStateSingleton::GetFlagsState()->SetFeatureEntryEnabled(
630 flags_storage, internal_name, enable);
631}
632
633void ResetAllFlags(flags_ui::FlagsStorage* flags_storage) {
634 FlagsStateSingleton::GetFlagsState()->ResetAllFlags(flags_storage);
635}
636
637namespace testing {
638
639const flags_ui::FeatureEntry* GetFeatureEntries(size_t* count) {
kokihoon189a4022018-10-13 02:43:44640 *count = base::size(kFeatureEntries);
sdefresne14900ee2015-11-27 14:43:21641 return kFeatureEntries;
642}
643
644} // namespace testing