chromium/components/exo/wayland/wp_fractional_scale.cc

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

#include "components/exo/wayland/wp_fractional_scale.h"

#include "base/memory/raw_ptr.h"
#include "components/exo/surface.h"
#include "components/exo/surface_observer.h"
#include "components/exo/wayland/server_util.h"

namespace exo::wayland {
namespace {

// A property key containing a boolean set to true if a fractional scale object
// is associated with with surface object.
DEFINE_UI_CLASS_PROPERTY_KEY(bool, kSurfaceHasFractionalScaleKey, false)

////////////////////////////////////////////////////////////////////////////////
// wp_fractional_scale_v1_interface:

// Implements the fractional scale interface to a Surface. The
// "fractional-scale"-state is set to null upon destruction. A window property
// will be set during the lifetime of this class to prevent multiple instances
// from being created for the same Surface.
class FractionalScale : public SurfaceObserver {
 public:
  explicit FractionalScale(Surface* surface, wl_resource* resource)
      : surface_(surface), resource_(resource) {
    surface_->AddSurfaceObserver(this);
    surface_->SetProperty(kSurfaceHasFractionalScaleKey, true);
    SendPreferredScale(0.0, surface_->GetDisplay().device_scale_factor());
  }
  FractionalScale(const FractionalScale&) = delete;
  FractionalScale& operator=(const FractionalScale&) = delete;
  ~FractionalScale() override {
    if (surface_) {
      surface_->RemoveSurfaceObserver(this);
      surface_->SetProperty(kSurfaceHasFractionalScaleKey, false);
    }
  }

  void SendPreferredScale(float old_scale_factor, float new_scale_factor) {
    uint32_t old_wire_scale = round(old_scale_factor * 120);
    uint32_t new_wire_scale = round(new_scale_factor * 120);
    DCHECK(new_wire_scale > 0);

    if (new_wire_scale != old_wire_scale) {
      wp_fractional_scale_v1_send_preferred_scale(resource_, new_wire_scale);
    }
  }

  void OnScaleFactorChanged(Surface* surface,
                            float old_scale_factor,
                            float new_scale_factor) override {
    SendPreferredScale(old_scale_factor, new_scale_factor);
  }

  void OnSurfaceDestroying(Surface* surface) override {
    surface->RemoveSurfaceObserver(this);
    surface_ = nullptr;
  }

 private:
  raw_ptr<Surface> surface_;
  const raw_ptr<wl_resource> resource_;
};

void wp_fractional_scale_destroy(struct wl_client* client,
                                 struct wl_resource* resource) {
  wl_resource_destroy(resource);
}

const struct wp_fractional_scale_v1_interface fractional_scale_implementation =
    {
        wp_fractional_scale_destroy,
};

////////////////////////////////////////////////////////////////////////////////
// wp_fractional_scale_manager_v1_interface:

void wp_fractional_scale_manager_destroy(struct wl_client* client,
                                         struct wl_resource* resource) {
  wl_resource_destroy(resource);
}

static void wp_fractional_scale_manager_get_fractional_scale(
    wl_client* client,
    wl_resource* resource,
    uint32_t id,
    wl_resource* surface_resource) {
  Surface* surface = GetUserDataAs<Surface>(surface_resource);
  if (surface->GetProperty(kSurfaceHasFractionalScaleKey)) {
    wl_resource_post_error(
        resource, WP_FRACTIONAL_SCALE_MANAGER_V1_ERROR_FRACTIONAL_SCALE_EXISTS,
        "a fractional scale object for that surface already exists");
    return;
  }

  wl_resource* fractional_scale_resource =
      wl_resource_create(client, &wp_fractional_scale_v1_interface,
                         wl_resource_get_version(resource), id);
  auto fractional_scale =
      std::make_unique<FractionalScale>(surface, fractional_scale_resource);
  SetImplementation(fractional_scale_resource, &fractional_scale_implementation,
                    std::move(fractional_scale));
}

static const struct wp_fractional_scale_manager_v1_interface
    fractional_scale_manager_implementation = {
        wp_fractional_scale_manager_destroy,
        wp_fractional_scale_manager_get_fractional_scale,
};

}  // namespace

void bind_fractional_scale_manager(wl_client* client,
                                   void* data,
                                   uint32_t version,
                                   uint32_t id) {
  wl_resource* resource =
      wl_resource_create(client, &wp_fractional_scale_manager_v1_interface,
                         std::min(version, kFractionalScaleVersion), id);

  wl_resource_set_implementation(
      resource, &fractional_scale_manager_implementation, data, nullptr);
}

}  // namespace exo::wayland