chromium/components/exo/data_offer.h

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

#ifndef COMPONENTS_EXO_DATA_OFFER_H_
#define COMPONENTS_EXO_DATA_OFFER_H_

#include <cstdint>
#include <string>

#include "base/containers/flat_map.h"
#include "base/containers/flat_set.h"
#include "base/files/scoped_file.h"
#include "base/functional/callback_forward.h"
#include "base/memory/raw_ptr.h"
#include "base/memory/ref_counted.h"
#include "base/observer_list.h"
#include "ui/base/class_property.h"
#include "url/gurl.h"

namespace aura {
class Window;
}

namespace base {
class Pickle;
class RefCountedMemory;
}

namespace ui {
class Clipboard;
class OSExchangeData;
enum class EndpointType;
}

namespace exo {

class DataOfferDelegate;
class DataOfferObserver;
class DataExchangeDelegate;
enum class DndAction;

// Object representing transferred data offered to a client.
class DataOffer final : public ui::PropertyHandler {
 public:
  using SendDataCallback =
      base::OnceCallback<void(scoped_refptr<base::RefCountedMemory>)>;
  using AsyncSendDataCallback = base::OnceCallback<void(SendDataCallback)>;

  DataOffer(DataOfferDelegate* delegate);

  DataOffer(const DataOffer&) = delete;
  DataOffer& operator=(const DataOffer&) = delete;

  ~DataOffer() override;

  void AddObserver(DataOfferObserver* observer);
  void RemoveObserver(DataOfferObserver* observer);

  // Notifies to the DataOffer that the client can accept |mime type|.
  void Accept(const std::string* mime_type);

  // Notifies to the DataOffer that the client start receiving data of
  // |mime_type|. DataOffer writes the request data to |fd|.
  void Receive(const std::string& mime_type, base::ScopedFD fd);

  // Notifies to the DataOffer that the client no longer uses the DataOffer
  // object.
  void Finish();

  // Notifies to the DataOffer that possible and preferred drag and drop
  // operations selected by the client.
  void SetActions(const base::flat_set<DndAction>& dnd_actions,
                  DndAction preferred_action);

  // Sets the dropped data from |data| to the DataOffer object.
  // |data_exchange_delegate| will be used to convert paths to handle mount
  // points which is mounted in the mount point namespace of client process.
  // |target| is the drop target window and can be used to apply the target
  // specitic logic to interpret the data. While this function immediately calls
  // DataOfferDelegate::OnOffer inside it with found mime types, dropped data
  // bytes may be populated asynchronously after this function call. (e.g.
  // Asynchronous lookup is required for resolving file system urls.)
  void SetDropData(DataExchangeDelegate* data_exchange_delegate,
                   aura::Window* target,
                   const ui::OSExchangeData& data);

  // Sets the clipboard data from |data| to the DataOffer object.
  void SetClipboardData(DataExchangeDelegate* data_exchange_delegate,
                        const ui::Clipboard& data,
                        ui::EndpointType endpoint_type);

  // Sets the drag and drop actions which is offered by data source to the
  // DataOffer object.
  void SetSourceActions(const base::flat_set<DndAction>& source_actions);

  DndAction dnd_action() const { return dnd_action_; }
  bool finished() const { return finished_; }

 private:
  void OnDataReady(const std::string& mime_type,
                   base::ScopedFD fd,
                   scoped_refptr<base::RefCountedMemory> data);
  void GetUrlsFromPickle(DataExchangeDelegate* data_exchange_delegate,
                         aura::Window* target,
                         const base::Pickle& pickle,
                         SendDataCallback callback);
  void OnPickledUrlsResolved(SendDataCallback callback,
                             const std::vector<GURL>& urls);

  const raw_ptr<DataOfferDelegate, DanglingUntriaged> delegate_;

  // Data for a given mime type may not ever be requested, or may be requested
  // more than once. Using callbacks and a cache allows us to delay any
  // expensive operations until they are required, and then ensure that they are
  // performed at most once. When we offer data for a given mime type we will
  // populate |data_callbacks_| with mime type and a callback which will produce
  // the required data. On the first request to |Receive()| we remove and invoke
  // the callback and set |data_cache_| with null data. When the callback
  // completes we populate |data_cache_| with data and fulfill any
  // |pending_receive_requests|.
  base::flat_map<std::string, AsyncSendDataCallback> data_callbacks_;
  base::flat_map<std::string, scoped_refptr<base::RefCountedMemory>>
      data_cache_;
  std::vector<std::pair<std::string, base::ScopedFD>> pending_receive_requests_;

  base::flat_set<DndAction> source_actions_;
  DndAction dnd_action_;
  base::ObserverList<DataOfferObserver>::Unchecked observers_;
  bool finished_;

  base::WeakPtrFactory<DataOffer> weak_ptr_factory_{this};
};

class ScopedDataOffer {
 public:
  ScopedDataOffer(DataOffer* data_offer, DataOfferObserver* observer);

  ScopedDataOffer(const ScopedDataOffer&) = delete;
  ScopedDataOffer& operator=(const ScopedDataOffer&) = delete;

  ~ScopedDataOffer();
  DataOffer* get() { return data_offer_; }

 private:
  const raw_ptr<DataOffer> data_offer_;
  const raw_ptr<DataOfferObserver> observer_;
};

}  // namespace exo

#endif  // COMPONENTS_EXO_DATA_OFFER_H_