chromium/chromeos/ash/components/phonehub/icon_decoder_impl.cc

// Copyright 2022 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "chromeos/ash/components/phonehub/icon_decoder_impl.h"

#include <functional>
#include <utility>

#include "base/barrier_closure.h"
#include "base/containers/flat_map.h"
#include "base/functional/bind.h"
#include "base/functional/callback.h"
#include "base/functional/callback_forward.h"
#include "base/memory/weak_ptr.h"
#include "base/strings/string_number_conversions.h"
#include "chromeos/ash/components/multidevice/logging/logging.h"
#include "chromeos/ash/components/phonehub/proto/phonehub_api.pb.h"
#include "services/data_decoder/public/cpp/decode_image.h"
#include "services/data_decoder/public/mojom/image_decoder.mojom.h"
#include "ubidiimp.h"
#include "ui/gfx/image/image.h"
#include "ui/gfx/image/image_skia.h"

namespace ash {
namespace phonehub {

IconDecoderImpl::DecoderDelegate::DecoderDelegate() = default;

IconDecoderImpl::DecoderDelegate::~DecoderDelegate() = default;

void IconDecoderImpl::DecoderDelegate::Decode(
    const DecodingData& request,
    data_decoder::DecodeImageCallback callback) {
  const std::string& encoded_icon = *request.input_data;
  data_decoder::DecodeImage(
      &data_decoder_, base::as_bytes(base::make_span(encoded_icon)),
      data_decoder::mojom::ImageCodec::kDefault,
      /*shrink_to_fit=*/true, data_decoder::kDefaultMaxSizeInBytes,
      /*desired_image_frame_size=*/gfx::Size(), std::move(callback));
}

IconDecoderImpl::IconDecoderImpl()
    : decoder_delegate_(std::make_unique<DecoderDelegate>()) {}

IconDecoderImpl::~IconDecoderImpl() = default;

void IconDecoderImpl::BatchDecode(
    std::unique_ptr<std::vector<DecodingData>> decode_items,
    base::OnceCallback<void(std::unique_ptr<std::vector<DecodingData>>)>
        finished_callback) {
  CancelPendingRequests();
  pending_items_ = std::move(decode_items);

  barrier_closure_ =
      base::BarrierClosure(pending_items_->size(),
                           base::BindOnce(&IconDecoderImpl::OnAllIconsDecoded,
                                          weak_ptr_factory_.GetWeakPtr(),
                                          std::move(finished_callback)));

  // If decode_items is empty, barrier closure must have been called by this
  // point and the pending_items_ pointer must be already reset.
  if (!pending_items_)
    return;
  for (DecodingData& request : *pending_items_) {
    decoder_delegate_->Decode(
        request,
        base::BindOnce(&IconDecoderImpl::OnIconDecoded,
                       weak_ptr_factory_.GetWeakPtr(), std::ref(request)));
  }
}

void IconDecoderImpl::OnAllIconsDecoded(
    base::OnceCallback<void(std::unique_ptr<std::vector<DecodingData>>)>
        finished_callback) {
  std::move(finished_callback).Run(std::move(pending_items_));
}

void IconDecoderImpl::OnIconDecoded(DecodingData& decoding_data,
                                    const SkBitmap& result) {
  gfx::ImageSkia image_skia = gfx::ImageSkia::CreateFrom1xBitmap(result);

  // If |image_skia| is null, indicating that the data decoder failed to decode
  // the image, the image will be empty, and cannot be made thread safe.
  if (!image_skia.isNull())
    image_skia.MakeThreadSafe();

  decoding_data.result = gfx::Image(image_skia);
  barrier_closure_.Run();
}

void IconDecoderImpl::CancelPendingRequests() {
  weak_ptr_factory_.InvalidateWeakPtrs();
}

}  // namespace phonehub
}  // namespace ash