blob: 32c4138ec358c974aeaec3563e827f9748def423 [file] [log] [blame]
// Copyright 2017 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/open_from_clipboard/clipboard_recent_content_generic.h"
#include <string>
#include "base/functional/bind.h"
#include "base/strings/string_util.h"
#include "build/build_config.h"
#include "ui/base/clipboard/clipboard.h"
#include "ui/base/data_transfer_policy/data_transfer_endpoint.h"
#include "url/url_util.h"
#if BUILDFLAG(IS_ANDROID)
#include "ui/base/clipboard/clipboard_android.h"
#endif
namespace {
// Schemes appropriate for suggestion by ClipboardRecentContent.
const char* kAuthorizedSchemes[] = {
url::kAboutScheme, url::kDataScheme, url::kHttpScheme, url::kHttpsScheme,
// TODO(mpearson): add support for chrome:// URLs. Right now the scheme
// for that lives in content and is accessible via
// GetEmbedderRepresentationOfAboutScheme() or content::kChromeUIScheme
// TODO(mpearson): when adding desktop support, add kFileScheme, kFtpScheme.
};
void OnGetRecentImageFromClipboard(
ClipboardRecentContent::GetRecentImageCallback callback,
const std::vector<uint8_t>& png_data) {
if (png_data.empty()) {
std::move(callback).Run(std::nullopt);
return;
}
std::move(callback).Run(
gfx::Image::CreateFrom1xPNGBytes(png_data.data(), png_data.size()));
}
bool HasRecentURLFromClipboard() {
ui::Clipboard* clipboard = ui::Clipboard::GetForCurrentThread();
ui::DataTransferEndpoint data_dst = ui::DataTransferEndpoint(
ui::EndpointType::kDefault, /*notify_if_restricted=*/false);
return clipboard->IsFormatAvailable(ui::ClipboardFormatType::UrlType(),
ui::ClipboardBuffer::kCopyPaste,
&data_dst);
}
bool HasRecentTextFromClipboard() {
ui::Clipboard* clipboard = ui::Clipboard::GetForCurrentThread();
ui::DataTransferEndpoint data_dst = ui::DataTransferEndpoint(
ui::EndpointType::kDefault, /*notify_if_restricted=*/false);
return clipboard->IsFormatAvailable(ui::ClipboardFormatType::PlainTextType(),
ui::ClipboardBuffer::kCopyPaste,
&data_dst);
}
} // namespace
ClipboardRecentContentGeneric::ClipboardRecentContentGeneric() = default;
ClipboardRecentContentGeneric::~ClipboardRecentContentGeneric() = default;
std::optional<GURL> ClipboardRecentContentGeneric::GetRecentURLFromClipboard() {
if (GetClipboardContentAge() > MaximumAgeOfClipboard())
return std::nullopt;
// Get and clean up the clipboard before processing.
std::string gurl_string;
ui::Clipboard* clipboard = ui::Clipboard::GetForCurrentThread();
ui::DataTransferEndpoint data_dst = ui::DataTransferEndpoint(
ui::EndpointType::kDefault, /*notify_if_restricted=*/false);
#if BUILDFLAG(IS_ANDROID)
clipboard->ReadBookmark(&data_dst, nullptr, &gurl_string);
#else
clipboard->ReadAsciiText(ui::ClipboardBuffer::kCopyPaste, &data_dst,
&gurl_string);
#endif // BUILDFLAG(IS_ANDROID)
base::TrimWhitespaceASCII(gurl_string, base::TrimPositions::TRIM_ALL,
&gurl_string);
// Interpret the clipboard as a URL if possible.
GURL url;
// If there is mid-string whitespace, don't attempt to interpret the string
// as a URL. (Otherwise gurl will happily try to convert
// "http://example.com extra words" into "http://example.com%20extra%20words",
// which is not likely to be a useful or intended destination.)
if (gurl_string.find_first_of(base::kWhitespaceASCII) != std::string::npos)
return std::nullopt;
if (!gurl_string.empty()) {
url = GURL(gurl_string);
} else {
// Fall back to unicode / UTF16, as some URLs may use international domain
// names, not punycode.
std::u16string gurl_string16;
clipboard->ReadText(ui::ClipboardBuffer::kCopyPaste, &data_dst,
&gurl_string16);
base::TrimWhitespace(gurl_string16, base::TrimPositions::TRIM_ALL,
&gurl_string16);
if (gurl_string16.find_first_of(base::kWhitespaceUTF16) !=
std::string::npos)
return std::nullopt;
if (!gurl_string16.empty())
url = GURL(gurl_string16);
}
if (!url.is_valid() || !IsAppropriateSuggestion(url)) {
return std::nullopt;
}
return url;
}
std::optional<std::u16string>
ClipboardRecentContentGeneric::GetRecentTextFromClipboard() {
if (GetClipboardContentAge() > MaximumAgeOfClipboard())
return std::nullopt;
std::u16string text_from_clipboard;
ui::DataTransferEndpoint data_dst = ui::DataTransferEndpoint(
ui::EndpointType::kDefault, /*notify_if_restricted=*/false);
ui::Clipboard::GetForCurrentThread()->ReadText(
ui::ClipboardBuffer::kCopyPaste, &data_dst, &text_from_clipboard);
base::TrimWhitespace(text_from_clipboard, base::TrimPositions::TRIM_ALL,
&text_from_clipboard);
if (text_from_clipboard.empty()) {
return std::nullopt;
}
return text_from_clipboard;
}
void ClipboardRecentContentGeneric::GetRecentImageFromClipboard(
GetRecentImageCallback callback) {
if (GetClipboardContentAge() > MaximumAgeOfClipboard())
return;
ui::DataTransferEndpoint data_dst = ui::DataTransferEndpoint(
ui::EndpointType::kDefault, /*notify_if_restricted=*/false);
ui::Clipboard::GetForCurrentThread()->ReadPng(
ui::ClipboardBuffer::kCopyPaste, &data_dst,
base::BindOnce(&OnGetRecentImageFromClipboard, std::move(callback)));
}
std::optional<std::set<ClipboardContentType>>
ClipboardRecentContentGeneric::GetCachedClipboardContentTypes() {
if (GetClipboardContentAge() > MaximumAgeOfClipboard()) {
return std::nullopt;
}
std::set<ClipboardContentType> clipboard_content_types;
if (HasRecentURLFromClipboard()) {
clipboard_content_types.insert(ClipboardContentType::URL);
}
if (HasRecentTextFromClipboard()) {
clipboard_content_types.insert(ClipboardContentType::Text);
}
if (HasRecentImageFromClipboard()) {
clipboard_content_types.insert(ClipboardContentType::Image);
}
return clipboard_content_types;
}
bool ClipboardRecentContentGeneric::HasRecentImageFromClipboard() {
if (GetClipboardContentAge() > MaximumAgeOfClipboard())
return false;
ui::DataTransferEndpoint data_dst = ui::DataTransferEndpoint(
ui::EndpointType::kDefault, /*notify_if_restricted=*/false);
return ui::Clipboard::GetForCurrentThread()->IsFormatAvailable(
ui::ClipboardFormatType::PngType(), ui::ClipboardBuffer::kCopyPaste,
&data_dst);
}
void ClipboardRecentContentGeneric::HasRecentContentFromClipboard(
std::set<ClipboardContentType> types,
HasDataCallback callback) {
std::set<ClipboardContentType> matching_types;
if (GetClipboardContentAge() > MaximumAgeOfClipboard()) {
std::move(callback).Run(matching_types);
return;
}
for (ClipboardContentType type : types) {
switch (type) {
case ClipboardContentType::URL:
if (HasRecentURLFromClipboard()) {
matching_types.insert(ClipboardContentType::URL);
}
break;
case ClipboardContentType::Text:
if (HasRecentTextFromClipboard()) {
matching_types.insert(ClipboardContentType::Text);
}
break;
case ClipboardContentType::Image:
if (HasRecentImageFromClipboard()) {
matching_types.insert(ClipboardContentType::Image);
}
break;
}
}
std::move(callback).Run(matching_types);
}
void ClipboardRecentContentGeneric::GetRecentURLFromClipboard(
GetRecentURLCallback callback) {
std::move(callback).Run(GetRecentURLFromClipboard());
}
void ClipboardRecentContentGeneric::GetRecentTextFromClipboard(
GetRecentTextCallback callback) {
std::move(callback).Run(GetRecentTextFromClipboard());
}
base::TimeDelta ClipboardRecentContentGeneric::GetClipboardContentAge() const {
const base::Time last_modified_time =
ui::Clipboard::GetForCurrentThread()->GetLastModifiedTime();
const base::Time now = base::Time::Now();
// In case of system clock change, assume the last modified time is now.
// (Don't return a negative age, i.e., a time in the future.)
if (last_modified_time > now)
return base::TimeDelta();
return now - last_modified_time;
}
void ClipboardRecentContentGeneric::SuppressClipboardContent() {
// User cleared the user data. The pasteboard entry must be removed from the
// omnibox list. Do this by pretending the current clipboard is ancient,
// not recent.
ui::Clipboard::GetForCurrentThread()->ClearLastModifiedTime();
}
void ClipboardRecentContentGeneric::ClearClipboardContent() {
ui::Clipboard::GetForCurrentThread()->Clear(ui::ClipboardBuffer::kCopyPaste);
}
// static
bool ClipboardRecentContentGeneric::IsAppropriateSuggestion(const GURL& url) {
// Check to make sure it's a scheme we're willing to suggest.
for (const auto* authorized_scheme : kAuthorizedSchemes) {
if (url.SchemeIs(authorized_scheme))
return true;
}
// Check if the schemes is an application-defined scheme.
std::vector<std::string> standard_schemes = url::GetStandardSchemes();
for (const auto& standard_scheme : standard_schemes) {
if (url.SchemeIs(standard_scheme))
return true;
}
// Not a scheme we're allowed to return.
return false;
}