// Copyright (c) 2010 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "chrome/browser/geolocation/geolocation_settings_state.h"
#include "base/string_piece.h"
#include "base/utf_string_conversions.h"
#include "chrome/browser/geolocation/geolocation_content_settings_map.h"
#include "chrome/browser/prefs/pref_service.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/common/pref_names.h"
#include "content/browser/tab_contents/navigation_entry.h"
#include "net/base/net_util.h"
GeolocationSettingsState::GeolocationSettingsState(Profile* profile)
: profile_(profile) {
}
GeolocationSettingsState::~GeolocationSettingsState() {
}
void GeolocationSettingsState::OnGeolocationPermissionSet(
const GURL& requesting_origin, bool allowed) {
state_map_[requesting_origin] =
allowed ? CONTENT_SETTING_ALLOW : CONTENT_SETTING_BLOCK;
}
void GeolocationSettingsState::DidNavigate(
const NavigationController::LoadCommittedDetails& details) {
if (details.entry)
embedder_url_ = details.entry->url();
if (state_map_.empty())
return;
if (!details.entry ||
details.previous_url.GetOrigin() != details.entry->url().GetOrigin()) {
state_map_.clear();
return;
}
// We're in the same origin, check if there's any icon to be displayed.
unsigned int tab_state_flags = 0;
GetDetailedInfo(NULL, &tab_state_flags);
if (!(tab_state_flags & TABSTATE_HAS_ANY_ICON))
state_map_.clear();
}
void GeolocationSettingsState::ClearStateMap() {
state_map_.clear();
}
void GeolocationSettingsState::GetDetailedInfo(
FormattedHostsPerState* formatted_hosts_per_state,
unsigned int* tab_state_flags) const {
DCHECK(tab_state_flags);
DCHECK(embedder_url_.is_valid());
const ContentSetting default_setting =
profile_->GetGeolocationContentSettingsMap()->GetDefaultContentSetting();
std::set<std::string> formatted_hosts;
std::set<std::string> repeated_formatted_hosts;
// Build a set of repeated formatted hosts
for (StateMap::const_iterator i(state_map_.begin());
i != state_map_.end(); ++i) {
std::string formatted_host = GURLToFormattedHost(i->first);
if (!formatted_hosts.insert(formatted_host).second) {
repeated_formatted_hosts.insert(formatted_host);
}
}
for (StateMap::const_iterator i(state_map_.begin());
i != state_map_.end(); ++i) {
if (i->second == CONTENT_SETTING_ALLOW)
*tab_state_flags |= TABSTATE_HAS_ANY_ALLOWED;
if (formatted_hosts_per_state) {
std::string formatted_host = GURLToFormattedHost(i->first);
std::string final_formatted_host =
repeated_formatted_hosts.find(formatted_host) ==
repeated_formatted_hosts.end() ?
formatted_host :
i->first.spec();
(*formatted_hosts_per_state)[i->second].insert(final_formatted_host);
}
const ContentSetting saved_setting =
profile_->GetGeolocationContentSettingsMap()->GetContentSetting(
i->first, embedder_url_);
if (saved_setting != default_setting)
*tab_state_flags |= TABSTATE_HAS_EXCEPTION;
if (saved_setting != i->second)
*tab_state_flags |= TABSTATE_HAS_CHANGED;
if (saved_setting != CONTENT_SETTING_ASK)
*tab_state_flags |= TABSTATE_HAS_ANY_ICON;
}
}
std::string GeolocationSettingsState::GURLToFormattedHost(
const GURL& url) const {
std::wstring display_host_wide;
net::AppendFormattedHost(
url, UTF8ToWide(profile_->GetPrefs()->GetString(prefs::kAcceptLanguages)),
&display_host_wide, NULL, NULL);
return WideToUTF8(display_host_wide);
}