Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright 2015 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "chrome/browser/ui/app_list/arc/arc_app_icon.h" | |
| 6 | |
| 7 #include <algorithm> | |
| 8 | |
| 9 #include "base/bind.h" | |
| 10 #include "base/files/file_path.h" | |
| 11 #include "base/files/file_util.h" | |
| 12 #include "base/task_runner_util.h" | |
| 13 #include "chrome/browser/image_decoder.h" | |
| 14 #include "chrome/browser/ui/app_list/arc/arc_app_list_prefs.h" | |
| 15 #include "content/public/browser/browser_thread.h" | |
| 16 #include "extensions/grit/extensions_browser_resources.h" | |
| 17 #include "ui/base/resource/resource_bundle.h" | |
| 18 #include "ui/gfx/geometry/size.h" | |
| 19 #include "ui/gfx/image/image_skia_source.h" | |
| 20 | |
| 21 namespace { | |
| 22 | |
| 23 bool g_disable_decoding = false; | |
| 24 | |
| 25 } | |
|
jochen (gone - plz use gerrit)
2015/12/03 12:55:52
} // namespace
khmel1
2015/12/03 14:12:41
Done.
| |
| 26 | |
| 27 //////////////////////////////////////////////////////////////////////////////// | |
| 28 // ArcAppIcon::ReadResult | |
| 29 | |
| 30 struct ArcAppIcon::ReadResult { | |
| 31 enum Status { | |
|
hidehiko
2015/12/03 13:08:39
enum class ?
khmel1
2015/12/03 14:12:41
Done.
| |
| 32 OK, | |
| 33 FAIL, | |
| 34 REQUEST_TO_INSTALL, | |
| 35 }; | |
| 36 | |
| 37 ReadResult(Status status, ui::ScaleFactor scale_factor) | |
| 38 : status(status), scale_factor(scale_factor) { | |
| 39 } | |
| 40 | |
| 41 Status status; | |
| 42 ui::ScaleFactor scale_factor; | |
| 43 std::string unsafe_icon_data; | |
| 44 }; | |
| 45 | |
| 46 //////////////////////////////////////////////////////////////////////////////// | |
| 47 // ArcAppIcon::Source | |
| 48 | |
| 49 class ArcAppIcon::Source : public gfx::ImageSkiaSource { | |
| 50 public: | |
| 51 explicit Source(const base::WeakPtr<ArcAppIcon>& host); | |
| 52 ~Source() override; | |
| 53 | |
| 54 private: | |
| 55 // gfx::ImageSkiaSource overrides: | |
| 56 gfx::ImageSkiaRep GetImageForScale(float scale) override; | |
| 57 | |
| 58 // Used to load images asynchronously. NULLed out when the ArcAppIcon is | |
| 59 // destroyed. | |
| 60 base::WeakPtr<ArcAppIcon> host_; | |
| 61 | |
| 62 DISALLOW_COPY_AND_ASSIGN(Source); | |
| 63 }; | |
| 64 | |
| 65 ArcAppIcon::Source::Source(const base::WeakPtr<ArcAppIcon>& host) | |
| 66 : host_(host) { | |
|
hidehiko
2015/12/03 13:08:39
4 indent.
khmel1
2015/12/03 14:12:41
Done.
| |
| 67 } | |
| 68 | |
| 69 ArcAppIcon::Source::~Source() { | |
| 70 } | |
| 71 | |
| 72 gfx::ImageSkiaRep ArcAppIcon::Source::GetImageForScale(float scale) { | |
| 73 if (host_) | |
| 74 host_->LoadForScaleFactor(ui::GetSupportedScaleFactor(scale)); | |
| 75 | |
| 76 // Host loads icon asynchronously, so use default icon so far. | |
| 77 const gfx::ImageSkia* default_image = ResourceBundle::GetSharedInstance(). | |
| 78 GetImageSkiaNamed(IDR_APP_DEFAULT_ICON); | |
| 79 CHECK(default_image); | |
| 80 | |
| 81 return default_image->GetRepresentation(scale); | |
| 82 } | |
| 83 | |
| 84 class ArcAppIcon::DecodeRequest : public ImageDecoder::ImageRequest { | |
| 85 public: | |
| 86 DecodeRequest(const base::WeakPtr<ArcAppIcon>& host, | |
| 87 int dimension, | |
| 88 ui::ScaleFactor scale_factor); | |
| 89 ~DecodeRequest() override; | |
| 90 | |
| 91 // ImageDecoder::ImageRequest | |
| 92 void OnImageDecoded(const SkBitmap& bitmap) override; | |
| 93 void OnDecodeImageFailed() override; | |
| 94 private: | |
| 95 base::WeakPtr<ArcAppIcon> host_; | |
| 96 int dimension_; | |
| 97 ui::ScaleFactor scale_factor_; | |
| 98 | |
| 99 DISALLOW_COPY_AND_ASSIGN(DecodeRequest); | |
| 100 }; | |
| 101 | |
| 102 //////////////////////////////////////////////////////////////////////////////// | |
| 103 // ArcAppIcon::DecodeRequest | |
| 104 | |
| 105 ArcAppIcon::DecodeRequest::DecodeRequest(const base::WeakPtr<ArcAppIcon>& host, | |
| 106 int dimension, | |
| 107 ui::ScaleFactor scale_factor) | |
| 108 : host_(host), | |
|
hidehiko
2015/12/03 13:08:39
Ditto.
khmel1
2015/12/03 14:12:42
Done.
| |
| 109 dimension_(dimension), | |
| 110 scale_factor_(scale_factor) { | |
| 111 } | |
| 112 | |
| 113 ArcAppIcon::DecodeRequest::~DecodeRequest() { | |
| 114 } | |
| 115 | |
| 116 void ArcAppIcon::DecodeRequest::OnImageDecoded(const SkBitmap& bitmap) { | |
| 117 DCHECK(!bitmap.isNull() && !bitmap.empty()); | |
| 118 | |
| 119 if (!host_) { | |
| 120 return; | |
|
jochen (gone - plz use gerrit)
2015/12/03 12:55:52
no {} for single line bodies, here and everywhere
khmel1
2015/12/03 14:12:42
Done.
| |
| 121 } | |
| 122 | |
| 123 int expected_dim = static_cast<int>( | |
| 124 ui::GetScaleForScaleFactor(scale_factor_) * dimension_ + 0.5f); | |
| 125 if (bitmap.width() != expected_dim || bitmap.height() != expected_dim) { | |
| 126 VLOG(2) << "Decoded ARC icon has unexpected dimension " | |
| 127 << bitmap.width() << "x" << bitmap.height() << ". Expected " | |
| 128 << expected_dim << "x" << "."; | |
| 129 return; | |
|
jochen (gone - plz use gerrit)
2015/12/03 12:55:51
don't you need to discard the request here as well
khmel1
2015/12/03 14:12:41
Yes, thanks for catching. In case something wrong
| |
| 130 } | |
| 131 | |
| 132 gfx::ImageSkia image_skia; | |
| 133 image_skia.AddRepresentation(gfx::ImageSkiaRep( | |
| 134 bitmap, | |
| 135 ui::GetScaleForScaleFactor(scale_factor_))); | |
| 136 | |
| 137 host_->Update(&image_skia); | |
| 138 host_->DiscardDecodeRequest(this); | |
| 139 } | |
| 140 | |
| 141 void ArcAppIcon::DecodeRequest::OnDecodeImageFailed() { | |
| 142 VLOG(2) << "Failed to decode ARC icon."; | |
| 143 | |
| 144 if (!host_) | |
| 145 return; | |
| 146 | |
| 147 host_->DiscardDecodeRequest(this); | |
| 148 } | |
| 149 | |
| 150 //////////////////////////////////////////////////////////////////////////////// | |
| 151 // ArcAppIcon | |
| 152 | |
| 153 // static | |
| 154 void ArcAppIcon::DisableDecodingForTesting() { | |
| 155 g_disable_decoding = true; | |
| 156 } | |
| 157 | |
| 158 ArcAppIcon::ArcAppIcon(content::BrowserContext* context, | |
| 159 const std::string& app_id, | |
| 160 int resource_size_in_dip, | |
| 161 Observer* observer) | |
| 162 : context_(context), | |
| 163 app_id_(app_id), | |
| 164 resource_size_in_dip_(resource_size_in_dip), | |
| 165 observer_(observer), | |
| 166 weak_ptr_factory_(this) { | |
| 167 CHECK(observer_ != NULL); | |
| 168 source_ = new Source(weak_ptr_factory_.GetWeakPtr()); | |
| 169 gfx::Size resource_size(resource_size_in_dip, resource_size_in_dip); | |
| 170 image_skia_ = gfx::ImageSkia(source_, resource_size); | |
| 171 } | |
| 172 | |
| 173 ArcAppIcon::~ArcAppIcon() { | |
| 174 } | |
| 175 | |
| 176 void ArcAppIcon::LoadForScaleFactor(ui::ScaleFactor scale_factor) { | |
| 177 ArcAppListPrefs* prefs = ArcAppListPrefs::Get(context_); | |
| 178 if (!prefs) { | |
| 179 VLOG(2) << "ARC preferences service is not available."; | |
|
jochen (gone - plz use gerrit)
2015/12/03 12:55:52
how can this ever happen?
khmel1
2015/12/03 14:12:42
It is impossible. I removed check here for consist
| |
| 180 return; | |
| 181 } | |
| 182 | |
| 183 base::FilePath path = prefs->GetIconPath(app_id_, scale_factor); | |
| 184 if (path.empty()) | |
| 185 return; | |
| 186 | |
| 187 base::PostTaskAndReplyWithResult(content::BrowserThread::GetBlockingPool(), | |
| 188 FROM_HERE, | |
| 189 base::Bind(&ArcAppIcon::ReadOnFileThread, | |
| 190 scale_factor, | |
| 191 path), | |
| 192 base::Bind(&ArcAppIcon::OnIconRead, | |
| 193 weak_ptr_factory_.GetWeakPtr())); | |
| 194 } | |
| 195 | |
| 196 void ArcAppIcon::RequestIcon(ui::ScaleFactor scale_factor) { | |
| 197 DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | |
| 198 ArcAppListPrefs* prefs = ArcAppListPrefs::Get(context_); | |
| 199 if (!prefs) { | |
| 200 VLOG(2) << "ARC preferences service is not available."; | |
| 201 return; | |
| 202 } | |
| 203 // ArcAppListPrefs notifies ArcAppModelBuilder via Observer when icon is ready | |
| 204 // and ArcAppModelBuilder refreshes the icon of the corresponding item by | |
| 205 // calling LoadScaleFactor. | |
| 206 prefs->RequestIcon(app_id_, scale_factor); | |
| 207 } | |
| 208 | |
| 209 // static | |
| 210 scoped_ptr<ArcAppIcon::ReadResult> ArcAppIcon::ReadOnFileThread( | |
| 211 ui::ScaleFactor scale_factor, | |
| 212 const base::FilePath& path) { | |
| 213 DCHECK(content::BrowserThread::GetBlockingPool()->RunsTasksOnCurrentThread()); | |
| 214 DCHECK(!path.empty()); | |
| 215 | |
| 216 scoped_ptr<ArcAppIcon::ReadResult> result(new ArcAppIcon::ReadResult( | |
|
hidehiko
2015/12/03 13:08:39
I think it's better to inline to the return statem
khmel1
2015/12/03 14:12:42
Good point, thanks
| |
| 217 ArcAppIcon::ReadResult::OK, scale_factor)); | |
| 218 | |
| 219 if (!base::PathExists(path)) { | |
| 220 result->status = ArcAppIcon::ReadResult::REQUEST_TO_INSTALL; | |
| 221 return result.Pass(); | |
| 222 } | |
| 223 | |
| 224 // Read the file from disk. | |
| 225 if (!base::ReadFileToString(path, &result->unsafe_icon_data)) { | |
| 226 VLOG(2) << "Failed to read an ARC icon from file " << path.MaybeAsASCII(); | |
| 227 result->status = ArcAppIcon::ReadResult::FAIL; | |
| 228 return result.Pass(); | |
| 229 } | |
| 230 | |
| 231 return result.Pass(); | |
| 232 } | |
| 233 | |
| 234 void ArcAppIcon::OnIconRead(scoped_ptr<ArcAppIcon::ReadResult> read_result) { | |
| 235 DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | |
| 236 | |
| 237 switch (read_result->status) { | |
| 238 case ReadResult::OK: | |
| 239 if (!g_disable_decoding) { | |
| 240 decode_requests_.push_back( | |
| 241 new DecodeRequest(weak_ptr_factory_.GetWeakPtr(), | |
| 242 resource_size_in_dip_, | |
| 243 read_result->scale_factor)); | |
| 244 ImageDecoder::Start(decode_requests_.back(), | |
| 245 read_result->unsafe_icon_data); | |
| 246 } | |
| 247 break; | |
| 248 case ReadResult::FAIL: | |
| 249 break; | |
| 250 case ReadResult::REQUEST_TO_INSTALL: | |
| 251 RequestIcon(read_result->scale_factor); | |
| 252 break; | |
| 253 default: | |
| 254 NOTREACHED(); | |
| 255 } | |
| 256 } | |
| 257 | |
| 258 void ArcAppIcon::Update(const gfx::ImageSkia* image) { | |
| 259 DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | |
| 260 CHECK(image && !image->isNull()); | |
| 261 | |
| 262 std::vector<gfx::ImageSkiaRep> reps = image->image_reps(); | |
| 263 for (const auto& image_rep : reps) { | |
| 264 if (ui::IsSupportedScale(image_rep.scale())) { | |
| 265 image_skia_.RemoveRepresentation(image_rep.scale()); | |
| 266 image_skia_.AddRepresentation(image_rep); | |
| 267 } | |
| 268 } | |
| 269 | |
| 270 image_ = gfx::Image(image_skia_); | |
| 271 | |
| 272 observer_->OnIconUpdated(); | |
| 273 } | |
| 274 | |
| 275 void ArcAppIcon::DiscardDecodeRequest(DecodeRequest* request) { | |
| 276 DCHECK_CURRENTLY_ON(content::BrowserThread::UI); | |
| 277 | |
| 278 ScopedVector<DecodeRequest>::iterator it = std::find(decode_requests_.begin(), | |
| 279 decode_requests_.end(), | |
| 280 request); | |
| 281 CHECK(it != decode_requests_.end()); | |
| 282 decode_requests_.erase(it); | |
| 283 } | |
| OLD | NEW |