blob: b961c3c3bcc02f6402cb7b24a22a88980e3ba5e0 [file] [log] [blame]
// Copyright 2014 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/android/logo_bridge.h"
#include <jni.h>
#include <stdint.h>
#include "base/android/jni_android.h"
#include "base/android/jni_array.h"
#include "base/android/jni_string.h"
#include "base/metrics/histogram_macros.h"
#include "chrome/browser/profiles/profile.h"
#include "chrome/browser/profiles/profile_android.h"
#include "chrome/browser/search_provider_logos/logo_service_factory.h"
#include "components/search_provider_logos/logo_observer.h"
#include "components/search_provider_logos/logo_service.h"
#include "content/public/browser/storage_partition.h"
#include "jni/LogoBridge_jni.h"
#include "net/http/http_status_code.h"
#include "services/network/public/cpp/resource_response_info.h"
#include "services/network/public/cpp/shared_url_loader_factory.h"
#include "services/network/public/cpp/simple_url_loader.h"
#include "third_party/skia/include/core/SkBitmap.h"
#include "ui/gfx/android/java_bitmap.h"
#include "url/gurl.h"
using base::android::ConvertJavaStringToUTF8;
using base::android::ConvertUTF8ToJavaString;
using base::android::JavaParamRef;
using base::android::JavaRef;
using base::android::ScopedJavaLocalRef;
using base::android::ToJavaByteArray;
namespace {
ScopedJavaLocalRef<jobject> JNI_LogoBridge_MakeJavaLogo(
JNIEnv* env,
const SkBitmap* bitmap,
const GURL& on_click_url,
const std::string& alt_text,
const GURL& animated_url) {
ScopedJavaLocalRef<jobject> j_bitmap = gfx::ConvertToJavaBitmap(bitmap);
ScopedJavaLocalRef<jstring> j_on_click_url;
if (on_click_url.is_valid())
j_on_click_url = ConvertUTF8ToJavaString(env, on_click_url.spec());
ScopedJavaLocalRef<jstring> j_alt_text;
if (!alt_text.empty())
j_alt_text = ConvertUTF8ToJavaString(env, alt_text);
ScopedJavaLocalRef<jstring> j_animated_url;
if (animated_url.is_valid())
j_animated_url = ConvertUTF8ToJavaString(env, animated_url.spec());
return Java_LogoBridge_createLogo(env, j_bitmap, j_on_click_url, j_alt_text,
j_animated_url);
}
// Converts a C++ Logo to a Java Logo.
ScopedJavaLocalRef<jobject> JNI_LogoBridge_ConvertLogoToJavaObject(
JNIEnv* env,
const search_provider_logos::Logo* logo) {
if (!logo)
return ScopedJavaLocalRef<jobject>();
return JNI_LogoBridge_MakeJavaLogo(
env, &logo->image, GURL(logo->metadata.on_click_url),
logo->metadata.alt_text, GURL(logo->metadata.animated_url));
}
class LogoObserverAndroid : public search_provider_logos::LogoObserver {
public:
LogoObserverAndroid(base::WeakPtr<LogoBridge> logo_bridge,
JNIEnv* env,
jobject j_logo_observer)
: logo_bridge_(logo_bridge) {
j_logo_observer_.Reset(env, j_logo_observer);
}
~LogoObserverAndroid() override {}
// seach_provider_logos::LogoObserver:
void OnLogoAvailable(const search_provider_logos::Logo* logo,
bool from_cache) override {
if (!logo_bridge_)
return;
JNIEnv* env = base::android::AttachCurrentThread();
ScopedJavaLocalRef<jobject> j_logo =
JNI_LogoBridge_ConvertLogoToJavaObject(env, logo);
Java_LogoObserver_onLogoAvailable(env, j_logo_observer_, j_logo,
from_cache);
}
void OnObserverRemoved() override { delete this; }
private:
// The associated LogoBridge. We won't call back to Java if the LogoBridge has
// been destroyed.
base::WeakPtr<LogoBridge> logo_bridge_;
base::android::ScopedJavaGlobalRef<jobject> j_logo_observer_;
DISALLOW_COPY_AND_ASSIGN(LogoObserverAndroid);
};
} // namespace
class LogoBridge::AnimatedLogoLoader {
public:
explicit AnimatedLogoLoader(
scoped_refptr<network::SharedURLLoaderFactory> factory)
: shared_url_loader_factory_(std::move(factory)) {}
void Start(JNIEnv* env,
const GURL& url,
const JavaParamRef<jobject>& j_callback) {
DCHECK(j_callback);
if (loader_ && url_ == url)
return;
url_ = url;
j_callback_.Reset(env, j_callback);
auto request = std::make_unique<network::ResourceRequest>();
request->url = url;
loader_ = network::SimpleURLLoader::Create(std::move(request),
NO_TRAFFIC_ANNOTATION_YET);
loader_->DownloadToStringOfUnboundedSizeUntilCrashAndDie(
shared_url_loader_factory_.get(),
base::BindOnce(&AnimatedLogoLoader::OnURLLoaderComplete,
base::Unretained(this)));
start_time_ = base::TimeTicks::Now();
}
private:
void OnURLLoaderComplete(std::unique_ptr<std::string> response_body) {
DCHECK(loader_);
DCHECK(!j_callback_.is_null());
int response_code = -1;
if (loader_->ResponseInfo() && loader_->ResponseInfo()->headers)
response_code = loader_->ResponseInfo()->headers->response_code();
if (!response_body || response_code != net::HTTP_OK) {
ClearLoader();
return;
}
UMA_HISTOGRAM_TIMES("NewTabPage.AnimatedLogoDownloadTime",
base::TimeTicks::Now() - start_time_);
JNIEnv* env = base::android::AttachCurrentThread();
ScopedJavaLocalRef<jbyteArray> j_bytes = ToJavaByteArray(
env, reinterpret_cast<const uint8_t*>(response_body->data()),
response_body->size());
ScopedJavaLocalRef<jobject> j_gif_image =
Java_LogoBridge_createGifImage(env, j_bytes);
Java_AnimatedLogoCallback_onAnimatedLogoAvailable(env, j_callback_,
j_gif_image);
ClearLoader();
}
void ClearLoader() {
loader_.reset();
j_callback_.Reset();
}
scoped_refptr<network::SharedURLLoaderFactory> shared_url_loader_factory_;
base::android::ScopedJavaGlobalRef<jobject> j_callback_;
// The original url used for the request.
GURL url_;
// The SimpleURLLoader currently fetching the animated logo, or nullptr when
// not fetching.
std::unique_ptr<network::SimpleURLLoader> loader_;
// The time when the current fetch was started.
base::TimeTicks start_time_;
DISALLOW_COPY_AND_ASSIGN(AnimatedLogoLoader);
};
static jlong JNI_LogoBridge_Init(JNIEnv* env,
const JavaParamRef<jobject>& obj,
const JavaParamRef<jobject>& j_profile) {
LogoBridge* logo_bridge = new LogoBridge(j_profile);
return reinterpret_cast<intptr_t>(logo_bridge);
}
LogoBridge::LogoBridge(const JavaRef<jobject>& j_profile)
: logo_service_(nullptr), weak_ptr_factory_(this) {
Profile* profile = ProfileAndroid::FromProfileAndroid(j_profile);
DCHECK(profile);
logo_service_ = LogoServiceFactory::GetForProfile(profile);
animated_logo_loader_ = std::make_unique<AnimatedLogoLoader>(
content::BrowserContext::GetDefaultStoragePartition(profile)
->GetURLLoaderFactoryForBrowserProcess());
}
LogoBridge::~LogoBridge() {}
void LogoBridge::Destroy(JNIEnv* env, const JavaParamRef<jobject>& obj) {
delete this;
}
void LogoBridge::GetCurrentLogo(JNIEnv* env,
const JavaParamRef<jobject>& obj,
const JavaParamRef<jobject>& j_logo_observer) {
// |observer| is deleted in LogoObserverAndroid::OnObserverRemoved().
LogoObserverAndroid* observer = new LogoObserverAndroid(
weak_ptr_factory_.GetWeakPtr(), env, j_logo_observer);
logo_service_->GetLogo(observer);
}
void LogoBridge::GetAnimatedLogo(JNIEnv* env,
const JavaParamRef<jobject>& obj,
const JavaParamRef<jobject>& j_callback,
const JavaParamRef<jstring>& j_url) {
GURL url = GURL(ConvertJavaStringToUTF8(env, j_url));
animated_logo_loader_->Start(env, url, j_callback);
}