#include "config.h"
#include "Location.h"
#include "DOMWindow.h"
#include "Document.h"
#include "Frame.h"
#include "FrameLoader.h"
#include "NavigationScheduler.h"
#include "URL.h"
#include "SecurityOrigin.h"
namespace WebCore {
Location::Location(Frame* frame)
: DOMWindowProperty(frame)
{
}
inline const URL& Location::url() const
{
ASSERT(m_frame);
const URL& url = m_frame->document()->url();
if (!url.isValid())
return blankURL();
return url;
}
String Location::href() const
{
if (!m_frame)
return String();
auto& url = this->url();
if (!url.hasUsername() && !url.hasPassword())
return url.string();
URL urlWithoutCredentials(url);
urlWithoutCredentials.setUser(WTF::emptyString());
urlWithoutCredentials.setPass(WTF::emptyString());
return urlWithoutCredentials.string();
}
String Location::protocol() const
{
if (!m_frame)
return String();
return makeString(url().protocol(), ":");
}
String Location::host() const
{
if (!m_frame)
return String();
return url().hostAndPort();
}
String Location::hostname() const
{
if (!m_frame)
return String();
return url().host();
}
String Location::port() const
{
if (!m_frame)
return String();
const URL& url = this->url();
return url.port() ? String::number(url.port().value()) : emptyString();
}
String Location::pathname() const
{
if (!m_frame)
return String();
const URL& url = this->url();
return url.path().isEmpty() ? "/" : url.path();
}
String Location::search() const
{
if (!m_frame)
return String();
const URL& url = this->url();
return url.query().isEmpty() ? emptyString() : "?" + url.query();
}
String Location::origin() const
{
if (!m_frame)
return String();
return SecurityOrigin::create(url())->toString();
}
Ref<DOMStringList> Location::ancestorOrigins() const
{
auto origins = DOMStringList::create();
if (!m_frame)
return origins;
for (Frame* frame = m_frame->tree().parent(); frame; frame = frame->tree().parent())
origins->append(frame->document()->securityOrigin().toString());
return origins;
}
String Location::hash() const
{
if (!m_frame)
return String();
const String& fragmentIdentifier = url().fragmentIdentifier();
return fragmentIdentifier.isEmpty() ? emptyString() : "#" + fragmentIdentifier;
}
ExceptionOr<void> Location::setHref(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& url)
{
if (!m_frame)
return { };
return setLocation(activeWindow, firstWindow, url);
}
ExceptionOr<void> Location::setProtocol(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& protocol)
{
if (!m_frame)
return { };
URL url = m_frame->document()->url();
if (!url.setProtocol(protocol))
return Exception { SyntaxError };
return setLocation(activeWindow, firstWindow, url.string());
}
ExceptionOr<void> Location::setHost(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& host)
{
if (!m_frame)
return { };
URL url = m_frame->document()->url();
url.setHostAndPort(host);
return setLocation(activeWindow, firstWindow, url.string());
}
ExceptionOr<void> Location::setHostname(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& hostname)
{
if (!m_frame)
return { };
URL url = m_frame->document()->url();
url.setHost(hostname);
return setLocation(activeWindow, firstWindow, url.string());
}
ExceptionOr<void> Location::setPort(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& portString)
{
if (!m_frame)
return { };
URL url = m_frame->document()->url();
int port = portString.toInt();
if (port < 0 || port > 0xFFFF || portString.isEmpty())
url.removePort();
else
url.setPort(port);
return setLocation(activeWindow, firstWindow, url.string());
}
ExceptionOr<void> Location::setPathname(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& pathname)
{
if (!m_frame)
return { };
URL url = m_frame->document()->url();
url.setPath(pathname);
return setLocation(activeWindow, firstWindow, url.string());
}
ExceptionOr<void> Location::setSearch(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& search)
{
if (!m_frame)
return { };
URL url = m_frame->document()->url();
url.setQuery(search);
return setLocation(activeWindow, firstWindow, url.string());
}
ExceptionOr<void> Location::setHash(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& hash)
{
if (!m_frame)
return { };
ASSERT(m_frame->document());
auto url = m_frame->document()->url();
auto oldFragmentIdentifier = url.fragmentIdentifier();
auto newFragmentIdentifier = hash;
if (hash[0] == '#')
newFragmentIdentifier = hash.substring(1);
url.setFragmentIdentifier(newFragmentIdentifier);
if (equalIgnoringNullity(oldFragmentIdentifier, url.fragmentIdentifier()))
return { };
return setLocation(activeWindow, firstWindow, url.string());
}
ExceptionOr<void> Location::assign(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& url)
{
if (!m_frame)
return { };
return setLocation(activeWindow, firstWindow, url);
}
void Location::replace(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& url)
{
if (!m_frame)
return;
ASSERT(m_frame->document());
ASSERT(m_frame->document()->domWindow());
m_frame->document()->domWindow()->setLocation(activeWindow, firstWindow, url, LockHistoryAndBackForwardList);
}
void Location::reload(DOMWindow& activeWindow)
{
if (!m_frame)
return;
ASSERT(activeWindow.document());
ASSERT(m_frame->document());
ASSERT(m_frame->document()->domWindow());
auto& activeDocument = *activeWindow.document();
auto& targetDocument = *m_frame->document();
if (!activeDocument.securityOrigin().canAccess(targetDocument.securityOrigin())) {
auto& targetWindow = *targetDocument.domWindow();
targetWindow.printErrorMessage(targetWindow.crossDomainAccessErrorMessage(activeWindow));
return;
}
if (protocolIsJavaScript(targetDocument.url()))
return;
m_frame->navigationScheduler().scheduleRefresh(activeDocument);
}
ExceptionOr<void> Location::setLocation(DOMWindow& activeWindow, DOMWindow& firstWindow, const String& url)
{
ASSERT(m_frame);
if (!activeWindow.document()->canNavigate(m_frame))
return Exception { SecurityError };
ASSERT(m_frame->document());
ASSERT(m_frame->document()->domWindow());
m_frame->document()->domWindow()->setLocation(activeWindow, firstWindow, url);
return { };
}
}