Chromium Code Reviews| Index: chrome/browser/extensions/chrome_app_icon.cc |
| diff --git a/chrome/browser/extensions/chrome_app_icon.cc b/chrome/browser/extensions/chrome_app_icon.cc |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..c5dd0515cf08e8135ce8ada452f16110e68c7cac |
| --- /dev/null |
| +++ b/chrome/browser/extensions/chrome_app_icon.cc |
| @@ -0,0 +1,137 @@ |
| +// Copyright 2017 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/extensions/chrome_app_icon.h" |
| + |
| +#include <algorithm> |
| + |
| +#include "chrome/browser/extensions/chrome_app_icon_delegate.h" |
| +#include "chrome/browser/extensions/extension_util.h" |
| +#include "extensions/browser/extension_registry.h" |
| +#include "extensions/common/manifest_handlers/icons_handler.h" |
| +#include "ui/gfx/canvas.h" |
| +#include "ui/gfx/geometry/rect.h" |
| +#include "ui/gfx/image/canvas_image_source.h" |
| +#include "ui/gfx/image/image_skia_operations.h" |
| + |
| +#if defined(OS_CHROMEOS) |
| +#include "chrome/browser/chromeos/extensions/gfx_utils.h" |
| +#endif |
| + |
| +namespace extensions { |
| + |
| +namespace { |
| + |
| +gfx::ImageSkia CreateDisabledIcon(const gfx::ImageSkia& icon) { |
|
msw
2017/05/03 01:17:37
nit: inline for the one caller.
khmel
2017/05/03 02:15:21
There are several places in code with dup function
|
| + const color_utils::HSL shift = {-1, 0, 0.6}; |
| + return gfx::ImageSkiaOperations::CreateHSLShiftedImage(icon, shift); |
| +} |
| + |
| +// Rounds the corners of a given image. |
| +class RoundedCornersImageSource : public gfx::CanvasImageSource { |
|
msw
2017/05/03 01:17:38
Can we avoid subclassing CanvasImageSource and jus
khmel
2017/05/03 02:15:21
Would TODO here be acceptable? That was legacy cod
msw
2017/05/08 19:59:53
Yes, since you are just moving this code, it's fin
khmel
2017/05/09 00:05:04
TODO added.
|
| + public: |
| + explicit RoundedCornersImageSource(const gfx::ImageSkia& icon) |
| + : gfx::CanvasImageSource(icon.size(), false), icon_(icon) {} |
| + ~RoundedCornersImageSource() override {} |
| + |
| + private: |
| + // gfx::CanvasImageSource overrides: |
| + void Draw(gfx::Canvas* canvas) override { |
| + // The radius used to round the app icon, based on 2 pixel per 48 pixels |
| + // icon size. |
| + const int rounding_radius = |
| + std::max(static_cast<int>(std::round(2.0 * icon_.width() / 48.0)), 1); |
|
msw
2017/05/03 01:17:38
int: use std::max<int> and avoid the cast, or use
khmel
2017/05/03 02:15:21
Done
|
| + |
| + canvas->DrawImageInt(icon_, 0, 0); |
| + |
| + cc::PaintFlags masking_flags; |
| + masking_flags.setBlendMode(SkBlendMode::kDstIn); |
| + canvas->SaveLayerWithFlags(masking_flags); |
| + |
| + cc::PaintFlags mask_flags; |
| + mask_flags.setAntiAlias(true); |
| + mask_flags.setColor(SK_ColorWHITE); |
| + canvas->DrawRoundRect(gfx::Rect(icon_.width(), icon_.height()), |
| + rounding_radius, mask_flags); |
| + |
| + canvas->Restore(); |
| + } |
| + |
| + gfx::ImageSkia icon_; |
| + |
| + DISALLOW_COPY_AND_ASSIGN(RoundedCornersImageSource); |
| +}; |
| + |
| +} // namespace |
| + |
| +ChromeAppIcon::ChromeAppIcon(ChromeAppIconDelegate* delegate, |
| + content::BrowserContext* context, |
| + DestroyedCallback destroyed_callback, |
| + const std::string& app_id, |
| + int resource_size_in_dip) |
| + : delegate_(delegate), |
| + context_(context), |
| + destroyed_callback_(std::move(destroyed_callback)), |
| + app_id_(app_id), |
| + resource_size_in_dip_(resource_size_in_dip) { |
| + DCHECK(delegate_); |
| + DCHECK(context_); |
| + DCHECK(!destroyed_callback_.is_null()); |
| + DCHECK_GE(resource_size_in_dip, 0); |
| + Reload(); |
| +} |
| + |
| +ChromeAppIcon::~ChromeAppIcon() { |
| + std::move(destroyed_callback_).Run(this); |
|
msw
2017/05/03 01:17:38
Why do you need to move the callback to run it?
khmel
2017/05/03 02:15:21
I was requested to use base::OnceCallback and this
msw
2017/05/08 19:59:53
Acknowledged.
|
| +} |
| + |
| +const Extension* ChromeAppIcon::GetExtension() { |
| + const Extension* extension = |
|
msw
2017/05/03 01:17:38
nit: inline this in the return statement.
khmel
2017/05/03 02:15:21
Done.
|
| + ExtensionRegistry::Get(context_)->GetInstalledExtension(app_id_); |
| + return extension; |
| +} |
| + |
| +bool ChromeAppIcon::IsValid() const { |
| + return icon_ && icon_->is_valid(); |
| +} |
| + |
| +void ChromeAppIcon::Reload() { |
| + const Extension* extension = GetExtension(); |
|
msw
2017/05/03 01:17:38
nit: inline in call below
khmel
2017/05/03 02:15:21
It is used twice below, extension, IconsInfo::GetI
|
| + icon_ = base::MakeUnique<IconImage>( |
| + context_, extension, IconsInfo::GetIcons(extension), |
| + resource_size_in_dip_, util::GetDefaultAppIcon(), this); |
| + UpdateIcon(); |
| +} |
| + |
| +void ChromeAppIcon::UpdateIcon() { |
| + DCHECK(icon_); |
| + |
| + image_skia_ = icon_->image_skia(); |
| +#if defined(OS_CHROMEOS) |
| + util::MaybeApplyChromeBadge(context_, app_id_, &image_skia_); |
| +#endif |
| + |
| + const bool enabled = util::IsAppLaunchable(app_id_, context_); |
|
msw
2017/05/03 01:17:37
nit: inline in conditional
khmel
2017/05/03 02:15:21
Done.
|
| + if (!enabled) |
| + image_skia_ = CreateDisabledIcon(image_skia_); |
| + |
| + const Extension* extension = GetExtension(); |
| + if (extension && extension->from_bookmark()) |
| + image_skia_ = gfx::ImageSkia(new RoundedCornersImageSource(image_skia_), |
| + image_skia_.size()); |
| + |
| + delegate_->OnIconUpdated(this); |
| +} |
| + |
| +void ChromeAppIcon::EnsureRepsForSupportedScales() { |
| + DCHECK(icon_); |
| + icon_->image_skia().EnsureRepsForSupportedScales(); |
| +} |
| + |
| +void ChromeAppIcon::OnExtensionIconImageChanged(IconImage* icon) { |
| + DCHECK_EQ(icon_.get(), icon); |
| + UpdateIcon(); |
| +} |
| + |
| +} // namespace extensions |