blob: ae12d94fabea0cee68758970d6744d647416f2b8 [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"
Mike Doughertya5e5ad52017-08-05 00:11:5858#include "ios/chrome/browser/ssl/captive_portal_features.h"
Kurt Horimoto9165404a2018-11-13 23:48:1259#import "ios/chrome/browser/ui/dialogs/dialog_features.h"
Kurt Horimoto4da682b022018-04-12 07:43:3560#import "ios/chrome/browser/ui/fullscreen/fullscreen_features.h"
sczs80a6f462018-12-05 20:37:1861#import "ios/chrome/browser/ui/infobars/infobar_feature.h"
Eugene Butc61230ab2018-11-09 00:31:5862#include "ios/chrome/browser/ui/sad_tab/features.h"
Mark Cogan02ed6c772018-05-31 17:00:5163#import "ios/chrome/browser/ui/toolbar/public/features.h"
Kurt Horimoto79d590b2018-09-12 19:09:2864#import "ios/chrome/browser/ui/toolbar_container/toolbar_container_features.h"
Justin Cohen2d81c582018-01-22 14:46:4465#include "ios/chrome/browser/ui/ui_feature_flags.h"
Yi Su392b3032018-06-05 07:26:4766#include "ios/chrome/browser/web/features.h"
sdefresne14900ee2015-11-27 14:43:2167#include "ios/chrome/grit/ios_strings.h"
msardafc76f662017-02-24 12:46:2868#include "ios/public/provider/chrome/browser/chrome_browser_provider.h"
Danyao Wang34da6ff2017-10-30 15:16:5669#include "ios/web/public/features.h"
sdefresne14900ee2015-11-27 14:43:2170#include "ios/web/public/user_agent.h"
71#include "ios/web/public/web_view_creation_util.h"
72
73#if !defined(OFFICIAL_BUILD)
74#include "components/variations/variations_switches.h"
vitaliii489217aa2017-01-30 14:50:2275#endif
stkhapuginc1be1792016-12-13 14:30:5376
77#if !defined(__has_feature) || !__has_feature(objc_arc)
78#error "This file requires ARC support."
79#endif
sdefresne14900ee2015-11-27 14:43:2180
elawrence816f6790e2017-06-16 18:19:2881using flags_ui::FeatureEntry;
82
sdefresne14900ee2015-11-27 14:43:2183namespace {
Emily Starkbafa9062017-12-27 15:22:4684
85const FeatureEntry::FeatureParam kMarkHttpAsDangerous[] = {
86 {security_state::features::kMarkHttpAsFeatureParameterName,
87 security_state::features::kMarkHttpAsParameterDangerous}};
Emily Starkbafa9062017-12-27 15:22:4688const FeatureEntry::FeatureParam kMarkHttpAsWarningAndDangerousOnFormEdits[] = {
89 {security_state::features::kMarkHttpAsFeatureParameterName,
90 security_state::features::
91 kMarkHttpAsParameterWarningAndDangerousOnFormEdits}};
Emily Starkbafa9062017-12-27 15:22:4692
93const FeatureEntry::FeatureVariation kMarkHttpAsFeatureVariations[] = {
94 {"(mark as actively dangerous)", kMarkHttpAsDangerous,
kokihoon189a4022018-10-13 02:43:4495 base::size(kMarkHttpAsDangerous), nullptr},
Emily Starkbafa9062017-12-27 15:22:4696 {"(mark with a Not Secure warning and dangerous on form edits)",
97 kMarkHttpAsWarningAndDangerousOnFormEdits,
Joe DeBlasio20e01fd2018-12-20 19:59:3098 base::size(kMarkHttpAsWarningAndDangerousOnFormEdits), nullptr}};
elawrence816f6790e2017-06-16 18:19:2899
Marc Treib5fc99e1f2017-11-08 12:19:18100const FeatureEntry::Choice kUseDdljsonApiChoices[] = {
101 {flags_ui::kGenericExperimentChoiceDefault, "", ""},
102 {"(force test doodle 0)", search_provider_logos::switches::kGoogleDoodleUrl,
103 "https://www.gstatic.com/chrome/ntp/doodle_test/ddljson_ios0.json"},
104 {"(force test doodle 1)", search_provider_logos::switches::kGoogleDoodleUrl,
105 "https://www.gstatic.com/chrome/ntp/doodle_test/ddljson_ios1.json"},
106 {"(force test doodle 2)", search_provider_logos::switches::kGoogleDoodleUrl,
107 "https://www.gstatic.com/chrome/ntp/doodle_test/ddljson_ios2.json"},
108 {"(force test doodle 3)", search_provider_logos::switches::kGoogleDoodleUrl,
109 "https://www.gstatic.com/chrome/ntp/doodle_test/ddljson_ios3.json"},
110 {"(force test doodle 4)", search_provider_logos::switches::kGoogleDoodleUrl,
111 "https://www.gstatic.com/chrome/ntp/doodle_test/ddljson_ios4.json"},
112};
Marc Treib2752e8b2017-08-04 14:12:09113
Olivier Robin3d60411622018-02-23 10:03:22114const FeatureEntry::Choice kAutofillIOSDelayBetweenFieldsChoices[] = {
115 {flags_ui::kGenericExperimentChoiceDefault, "", ""},
116 {"0", autofill::switches::kAutofillIOSDelayBetweenFields, "0"},
117 {"10", autofill::switches::kAutofillIOSDelayBetweenFields, "10"},
118 {"20", autofill::switches::kAutofillIOSDelayBetweenFields, "20"},
119 {"50", autofill::switches::kAutofillIOSDelayBetweenFields, "50"},
120 {"100", autofill::switches::kAutofillIOSDelayBetweenFields, "100"},
121 {"200", autofill::switches::kAutofillIOSDelayBetweenFields, "200"},
122 {"500", autofill::switches::kAutofillIOSDelayBetweenFields, "500"},
123 {"1000", autofill::switches::kAutofillIOSDelayBetweenFields, "1000"},
124};
125
Gauthier Ambard2392f2a2018-10-24 16:33:27126const FeatureEntry::FeatureParam kIconForSearchButtonGrey[] = {
127 {kIconForSearchButtonFeatureParameterName,
128 kIconForSearchButtonParameterGrey}};
129const FeatureEntry::FeatureParam kIconForSearchButtonColorful[] = {
130 {kIconForSearchButtonFeatureParameterName,
131 kIconForSearchButtonParameterColorful}};
132const FeatureEntry::FeatureParam kIconForSearchButtonMagnifying[] = {
133 {kIconForSearchButtonFeatureParameterName,
134 kIconForSearchButtonParameterMagnifying}};
135
136const FeatureEntry::FeatureVariation kIconForSearchButtonVariations[] = {
137 {"Grey search engine logo", kIconForSearchButtonGrey,
138 base::size(kIconForSearchButtonGrey), nullptr},
139 {"Colorful search engine logo", kIconForSearchButtonColorful,
140 base::size(kIconForSearchButtonColorful), nullptr},
141 {"Magnifying glass", kIconForSearchButtonMagnifying,
142 base::size(kIconForSearchButtonMagnifying), nullptr}};
143
Olivier Robince9386ebd2018-11-29 09:07:38144const FeatureEntry::FeatureParam kDetectMainThreadFreezeTimeout3s[] = {
145 {crash_report::kDetectMainThreadFreezeParameterName,
146 crash_report::kDetectMainThreadFreezeParameter3s}};
147const FeatureEntry::FeatureParam kDetectMainThreadFreezeTimeout5s[] = {
148 {crash_report::kDetectMainThreadFreezeParameterName,
149 crash_report::kDetectMainThreadFreezeParameter5s}};
150const FeatureEntry::FeatureParam kDetectMainThreadFreezeTimeout7s[] = {
151 {crash_report::kDetectMainThreadFreezeParameterName,
152 crash_report::kDetectMainThreadFreezeParameter7s}};
153const FeatureEntry::FeatureParam kDetectMainThreadFreezeTimeout9s[] = {
154 {crash_report::kDetectMainThreadFreezeParameterName,
155 crash_report::kDetectMainThreadFreezeParameter9s}};
156
157const FeatureEntry::FeatureVariation kDetectMainThreadFreezeVariations[] = {
158 {"3s", kDetectMainThreadFreezeTimeout3s,
159 base::size(kDetectMainThreadFreezeTimeout3s), nullptr},
160 {"5s", kDetectMainThreadFreezeTimeout5s,
161 base::size(kDetectMainThreadFreezeTimeout5s), nullptr},
162 {"7s", kDetectMainThreadFreezeTimeout7s,
163 base::size(kDetectMainThreadFreezeTimeout7s), nullptr},
164 {"9s", kDetectMainThreadFreezeTimeout9s,
165 base::size(kDetectMainThreadFreezeTimeout9s), nullptr}};
166
Cooper Knaak1e026562017-07-26 05:22:28167// To add a new entry, add to the end of kFeatureEntries. There are four
sdefresne14900ee2015-11-27 14:43:21168// distinct types of entries:
Cooper Knaak1e026562017-07-26 05:22:28169// . ENABLE_DISABLE_VALUE: entry is either enabled, disabled, or uses the
170// default value for this feature. Use the ENABLE_DISABLE_VALUE_TYPE
sdefresne14900ee2015-11-27 14:43:21171// macro for this type supplying the command line to the macro.
172// . MULTI_VALUE: a list of choices, the first of which should correspond to a
173// deactivated state for this lab (i.e. no command line option). To specify
174// this type of entry use the macro MULTI_VALUE_TYPE supplying it the
175// array of choices.
Cooper Knaak1e026562017-07-26 05:22:28176// . FEATURE_VALUE: entry is associated with a base::Feature instance. Entry is
177// either enabled, disabled, or uses the default value of the associated
178// base::Feature instance. To specify this type of entry use the macro
179// FEATURE_VALUE_TYPE supplying it the base::Feature instance.
180// . FEATURE_WITH_PARAM_VALUES: a list of choices associated with a
181// base::Feature instance. Choices corresponding to the default state, a
182// universally enabled state, and a universally disabled state are
183// automatically included. To specify this type of entry use the macro
184// FEATURE_WITH_PARAMS_VALUE_TYPE supplying it the base::Feature instance and
185// the array of choices.
186//
sdefresne14900ee2015-11-27 14:43:21187// See the documentation of FeatureEntry for details on the fields.
188//
189// When adding a new choice, add it to the end of the list.
190const flags_ui::FeatureEntry kFeatureEntries[] = {
Emily Starkbafa9062017-12-27 15:22:46191 {"enable-mark-http-as", flag_descriptions::kMarkHttpAsName,
elawrence816f6790e2017-06-16 18:19:28192 flag_descriptions::kMarkHttpAsDescription, flags_ui::kOsIos,
Emily Starkbafa9062017-12-27 15:22:46193 FEATURE_WITH_PARAMS_VALUE_TYPE(
194 security_state::features::kMarkHttpAsFeature,
195 kMarkHttpAsFeatureVariations,
196 "MarkHttpAs")},
Mike Doughertyc20eefa2018-02-20 19:20:33197 {"ios-captive-portal-metrics", flag_descriptions::kCaptivePortalMetricsName,
198 flag_descriptions::kCaptivePortalMetricsDescription, flags_ui::kOsIos,
199 FEATURE_VALUE_TYPE(kCaptivePortalMetrics)},
Cooper Knaak1e026562017-07-26 05:22:28200 {"in-product-help-demo-mode-choice",
201 flag_descriptions::kInProductHelpDemoModeName,
202 flag_descriptions::kInProductHelpDemoModeDescription, flags_ui::kOsIos,
203 FEATURE_WITH_PARAMS_VALUE_TYPE(
Tommy Nyquistc1d6dea12017-07-26 20:37:23204 feature_engagement::kIPHDemoMode,
205 feature_engagement::kIPHDemoModeChoiceVariations,
Marc Treib2752e8b2017-08-04 14:12:09206 "IPH_DemoMode")},
Moe Ahmadi2d4c8452018-11-30 04:44:26207 {"preview-usdz", flag_descriptions::kUsdzPreviewName,
208 flag_descriptions::kUsdzPreviewDescription, flags_ui::kOsIos,
209 FEATURE_VALUE_TYPE(download::kUsdzPreview)},
Marc Treib2752e8b2017-08-04 14:12:09210 {"use-ddljson-api", flag_descriptions::kUseDdljsonApiName,
211 flag_descriptions::kUseDdljsonApiDescription, flags_ui::kOsIos,
Marc Treib5fc99e1f2017-11-08 12:19:18212 MULTI_VALUE_TYPE(kUseDdljsonApiChoices)},
Jean-François Geyelin30be5912017-08-25 12:05:38213 {"drag_and_drop", flag_descriptions::kDragAndDropName,
214 flag_descriptions::kDragAndDropDescription, flags_ui::kOsIos,
Rohit Rao6a3a481d2017-09-25 19:31:41215 FEATURE_VALUE_TYPE(kDragAndDrop)},
Eugene But468b6d5f02018-10-18 21:17:45216 {"ignores-viewport-scale-limits",
217 flag_descriptions::kIgnoresViewportScaleLimitsName,
218 flag_descriptions::kIgnoresViewportScaleLimitsDescription,
219 flags_ui::kOsIos,
220 FEATURE_VALUE_TYPE(web::features::kIgnoresViewportScaleLimits)},
Danyao Wang34da6ff2017-10-30 15:16:56221 {"slim-navigation-manager", flag_descriptions::kSlimNavigationManagerName,
222 flag_descriptions::kSlimNavigationManagerDescription, flags_ui::kOsIos,
223 FEATURE_VALUE_TYPE(web::features::kSlimNavigationManager)},
sczsdc2e8e02018-01-09 18:39:32224 {"memex-tab-switcher", flag_descriptions::kMemexTabSwitcherName,
225 flag_descriptions::kMemexTabSwitcherDescription, flags_ui::kOsIos,
226 FEATURE_VALUE_TYPE(kMemexTabSwitcher)},
mrefaata6501d92017-12-11 19:50:50227 {"wk-http-system-cookie-store",
228 flag_descriptions::kWKHTTPSystemCookieStoreName,
229 flag_descriptions::kWKHTTPSystemCookieStoreName, flags_ui::kOsIos,
Mohamad Ahmadic4df81f2017-12-20 04:41:59230 FEATURE_VALUE_TYPE(web::features::kWKHTTPSystemCookieStore)},
Moe Ahmadic3fd7cd2018-05-11 21:40:22231 {"enable-autofill-credit-card-upload",
232 flag_descriptions::kAutofillCreditCardUploadName,
233 flag_descriptions::kAutofillCreditCardUploadDescription, flags_ui::kOsIos,
Anne Lim579b5732018-08-30 18:24:24234 FEATURE_VALUE_TYPE(autofill::features::kAutofillUpstream)},
Lakshmi Kumar Dabbiruee22004e2018-06-01 01:52:27235 {"enable-autofill-credit-card-downstream-google-pay-branding",
236 flag_descriptions::kAutofillDownstreamUseGooglePayBrandingOniOSName,
237 flag_descriptions::kAutofillDownstreamUseGooglePayBrandingOniOSDescription,
238 flags_ui::kOsIos,
239 FEATURE_VALUE_TYPE(
240 autofill::features::kAutofillDownstreamUseGooglePayBrandingOniOS)},
Jared Saul2e55e7f12018-10-26 20:44:43241 {"enable-autofill-save-credit-card-uses-strike-system",
242 flag_descriptions::kEnableAutofillSaveCreditCardUsesStrikeSystemName,
243 flag_descriptions::
244 kEnableAutofillSaveCreditCardUsesStrikeSystemDescription,
245 flags_ui::kOsIos,
246 FEATURE_VALUE_TYPE(
247 autofill::features::kAutofillSaveCreditCardUsesStrikeSystem)},
Anne Lim9d8f66a2019-01-07 23:49:21248 {"enable-autofill-save-credit-card-uses-strike-system-v2",
249 flag_descriptions::kEnableAutofillSaveCreditCardUsesStrikeSystemV2Name,
250 flag_descriptions::
251 kEnableAutofillSaveCreditCardUsesStrikeSystemV2Description,
252 flags_ui::kOsIos,
253 FEATURE_VALUE_TYPE(
254 autofill::features::kAutofillSaveCreditCardUsesStrikeSystemV2)},
Moe Ahmadid6d5d172018-06-20 17:20:23255 {"use-sync-sandbox", flag_descriptions::kSyncSandboxName,
256 flag_descriptions::kSyncSandboxDescription, flags_ui::kOsIos,
257 SINGLE_VALUE_TYPE_AND_VALUE(
258 switches::kSyncServiceURL,
259 "https://chrome-sync.sandbox.google.com/chrome-sync/alpha")},
260 {"wallet-service-use-sandbox",
261 flag_descriptions::kWalletServiceUseSandboxName,
262 flag_descriptions::kWalletServiceUseSandboxDescription, flags_ui::kOsIos,
263 ENABLE_DISABLE_VALUE_TYPE_AND_VALUE(
264 autofill::switches::kWalletServiceUseSandbox,
265 "1",
266 autofill::switches::kWalletServiceUseSandbox,
267 "0")},
Mohamad Ahmadic4df81f2017-12-20 04:41:59268 {"show-autofill-type-predictions",
269 flag_descriptions::kShowAutofillTypePredictionsName,
270 flag_descriptions::kShowAutofillTypePredictionsDescription,
271 flags_ui::kOsIos,
Gauthier Ambardcfd85182018-01-05 10:37:46272 FEATURE_VALUE_TYPE(autofill::features::kAutofillShowTypePredictions)},
Olivier Robin3d60411622018-02-23 10:03:22273 {"autofill-ios-delay-between-fields",
274 flag_descriptions::kAutofillIOSDelayBetweenFieldsName,
275 flag_descriptions::kAutofillIOSDelayBetweenFieldsDescription,
276 flags_ui::kOsIos, MULTI_VALUE_TYPE(kAutofillIOSDelayBetweenFieldsChoices)},
Sylvain Defresneae6987e2018-03-01 13:21:40277 {"new-clear-browsing-data-ui",
278 flag_descriptions::kNewClearBrowsingDataUIName,
279 flag_descriptions::kNewClearBrowsingDataUIDescription, flags_ui::kOsIos,
280 FEATURE_VALUE_TYPE(kNewClearBrowsingDataUI)},
mrefaat390f99c0d2018-05-10 19:44:24281 {"itunes-urls-store-kit-handling",
282 flag_descriptions::kITunesUrlsStoreKitHandlingName,
283 flag_descriptions::kITunesUrlsStoreKitHandlingDescription,
284 flags_ui::kOsIos, FEATURE_VALUE_TYPE(kITunesUrlsStoreKitHandling)},
Jérôme Lebelfa6c5ae2018-03-29 13:46:57285 {"unified-consent", flag_descriptions::kUnifiedConsentName,
286 flag_descriptions::kUnifiedConsentDescription, flags_ui::kOsIos,
Mihai Sardarescu160ec662018-07-18 21:13:51287 FEATURE_VALUE_TYPE(unified_consent::kUnifiedConsent)},
sebsg93dc01c2018-04-05 16:42:48288 {"autofill-dynamic-forms", flag_descriptions::kAutofillDynamicFormsName,
289 flag_descriptions::kAutofillDynamicFormsDescription, flags_ui::kOsIos,
290 FEATURE_VALUE_TYPE(autofill::features::kAutofillDynamicForms)},
Olivier Robin081912dd2018-06-11 09:14:26291 {"autofill-prefilled-fields",
292 flag_descriptions::kAutofillPrefilledFieldsName,
293 flag_descriptions::kAutofillPrefilledFieldsDescription, flags_ui::kOsIos,
294 FEATURE_VALUE_TYPE(autofill::features::kAutofillPrefilledFields)},
Olivier Robind81475b02018-06-20 21:30:11295 {"autofill-show-all-profiles-on-prefilled-forms",
296 flag_descriptions::kAutofillShowAllSuggestionsOnPrefilledFormsName,
297 flag_descriptions::kAutofillShowAllSuggestionsOnPrefilledFormsDescription,
298 flags_ui::kOsIos,
299 FEATURE_VALUE_TYPE(
300 autofill::features::kAutofillShowAllSuggestionsOnPrefilledForms)},
Moe Ahmadid3ddfcb2018-04-20 19:51:08301 {"autofill-restrict-formless-form-extraction",
302 flag_descriptions::kAutofillRestrictUnownedFieldsToFormlessCheckoutName,
303 flag_descriptions::
304 kAutofillRestrictUnownedFieldsToFormlessCheckoutDescription,
305 flags_ui::kOsIos,
306 FEATURE_VALUE_TYPE(
307 autofill::features::kAutofillRestrictUnownedFieldsToFormlessCheckout)},
Roger McFarlanee193ab122018-11-22 20:55:07308 {"autofill-rich-metadata-queries",
309 flag_descriptions::kAutofillRichMetadataQueriesName,
310 flag_descriptions::kAutofillRichMetadataQueriesDescription,
311 flags_ui::kOsIos,
312 FEATURE_VALUE_TYPE(autofill::features::kAutofillRichMetadataQueries)},
Kurt Horimotodc33af32018-05-01 01:39:14313 {"fullscreen-viewport-adjustment-experiment",
314 flag_descriptions::kFullscreenViewportAdjustmentExperimentName,
315 flag_descriptions::kFullscreenViewportAdjustmentExperimentDescription,
316 flags_ui::kOsIos,
317 MULTI_VALUE_TYPE(
318 fullscreen::features::kViewportAdjustmentExperimentChoices)},
Roger McFarlane1a37d7b2018-04-20 22:45:19319 {"autofill-enforce-min-required-fields-for-heuristics",
320 flag_descriptions::kAutofillEnforceMinRequiredFieldsForHeuristicsName,
321 flag_descriptions::
322 kAutofillEnforceMinRequiredFieldsForHeuristicsDescription,
323 flags_ui::kOsIos,
324 FEATURE_VALUE_TYPE(
325 autofill::features::kAutofillEnforceMinRequiredFieldsForHeuristics)},
326 {"autofill-enforce-min-required-fields-for-query",
327 flag_descriptions::kAutofillEnforceMinRequiredFieldsForQueryName,
328 flag_descriptions::kAutofillEnforceMinRequiredFieldsForQueryDescription,
329 flags_ui::kOsIos,
330 FEATURE_VALUE_TYPE(
331 autofill::features::kAutofillEnforceMinRequiredFieldsForQuery)},
332 {"autofill-enforce-min-required-fields-for-upload",
333 flag_descriptions::kAutofillEnforceMinRequiredFieldsForUploadName,
334 flag_descriptions::kAutofillEnforceMinRequiredFieldsForUploadDescription,
335 flags_ui::kOsIos,
336 FEATURE_VALUE_TYPE(
337 autofill::features::kAutofillEnforceMinRequiredFieldsForUpload)},
Gauthier Ambard4417f852018-04-26 14:24:32338 {"browser-container-fullscreen",
339 flag_descriptions::kBrowserContainerFullscreenName,
340 flag_descriptions::kBrowserContainerFullscreenDescription,
Gauthier Ambard5d7627282018-05-18 08:20:07341 flags_ui::kOsIos,
342 FEATURE_VALUE_TYPE(web::features::kBrowserContainerFullscreen)},
Roger McFarlane2eceba982018-05-18 01:37:11343 {"autofill-cache-query-responses",
344 flag_descriptions::kAutofillCacheQueryResponsesName,
345 flag_descriptions::kAutofillCacheQueryResponsesDescription,
346 flags_ui::kOsIos,
347 FEATURE_VALUE_TYPE(autofill::features::kAutofillCacheQueryResponses)},
Jeffrey Cohen05cdafa392018-08-15 20:49:17348 {"autofill-enable-company-name",
349 flag_descriptions::kAutofillEnableCompanyNameName,
350 flag_descriptions::kAutofillEnableCompanyNameDescription, flags_ui::kOsIos,
351 FEATURE_VALUE_TYPE(autofill::features::kAutofillEnableCompanyName)},
Javier Ernesto Flores Robles23c71a7d2018-06-01 09:51:17352 {"autofill-manual-fallback", flag_descriptions::kAutofillManualFallbackName,
353 flag_descriptions::kAutofillManualFallbackDescription, flags_ui::kOsIos,
354 FEATURE_VALUE_TYPE(autofill::features::kAutofillManualFallback)},
Yi Su392b3032018-06-05 07:26:47355 {"webpage-text-accessibility",
356 flag_descriptions::kWebPageTextAccessibilityName,
357 flag_descriptions::kWebPageTextAccessibilityDescription, flags_ui::kOsIos,
358 FEATURE_VALUE_TYPE(web::kWebPageTextAccessibility)},
Mike Dougherty39d36b02018-06-19 19:21:40359 {"web-frame-messaging", flag_descriptions::kWebFrameMessagingName,
360 flag_descriptions::kWebFrameMessagingDescription, flags_ui::kOsIos,
361 FEATURE_VALUE_TYPE(web::features::kWebFrameMessaging)},
Yi Suf63219e2018-06-25 17:41:01362 {"copy-image", flag_descriptions::kCopyImageName,
363 flag_descriptions::kCopyImageDescription, flags_ui::kOsIos,
364 FEATURE_VALUE_TYPE(kCopyImage)},
Vadym Doroshenkoa9a914942018-07-19 08:23:30365 {"new-password-form-parsing",
366 flag_descriptions::kNewPasswordFormParsingName,
367 flag_descriptions::kNewPasswordFormParsingDescription, flags_ui::kOsIos,
368 FEATURE_VALUE_TYPE(password_manager::features::kNewPasswordFormParsing)},
mrefaat4539b032018-08-02 22:25:45369 {"app-launcher-refresh", flag_descriptions::kAppLauncherRefreshName,
370 flag_descriptions::kAppLauncherRefreshDescription, flags_ui::kOsIos,
371 FEATURE_VALUE_TYPE(kAppLauncherRefresh)},
Marc Treib6f0bbd22018-08-28 10:07:19372 {"sync-standalone-transport",
373 flag_descriptions::kSyncStandaloneTransportName,
374 flag_descriptions::kSyncStandaloneTransportDescription, flags_ui::kOsIos,
375 FEATURE_VALUE_TYPE(switches::kSyncStandaloneTransport)},
Marc Treib4bd95e72018-08-30 06:30:58376 {"sync-support-secondary-account",
377 flag_descriptions::kSyncSupportSecondaryAccountName,
378 flag_descriptions::kSyncSupportSecondaryAccountDescription,
379 flags_ui::kOsIos,
380 FEATURE_VALUE_TYPE(switches::kSyncSupportSecondaryAccount)},
Gauthier Ambard66de93d2018-08-30 15:58:19381 {"out-of-web-fullscreen", flag_descriptions::kOutOfWebFullscreenName,
382 flag_descriptions::kOutOfWebFullscreenDescription, flags_ui::kOsIos,
383 FEATURE_VALUE_TYPE(web::features::kOutOfWebFullscreen)},
Javier Ernesto Flores Robles8e66f732018-09-12 15:41:15384 {"autofill-manual-fallback-phase-two",
385 flag_descriptions::kAutofillManualFallbackPhaseTwoName,
386 flag_descriptions::kAutofillManualFallbackPhaseTwoDescription,
387 flags_ui::kOsIos,
388 FEATURE_VALUE_TYPE(autofill::features::kAutofillManualFallbackPhaseTwo)},
Kurt Horimoto79d590b2018-09-12 19:09:28389 {"toolbar-container", flag_descriptions::kToolbarContainerName,
390 flag_descriptions::kToolbarContainerDescription, flags_ui::kOsIos,
391 FEATURE_VALUE_TYPE(toolbar_container::kToolbarContainerEnabled)},
Eugene Butc61230ab2018-11-09 00:31:58392 {"present-sad-tab-in-view-controller",
393 flag_descriptions::kPresentSadTabInViewControllerName,
394 flag_descriptions::kPresentSadTabInViewControllerDescription,
395 flags_ui::kOsIos, FEATURE_VALUE_TYPE(kPresentSadTabInViewController)},
[email protected]5b899dc2018-10-02 09:20:01396 {"omnibox-popup-shortcuts",
397 flag_descriptions::kOmniboxPopupShortcutIconsInZeroStateName,
398 flag_descriptions::kOmniboxPopupShortcutIconsInZeroStateDescription,
399 flags_ui::kOsIos,
400 FEATURE_VALUE_TYPE(omnibox::kOmniboxPopupShortcutIconsInZeroState)},
Jérôme Lebelec282832018-09-18 08:34:18401 {"sso-with-wkwebview", flag_descriptions::kSSOWithWKWebViewName,
402 flag_descriptions::kSSOWithWKWebViewDescription, flags_ui::kOsIos,
403 FEATURE_VALUE_TYPE(kSSOWithWKWebView)},
Yi Sua3d55f3d2018-09-24 17:49:41404 {"custom-search-engines", flag_descriptions::kCustomSearchEnginesName,
405 flag_descriptions::kCustomSearchEnginesDescription, flags_ui::kOsIos,
406 FEATURE_VALUE_TYPE(kCustomSearchEngines)},
Valeriya Sinevicha79dc612018-09-25 14:49:09407 {"use-multilogin-endpoint", flag_descriptions::kUseMultiloginEndpointName,
408 flag_descriptions::kUseMultiloginEndpointDescription, flags_ui::kOsIos,
409 FEATURE_VALUE_TYPE(kUseMultiloginEndpoint)},
edchin18a3e602018-10-08 23:39:35410 {"closing-last-incognito-tab",
411 flag_descriptions::kClosingLastIncognitoTabName,
412 flag_descriptions::kClosingLastIncognitoTabDescription, flags_ui::kOsIos,
413 FEATURE_VALUE_TYPE(kClosingLastIncognitoTab)},
Gauthier Ambardce7fd582018-10-09 08:37:05414 {"omnibox-tab-switch-suggestions",
415 flag_descriptions::kOmniboxTabSwitchSuggestionsName,
416 flag_descriptions::kOmniboxTabSwitchSuggestionsDescription,
417 flags_ui::kOsIos,
418 FEATURE_VALUE_TYPE(omnibox::kOmniboxTabSwitchSuggestions)},
Tanja Gornakb0985dd2018-10-11 17:24:40419 {"fcm-invalidations", flag_descriptions::kFCMInvalidationsName,
420 flag_descriptions::kFCMInvalidationsDescription, flags_ui::kOsIos,
421 FEATURE_VALUE_TYPE(invalidation::switches::kFCMInvalidations)},
Justin Cohen9a5023212018-10-15 14:32:37422 {"browser-container-contains-ntp",
423 flag_descriptions::kBrowserContainerContainsNTPName,
424 flag_descriptions::kBrowserContainerContainsNTPDescription,
425 flags_ui::kOsIos, FEATURE_VALUE_TYPE(kBrowserContainerContainsNTP)},
Eugene But66e917002019-01-07 23:38:52426 {"external-files-loaded-in-web-state",
427 flag_descriptions::kExternalFilesLoadedInWebStateName,
428 flag_descriptions::kExternalFilesLoadedInWebStateDescription,
429 flags_ui::kOsIos,
430 FEATURE_VALUE_TYPE(experimental_flags::kExternalFilesLoadedInWebState)},
Gauthier Ambard2392f2a2018-10-24 16:33:27431 {"search-icon-toggle", flag_descriptions::kSearchIconToggleName,
432 flag_descriptions::kSearchIconToggleDescription, flags_ui::kOsIos,
433 FEATURE_WITH_PARAMS_VALUE_TYPE(kIconForSearchButtonFeature,
434 kIconForSearchButtonVariations,
435 "ToggleSearchButtonIcon")},
Olivier Robin2a898dc2018-10-31 14:37:04436 {"enable-breakpad-upload-no-delay",
437 flag_descriptions::kBreakpadNoDelayInitialUploadName,
438 flag_descriptions::kBreakpadNoDelayInitialUploadDescription,
439 flags_ui::kOsIos,
440 FEATURE_VALUE_TYPE(crash_report::kBreakpadNoDelayInitialUpload)},
Moe Ahmadie4cce1f2018-12-18 23:48:01441 {"enable-compact-translate-infobar",
442 flag_descriptions::kCompactTranslateInfobarName,
443 flag_descriptions::kCompactTranslateInfobarDescription, flags_ui::kOsIos,
444 FEATURE_VALUE_TYPE(translate::kCompactTranslateInfobarIOS)},
Kurt Horimoto9165404a2018-11-13 23:48:12445 {"non-modal-dialogs", flag_descriptions::kNonModalDialogsName,
446 flag_descriptions::kNonModalDialogsDescription, flags_ui::kOsIos,
447 FEATURE_VALUE_TYPE(dialogs::kNonModalDialogs)},
Mikel Astizdb43c5652018-11-22 16:52:05448 {"sync-pseudo-uss-favicons", flag_descriptions::kSyncPseudoUSSFaviconsName,
449 flag_descriptions::kSyncPseudoUSSFaviconsDescription, flags_ui::kOsIos,
450 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSFavicons)},
451 {"sync-pseudo-uss-history-delete-directives",
452 flag_descriptions::kSyncPseudoUSSHistoryDeleteDirectivesName,
453 flag_descriptions::kSyncPseudoUSSHistoryDeleteDirectivesDescription,
454 flags_ui::kOsIos,
455 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSHistoryDeleteDirectives)},
456 {"sync-pseudo-uss-passwords",
457 flag_descriptions::kSyncPseudoUSSPasswordsName,
458 flag_descriptions::kSyncPseudoUSSPasswordsDescription, flags_ui::kOsIos,
459 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSPasswords)},
460 {"sync-pseudo-uss-preferences",
461 flag_descriptions::kSyncPseudoUSSPreferencesName,
462 flag_descriptions::kSyncPseudoUSSPreferencesDescription, flags_ui::kOsIos,
463 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSPreferences)},
464 {"sync-pseudo-uss-priority-preferences",
465 flag_descriptions::kSyncPseudoUSSPriorityPreferencesName,
466 flag_descriptions::kSyncPseudoUSSPriorityPreferencesDescription,
467 flags_ui::kOsIos,
468 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSPriorityPreferences)},
Jérôme Lebel5717fa852018-12-19 12:11:13469 {"use-nsurlsession-for-signin",
470 flag_descriptions::kUseNSURLSessionForGaiaSigninRequestsName,
471 flag_descriptions::kUseNSURLSessionForGaiaSigninRequestsDescription,
472 flags_ui::kOsIos,
473 FEATURE_VALUE_TYPE(kUseNSURLSessionForGaiaSigninRequests)},
Mikel Astizdb43c5652018-11-22 16:52:05474 {"sync-pseudo-uss-supervised-users",
475 flag_descriptions::kSyncPseudoUSSSupervisedUsersName,
476 flag_descriptions::kSyncPseudoUSSSupervisedUsersDescription,
477 flags_ui::kOsIos,
478 FEATURE_VALUE_TYPE(switches::kSyncPseudoUSSSupervisedUsers)},
Olivier Robince9386ebd2018-11-29 09:07:38479 {"detect-main-thread-freeze",
480 flag_descriptions::kDetectMainThreadFreezeName,
481 flag_descriptions::kDetectMainThreadFreezeDescription, flags_ui::kOsIos,
482 FEATURE_WITH_PARAMS_VALUE_TYPE(crash_report::kDetectMainThreadFreeze,
483 kDetectMainThreadFreezeVariations,
484 "DetectMainThreadFreeze")},
Mohamed Amir Yosef9e8a1f02018-12-03 21:41:14485 {"enable-sync-uss-bookmarks",
486 flag_descriptions::kEnableSyncUSSBookmarksName,
487 flag_descriptions::kEnableSyncUSSBookmarksDescription, flags_ui::kOsIos,
488 FEATURE_VALUE_TYPE(switches::kSyncUSSBookmarks)},
sczs80a6f462018-12-05 20:37:18489 {"infobar-ui-reboot", flag_descriptions::kInfobarUIRebootName,
490 flag_descriptions::kInfobarUIRebootDescription, flags_ui::kOsIos,
491 FEATURE_VALUE_TYPE(kInfobarUIReboot)},
Chris Lu77905ac2018-12-06 20:46:51492 {"find-in-page-iframe", flag_descriptions::kFindInPageiFrameName,
493 flag_descriptions::kFindInPageiFrameDescription, flags_ui::kOsIos,
494 FEATURE_VALUE_TYPE(kFindInPageiFrame)},
Robbie Gibsoncd964bb2018-12-11 18:22:21495 {"search-copied-image", flag_descriptions::kSearchCopiedImageName,
496 flag_descriptions::kSearchCopiedImageDescription, flags_ui::kOsIos,
497 FEATURE_VALUE_TYPE(kSearchCopiedImage)},
Robbie Gibson640d842b2018-12-18 11:55:13498 {"copied-text-behavior", flag_descriptions::kCopiedTextBehaviorName,
499 flag_descriptions::kCopiedTextBehaviorName, flags_ui::kOsIos,
Robbie Gibsone6915ce12018-12-26 12:08:10500 FEATURE_VALUE_TYPE(omnibox::kCopiedTextBehavior)},
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)},
Gauthier Ambardcfd85182018-01-05 10:37:46515};
sdefresne14900ee2015-11-27 14:43:21516
517// Add all switches from experimental flags to |command_line|.
518void AppendSwitchesFromExperimentalSettings(base::CommandLine* command_line) {
519 NSUserDefaults* defaults = [NSUserDefaults standardUserDefaults];
520
sdefresne14900ee2015-11-27 14:43:21521 // Set the UA flag if UseMobileSafariUA is enabled.
522 if ([defaults boolForKey:@"UseMobileSafariUA"]) {
523 // Safari uses "Vesion/", followed by the OS version excluding bugfix, where
524 // Chrome puts its product token.
avi571943672015-12-22 02:12:49525 int32_t major = 0;
526 int32_t minor = 0;
527 int32_t bugfix = 0;
sdefresne14900ee2015-11-27 14:43:21528 base::SysInfo::OperatingSystemVersionNumbers(&major, &minor, &bugfix);
529 std::string product = base::StringPrintf("Version/%d.%d", major, minor);
530
531 command_line->AppendSwitchASCII(switches::kUserAgent,
532 web::BuildUserAgentFromProduct(product));
533 }
534
535 // Freeform commandline flags. These are added last, so that any flags added
536 // earlier in this function take precedence.
537 if ([defaults boolForKey:@"EnableFreeformCommandLineFlags"]) {
538 base::CommandLine::StringVector flags;
539 // Append an empty "program" argument.
540 flags.push_back("");
541
542 // The number of flags corresponds to the number of text fields in
543 // Experimental.plist.
544 const int kNumFreeformFlags = 5;
545 for (int i = 1; i <= kNumFreeformFlags; ++i) {
546 NSString* key =
547 [NSString stringWithFormat:@"FreeformCommandLineFlag%d", i];
548 NSString* flag = [defaults stringForKey:key];
549 if ([flag length]) {
550 flags.push_back(base::SysNSStringToUTF8(flag));
551 }
552 }
553
554 base::CommandLine temp_command_line(flags);
555 command_line->AppendArguments(temp_command_line, false);
556 }
msardafc76f662017-02-24 12:46:28557
justincohendacc85d2017-06-28 23:34:10558 // Populate command line flag for 3rd party keyboard omnibox workaround.
559 NSString* enableThirdPartyKeyboardWorkaround =
560 [defaults stringForKey:@"EnableThirdPartyKeyboardWorkaround"];
561 if ([enableThirdPartyKeyboardWorkaround isEqualToString:@"Enabled"]) {
562 command_line->AppendSwitch(switches::kEnableThirdPartyKeyboardWorkaround);
563 } else if ([enableThirdPartyKeyboardWorkaround isEqualToString:@"Disabled"]) {
564 command_line->AppendSwitch(switches::kDisableThirdPartyKeyboardWorkaround);
565 }
566
msardafc76f662017-02-24 12:46:28567 ios::GetChromeBrowserProvider()->AppendSwitchesFromExperimentalSettings(
568 defaults, command_line);
sdefresne14900ee2015-11-27 14:43:21569}
570
571bool SkipConditionalFeatureEntry(const flags_ui::FeatureEntry& entry) {
572 return false;
573}
574
575class FlagsStateSingleton {
576 public:
577 FlagsStateSingleton()
kokihoon189a4022018-10-13 02:43:44578 : flags_state_(kFeatureEntries, base::size(kFeatureEntries)) {}
sdefresne14900ee2015-11-27 14:43:21579 ~FlagsStateSingleton() {}
580
581 static FlagsStateSingleton* GetInstance() {
582 return base::Singleton<FlagsStateSingleton>::get();
583 }
584
585 static flags_ui::FlagsState* GetFlagsState() {
586 return &GetInstance()->flags_state_;
587 }
588
589 private:
590 flags_ui::FlagsState flags_state_;
591
592 DISALLOW_COPY_AND_ASSIGN(FlagsStateSingleton);
593};
594} // namespace
595
596void ConvertFlagsToSwitches(flags_ui::FlagsStorage* flags_storage,
597 base::CommandLine* command_line) {
Gregory Chatzinoff31305112017-12-20 21:56:02598 AppendSwitchesFromExperimentalSettings(command_line);
sdefresne14900ee2015-11-27 14:43:21599 FlagsStateSingleton::GetFlagsState()->ConvertFlagsToSwitches(
sdefresnec9763902015-12-02 10:30:11600 flags_storage, command_line, flags_ui::kAddSentinels,
Gregory Chatzinofff6523722017-11-21 01:33:53601 switches::kEnableFeatures, switches::kDisableFeatures);
sdefresne14900ee2015-11-27 14:43:21602}
603
jkrcalbf073372016-07-29 07:21:31604std::vector<std::string> RegisterAllFeatureVariationParameters(
605 flags_ui::FlagsStorage* flags_storage,
606 base::FeatureList* feature_list) {
607 return FlagsStateSingleton::GetFlagsState()
608 ->RegisterAllFeatureVariationParameters(flags_storage, feature_list);
609}
610
sdefresne14900ee2015-11-27 14:43:21611void GetFlagFeatureEntries(flags_ui::FlagsStorage* flags_storage,
612 flags_ui::FlagAccess access,
613 base::ListValue* supported_entries,
614 base::ListValue* unsupported_entries) {
615 FlagsStateSingleton::GetFlagsState()->GetFlagFeatureEntries(
616 flags_storage, access, supported_entries, unsupported_entries,
617 base::Bind(&SkipConditionalFeatureEntry));
618}
619
620void SetFeatureEntryEnabled(flags_ui::FlagsStorage* flags_storage,
621 const std::string& internal_name,
622 bool enable) {
623 FlagsStateSingleton::GetFlagsState()->SetFeatureEntryEnabled(
624 flags_storage, internal_name, enable);
625}
626
627void ResetAllFlags(flags_ui::FlagsStorage* flags_storage) {
628 FlagsStateSingleton::GetFlagsState()->ResetAllFlags(flags_storage);
629}
630
631namespace testing {
632
633const flags_ui::FeatureEntry* GetFeatureEntries(size_t* count) {
kokihoon189a4022018-10-13 02:43:44634 *count = base::size(kFeatureEntries);
sdefresne14900ee2015-11-27 14:43:21635 return kFeatureEntries;
636}
637
638} // namespace testing