Index: chrome/browser/banners/app_banner_settings_helper.cc |
diff --git a/chrome/browser/banners/app_banner_settings_helper.cc b/chrome/browser/banners/app_banner_settings_helper.cc |
index 40632dc61b8973d189b81fa9824ec874270886a4..1847f506e39c27df9660e4ce65995698849d5ad4 100644 |
--- a/chrome/browser/banners/app_banner_settings_helper.cc |
+++ b/chrome/browser/banners/app_banner_settings_helper.cc |
@@ -23,21 +23,28 @@ const size_t kMaxAppsPerSite = 3; |
// Oldest could show banner event we care about, in days. |
const unsigned int kOldestCouldShowBannerEventInDays = 14; |
-// Dictionary key to use for the 'could show banner' events. |
-const char kCouldShowBannerEventsKey[] = "couldShowBannerEvents"; |
+// Number of times that the banner could have been shown before the banner will |
+// actually be triggered. |
+const unsigned int kCouldShowEventsToTrigger = 2; |
+ |
+// Number of days that showing the banner will prevent it being seen again for. |
+const unsigned int kMinimumDaysBetweenBannerShows = 60; |
+ |
+// Number of days that the banner being blocked will prevent it being seen again |
+// for. |
+const unsigned int kMinimumBannerBlockedToBannerShown = 90; |
+ |
+// Dictionary keys to use for the events. |
+const char* kBannerEventKeys[] = { |
+ "couldShowBannerEvents", |
+ "didShowBannerEvent", |
+ "didBlockBannerEvent", |
+ "didAddToHomescreenEvent", |
+}; |
// Dictionary key to use whether the banner has been blocked. |
const char kHasBlockedKey[] = "hasBlocked"; |
-base::Time DateFromTime(base::Time time) { |
- base::Time::Exploded exploded; |
- time.LocalExplode(&exploded); |
- exploded.hour = 0; |
- exploded.minute = 0; |
- exploded.second = 0; |
- return base::Time::FromLocalExploded(exploded); |
-} |
- |
scoped_ptr<base::DictionaryValue> GetOriginDict( |
HostContentSettingsMap* settings, |
const GURL& origin_url) { |
@@ -72,10 +79,11 @@ base::DictionaryValue* GetAppDict(base::DictionaryValue* origin_dict, |
} // namespace |
-void AppBannerSettingsHelper::RecordCouldShowBannerEvent( |
+void AppBannerSettingsHelper::RecordBannerEvent( |
content::WebContents* web_contents, |
const GURL& origin_url, |
const std::string& package_name_or_start_url, |
+ AppBannerEvent event, |
base::Time time) { |
Profile* profile = |
Profile::FromBrowserContext(web_contents->GetBrowserContext()); |
@@ -99,50 +107,92 @@ void AppBannerSettingsHelper::RecordCouldShowBannerEvent( |
if (!app_dict) |
return; |
- base::ListValue* could_show_list = nullptr; |
- if (!app_dict->GetList(kCouldShowBannerEventsKey, &could_show_list)) { |
- could_show_list = new base::ListValue(); |
- app_dict->Set(kCouldShowBannerEventsKey, make_scoped_ptr(could_show_list)); |
- } |
+ std::string event_key(kBannerEventKeys[event]); |
- // Trim any items that are older than we should care about. For comparisons |
- // the times are converted to local dates. |
- base::Time date = DateFromTime(time); |
- base::ValueVector::iterator it = could_show_list->begin(); |
- while (it != could_show_list->end()) { |
- if ((*it)->IsType(base::Value::TYPE_DOUBLE)) { |
- double internal_date; |
- (*it)->GetAsDouble(&internal_date); |
- base::Time other_date = |
- DateFromTime(base::Time::FromInternalValue(internal_date)); |
- // This date has already been added. Don't add the date again, and don't |
- // bother trimming values as it will have been done the first time the |
- // date was added (unless the local date has changed, which we can live |
- // with). |
- if (other_date == date) |
- return; |
- |
- base::TimeDelta delta = date - other_date; |
- if (delta < |
- base::TimeDelta::FromDays(kOldestCouldShowBannerEventInDays)) { |
- ++it; |
- continue; |
+ if (event == APP_BANNER_EVENT_COULD_SHOW) { |
+ base::ListValue* could_show_list = nullptr; |
+ if (!app_dict->GetList(event_key, &could_show_list)) { |
+ could_show_list = new base::ListValue(); |
+ app_dict->Set(event_key, make_scoped_ptr(could_show_list)); |
+ } |
+ |
+ // Trim any items that are older than we should care about. For comparisons |
+ // the times are converted to local dates. |
+ base::Time date = time.LocalMidnight(); |
+ base::ValueVector::iterator it = could_show_list->begin(); |
+ while (it != could_show_list->end()) { |
+ if ((*it)->IsType(base::Value::TYPE_DOUBLE)) { |
+ double internal_date; |
+ (*it)->GetAsDouble(&internal_date); |
+ base::Time other_date = |
+ base::Time::FromInternalValue(internal_date).LocalMidnight(); |
+ // This date has already been added. Don't add the date again, and don't |
+ // bother trimming values as it will have been done the first time the |
+ // date was added (unless the local date has changed, which we can live |
+ // with). |
+ if (other_date == date) |
+ return; |
+ |
+ base::TimeDelta delta = date - other_date; |
+ if (delta < |
+ base::TimeDelta::FromDays(kOldestCouldShowBannerEventInDays)) { |
+ ++it; |
+ continue; |
+ } |
} |
+ |
+ // Either this date is older than we care about, or it isn't a date, so |
+ // remove it; |
+ it = could_show_list->Erase(it, nullptr); |
} |
- // Either this date is older than we care about, or it isn't a date, so |
- // remove it; |
- it = could_show_list->Erase(it, nullptr); |
+ // Dates are stored in their raw form (i.e. not local dates) to be resilient |
+ // to time zone changes. |
+ could_show_list->AppendDouble(time.ToInternalValue()); |
+ } else { |
+ app_dict->SetDouble(event_key, time.ToInternalValue()); |
} |
- |
- // Dates are stored in their raw form (i.e. not local dates) to be resilient |
- // to time zone changes. |
- could_show_list->AppendDouble(time.ToInternalValue()); |
settings->SetWebsiteSetting(pattern, ContentSettingsPattern::Wildcard(), |
CONTENT_SETTINGS_TYPE_APP_BANNER, std::string(), |
origin_dict.release()); |
} |
+bool AppBannerSettingsHelper::ShouldShowBanner( |
+ content::WebContents* web_contents, |
+ const GURL& origin_url, |
+ const std::string& package_name_or_start_url, |
+ base::Time time) { |
+ // Don't show if it has been added to the homescreen. |
+ base::Time added_time = |
+ GetSingleBannerEvent(web_contents, origin_url, package_name_or_start_url, |
+ APP_BANNER_EVENT_DID_ADD_TO_HOMESCREEN); |
+ if (!added_time.is_null()) |
+ return false; |
+ |
+ base::Time blocked_time = |
+ GetSingleBannerEvent(web_contents, origin_url, package_name_or_start_url, |
+ APP_BANNER_EVENT_DID_BLOCK); |
+ |
+ // Null times are in the distant past, so the delta between real times and |
+ // null events will always be greater than the limits. |
+ if (time - blocked_time < |
+ base::TimeDelta::FromDays(kMinimumBannerBlockedToBannerShown)) { |
+ return false; |
+ } |
+ |
+ base::Time shown_time = |
+ GetSingleBannerEvent(web_contents, origin_url, package_name_or_start_url, |
+ APP_BANNER_EVENT_DID_SHOW); |
+ if (time - shown_time < |
+ base::TimeDelta::FromDays(kMinimumDaysBetweenBannerShows)) { |
+ return false; |
+ } |
+ |
+ std::vector<base::Time> could_show_events = GetCouldShowBannerEvents( |
+ web_contents, origin_url, package_name_or_start_url); |
+ return could_show_events.size() >= kCouldShowEventsToTrigger; |
+} |
+ |
std::vector<base::Time> AppBannerSettingsHelper::GetCouldShowBannerEvents( |
content::WebContents* web_contents, |
const GURL& origin_url, |
@@ -165,8 +215,9 @@ std::vector<base::Time> AppBannerSettingsHelper::GetCouldShowBannerEvents( |
if (!app_dict) |
return result; |
+ std::string event_key(kBannerEventKeys[APP_BANNER_EVENT_COULD_SHOW]); |
base::ListValue* could_show_list = nullptr; |
- if (!app_dict->GetList(kCouldShowBannerEventsKey, &could_show_list)) |
+ if (!app_dict->GetList(event_key, &could_show_list)) |
return result; |
for (auto value : *could_show_list) { |
@@ -181,6 +232,39 @@ std::vector<base::Time> AppBannerSettingsHelper::GetCouldShowBannerEvents( |
return result; |
} |
+base::Time AppBannerSettingsHelper::GetSingleBannerEvent( |
+ content::WebContents* web_contents, |
+ const GURL& origin_url, |
+ const std::string& package_name_or_start_url, |
+ AppBannerEvent event) { |
+ DCHECK(event != APP_BANNER_EVENT_COULD_SHOW); |
+ DCHECK(event < APP_BANNER_EVENT_NUM_EVENTS); |
+ |
+ if (package_name_or_start_url.empty()) |
+ return base::Time(); |
+ |
+ Profile* profile = |
+ Profile::FromBrowserContext(web_contents->GetBrowserContext()); |
+ HostContentSettingsMap* settings = profile->GetHostContentSettingsMap(); |
+ scoped_ptr<base::DictionaryValue> origin_dict = |
+ GetOriginDict(settings, origin_url); |
+ |
+ if (!origin_dict) |
+ return base::Time(); |
+ |
+ base::DictionaryValue* app_dict = |
+ GetAppDict(origin_dict.get(), package_name_or_start_url); |
+ if (!app_dict) |
+ return base::Time(); |
+ |
+ std::string event_key(kBannerEventKeys[event]); |
+ double internal_time; |
+ if (!app_dict->GetDouble(event_key, &internal_time)) |
+ return base::Time(); |
+ |
+ return base::Time::FromInternalValue(internal_time); |
+} |
+ |
bool AppBannerSettingsHelper::IsAllowed( |
content::WebContents* web_contents, |
const GURL& origin_url, |