Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(41)

Unified Diff: chrome/browser/ui/app_list/arc_app_icon.cc

Issue 1413153007: arc-app-launcher: Minimal support for ARC app launcher. (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: switch to new IPC Created 5 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: chrome/browser/ui/app_list/arc_app_icon.cc
diff --git a/chrome/browser/ui/app_list/arc_app_icon.cc b/chrome/browser/ui/app_list/arc_app_icon.cc
new file mode 100644
index 0000000000000000000000000000000000000000..e5371d4e655c5fc1d1abea78af61dd88ae65b6f4
--- /dev/null
+++ b/chrome/browser/ui/app_list/arc_app_icon.cc
@@ -0,0 +1,258 @@
+// Copyright 2015 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/ui/app_list/arc_app_icon.h"
+
+#include <algorithm>
+
+#include "base/bind.h"
+#include "base/files/file_path.h"
+#include "base/files/file_util.h"
+#include "chrome/browser/image_decoder.h"
+#include "chrome/browser/ui/app_list/arc_app_prefs.h"
+#include "content/public/browser/browser_thread.h"
+#include "extensions/grit/extensions_browser_resources.h"
+#include "ui/base/resource/resource_bundle.h"
+#include "ui/gfx/geometry/size.h"
+#include "ui/gfx/image/image_skia_source.h"
+
+////////////////////////////////////////////////////////////////////////////////
+// ArcAppIcon::Source
+
+class ArcAppIcon::Source : public gfx::ImageSkiaSource {
+ public:
+ Source(ArcAppIcon* host, const gfx::Size& size_in_dip);
xiyuan 2015/11/11 22:15:49 |size_in_dip| seems not used?
khmel1 2015/11/12 08:05:29 It is planned to use in future for new UI items bu
+ ~Source() override;
+
+ void ResetHost();
+
+ private:
+ // gfx::ImageSkiaSource overrides:
+ gfx::ImageSkiaRep GetImageForScale(float scale) override;
+
+ // Used to load images asynchronously. NULLed out when the ArcAppIcon is
+ // destroyed.
+ ArcAppIcon* host_;
xiyuan 2015/11/11 22:15:49 Do we want to use a WeakPtr<ArcAppIcon> here? This
khmel1 2015/11/12 08:05:29 Thanks for good advice! It is more elegant (I used
+
+ DISALLOW_COPY_AND_ASSIGN(Source);
+};
+
+ArcAppIcon::Source::Source(ArcAppIcon* host, const gfx::Size& size_in_dip)
+ : host_(host) {
+}
+
+ArcAppIcon::Source::~Source() {
+}
+
+void ArcAppIcon::Source::ResetHost() {
+ host_ = NULL;
xiyuan 2015/11/11 22:15:49 nit: NULL -> nullptr
khmel1 2015/11/12 08:05:29 Removed once WeakPtr is used.
+}
+
+gfx::ImageSkiaRep ArcAppIcon::Source::GetImageForScale(float scale) {
+ if (host_) {
+ host_->LoadForScaleFactor(ui::GetSupportedScaleFactor(scale));
+ }
+
+ return gfx::ImageSkiaRep();
+}
+
+class ArcAppIcon::DecodeRequest : public ImageDecoder::ImageRequest {
+ public:
+ DecodeRequest(ArcAppIcon* host, int dimension, ui::ScaleFactor scale_factor);
+ ~DecodeRequest() override;
+
+ void ResetHost();
+
+ // ImageDecoder::ImageRequest
+ void OnImageDecoded(const SkBitmap& bitmap) override;
+ void OnDecodeImageFailed() override;
+ private:
+ ArcAppIcon* host_;
+ int dimension_;
+ ui::ScaleFactor scale_factor_;
+
+ DISALLOW_COPY_AND_ASSIGN(DecodeRequest);
+};
+
+////////////////////////////////////////////////////////////////////////////////
+// ArcAppIcon::DecodeRequest
+
+ArcAppIcon::DecodeRequest::DecodeRequest(ArcAppIcon* host,
+ int dimension,
+ ui::ScaleFactor scale_factor)
+ : host_(host),
+ dimension_(dimension),
+ scale_factor_(scale_factor) {
+}
+
+ArcAppIcon::DecodeRequest::~DecodeRequest() {
+}
+
+void ArcAppIcon::DecodeRequest::ResetHost() {
+ host_ = NULL;
xiyuan 2015/11/11 22:15:49 nit: NULL -> nullptr
khmel1 2015/11/12 08:05:29 Done.
+}
+
+void ArcAppIcon::DecodeRequest::OnImageDecoded(const SkBitmap& bitmap) {
+ DCHECK(!bitmap.isNull() && !bitmap.empty());
+
+ if (!host_) {
+ return;
+ }
+
+ int expected_dim = static_cast<int>(
+ ui::GetScaleForScaleFactor(scale_factor_) * dimension_ + 0.5f);
+ if (bitmap.width() != expected_dim || bitmap.height() != expected_dim) {
+ LOG(ERROR) << "Decoded ARC icon has unexpected dimension "
+ << bitmap.width() << "x" << bitmap.height() << ". Expected "
+ << expected_dim << "x" << ".";
+ return;
+ }
+
+ gfx::ImageSkia image_skia;
+ image_skia.AddRepresentation(gfx::ImageSkiaRep(
+ bitmap,
+ ui::GetScaleForScaleFactor(scale_factor_)));
+
+ host_->Update(&image_skia);
+ host_->DiscardDecodeRequest(this);
+
+ host_->observer_->OnIconUpdated();
xiyuan 2015/11/11 22:15:49 Would this be a use-after-free? |this| should be d
khmel1 2015/11/12 08:05:29 Done.
+}
+
+void ArcAppIcon::DecodeRequest::OnDecodeImageFailed() {
+ LOG(ERROR) << "Failed to decode ARC icon.";
+
+ if (!host_) {
+ return;
+ }
+
+ host_->DiscardDecodeRequest(this);
+}
+
+////////////////////////////////////////////////////////////////////////////////
+// ArcAppIcon
+
+ArcAppIcon::ArcAppIcon(content::BrowserContext* context,
+ const std::string& app_id,
+ int resource_size_in_dip,
+ Observer* observer)
+ : context_(context),
+ app_id_(app_id),
+ resource_size_in_dip_(resource_size_in_dip),
+ observer_(observer),
+ source_(NULL),
xiyuan 2015/11/11 22:15:49 nit: think in-class member initialization is prefe
khmel1 2015/11/12 08:05:29 Done.
+ weak_ptr_factory_(this) {
+ CHECK(observer_ != NULL);
+ gfx::Size resource_size(resource_size_in_dip, resource_size_in_dip);
+ source_ = new Source(this, resource_size);
+ image_skia_ = gfx::ImageSkia(source_, resource_size);
+ Update(ResourceBundle::GetSharedInstance().GetImageSkiaNamed(
xiyuan 2015/11/11 22:15:49 This Update() call has some side effects. ImageSk
khmel1 2015/11/12 08:05:29 Yes, I also noticed this issue when was implementi
+ IDR_APP_DEFAULT_ICON));
+}
+
+ArcAppIcon::~ArcAppIcon() {
+ source_->ResetHost();
+ for (size_t i = 0; i < decode_requests_.size(); ++i) {
+ decode_requests_[i]->ResetHost();
+ ImageDecoder::Cancel(decode_requests_[i]);
xiyuan 2015/11/11 22:15:49 nit: Cancel() is not necessary here since ImageDec
khmel1 2015/11/12 08:05:29 Done.
+ }
+}
+
+void ArcAppIcon::LoadForScaleFactor(ui::ScaleFactor scale_factor) {
+ ArcAppPrefs* prefs = ArcAppPrefs::Get(context_);
+ if (!prefs) {
+ LOG(ERROR) << "ARC preferences service is not available.";
+ return;
+ }
+
+ base::FilePath path = prefs->GetIconPath(app_id_, scale_factor);
+ if (path.empty()) {
+ return;
+ }
+
+ base::Closure task = base::Bind(&ArcAppIcon::ReadOnFileThread,
+ base::Unretained(this),
xiyuan 2015/11/11 22:15:49 Should this be weak_ptr_factory_.GetWeakPtr() inst
khmel1 2015/11/12 08:05:29 Yes, it should
+ scale_factor,
+ path);
+ content::BrowserThread::PostTask(content::BrowserThread::FILE,
xiyuan 2015/11/11 22:15:49 File thread is deprecated (because it is only one
khmel1 2015/11/12 08:05:29 Good to know! Done
+ FROM_HERE,
+ task);
+}
+
+void ArcAppIcon::RequestIcon(ui::ScaleFactor scale_factor) {
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+ ArcAppPrefs* prefs = ArcAppPrefs::Get(context_);
+ if (!prefs) {
+ LOG(ERROR) << "ARC preferences service is not available.";
+ return;
+ }
+ prefs->RequestIcon(app_id_, scale_factor);
xiyuan 2015/11/11 22:15:49 How would we find out after ArcAppPrefs gets the i
khmel1 2015/11/12 08:05:29 I added comment.
+}
+
+void ArcAppIcon::ReadOnFileThread(ui::ScaleFactor scale_factor,
+ const base::FilePath& path) {
+ DCHECK_CURRENTLY_ON(content::BrowserThread::FILE);
+ DCHECK(!path.empty());
+
+ if (!base::PathExists(path)) {
+ base::Closure task = base::Bind(&ArcAppIcon::RequestIcon,
+ base::Unretained(this),
xiyuan 2015/11/11 22:15:49 weak_ptr_factory_.GetWeakPtr()?
khmel1 2015/11/12 08:05:29 Done.
+ scale_factor);
+ content::BrowserThread::PostTask(content::BrowserThread::UI,
+ FROM_HERE,
+ task);
+ return;
+ }
+
+ std::string unsafe_icon_data;
+ // Read the file from disk.
+ if (!base::ReadFileToString(path, &unsafe_icon_data)) {
+ LOG(ERROR) << "Failed to read an ARC icon from file "
+ << path.MaybeAsASCII();
+ return;
+ }
+
+ base::Closure task = base::Bind(&ArcAppIcon::OnIconRead,
+ base::Unretained(this),
xiyuan 2015/11/11 22:15:49 weak_ptr_factory_.GetWeakPtr()?
khmel1 2015/11/12 08:05:29 Done.
+ scale_factor,
+ unsafe_icon_data);
+ content::BrowserThread::PostTask(content::BrowserThread::UI,
+ FROM_HERE,
+ task);
+}
+
+void ArcAppIcon::OnIconRead(ui::ScaleFactor scale_factor,
+ const std::string& unsafe_image_data) {
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+
+ decode_requests_.push_back(new DecodeRequest(this,
+ resource_size_in_dip_,
+ scale_factor));
xiyuan 2015/11/11 22:15:49 nit: align with previous line.
khmel1 2015/11/12 08:05:29 Done.
+ ImageDecoder::Start(decode_requests_.back(), unsafe_image_data);
+}
+
+void ArcAppIcon::Update(const gfx::ImageSkia* image) {
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+ CHECK(image && !image->isNull());
+
+ std::vector<gfx::ImageSkiaRep> reps = image->image_reps();
+ for (const auto& image_rep : reps) {
+ if (ui::IsSupportedScale(image_rep.scale())) {
+ image_skia_.RemoveRepresentation(image_rep.scale());
+ image_skia_.AddRepresentation(image_rep);
+ }
+ }
+
+ image_ = gfx::Image(image_skia_);
+}
+
+void ArcAppIcon::DiscardDecodeRequest(DecodeRequest* request) {
+ DCHECK_CURRENTLY_ON(content::BrowserThread::UI);
+
+ ScopedVector<DecodeRequest>::iterator it = std::find(decode_requests_.begin(),
+ decode_requests_.end(),
+ request);
+ CHECK(it != decode_requests_.end());
+ decode_requests_.erase(it);
+}

Powered by Google App Engine
This is Rietveld 408576698