#include "ui/base/x/x11_display_manager.h"
#include <utility>
#include "base/functional/bind.h"
#include "base/task/single_thread_task_runner.h"
#include "ui/base/x/x11_display_util.h"
#include "ui/gfx/x/future.h"
#include "ui/gfx/x/randr.h"
#include "ui/gfx/x/x11_atom_cache.h"
#include "ui/gfx/x/xproto.h"
namespace ui {
namespace {
constexpr int kMinXrandrVersion = 103;
}
XDisplayManager::XDisplayManager(Delegate* delegate)
: delegate_(delegate),
connection_(x11::Connection::Get()),
x_root_window_(connection_->default_screen().root),
xrandr_version_(GetXrandrVersion()),
workspace_handler_(this) {}
XDisplayManager::~XDisplayManager() = default;
void XDisplayManager::Init() {
if (IsXrandrAvailable()) {
auto& randr = connection_->randr();
randr.SelectInput(
{x_root_window_, x11::RandR::NotifyMask::ScreenChange |
x11::RandR::NotifyMask::OutputChange |
x11::RandR::NotifyMask::CrtcChange});
}
FetchDisplayList();
}
bool XDisplayManager::IsXrandrAvailable() const {
return xrandr_version_ >= kMinXrandrVersion;
}
display::Display XDisplayManager::GetPrimaryDisplay() const {
DCHECK(!displays_.empty());
return displays_[primary_display_index_];
}
void XDisplayManager::AddObserver(display::DisplayObserver* observer) {
change_notifier_.AddObserver(observer);
}
void XDisplayManager::RemoveObserver(display::DisplayObserver* observer) {
change_notifier_.RemoveObserver(observer);
}
void XDisplayManager::OnEvent(const x11::Event& xev) {
auto* prop = xev.As<x11::PropertyNotifyEvent>();
if (xev.As<x11::RandR::NotifyEvent>() ||
(prop && prop->atom == x11::GetAtom("_NET_WORKAREA"))) {
DispatchDelayedDisplayListUpdate();
}
}
void XDisplayManager::SetDisplayList(std::vector<display::Display> displays) {
displays_ = std::move(displays);
delegate_->OnXDisplayListUpdated();
}
void XDisplayManager::FetchDisplayList() {
std::vector<display::Display> displays;
float scale = delegate_->GetXDisplayScaleFactor();
if (IsXrandrAvailable()) {
displays = BuildDisplaysFromXRandRInfo(xrandr_version_, scale,
&primary_display_index_);
} else {
displays = GetFallbackDisplayList(scale);
}
SetDisplayList(std::move(displays));
}
void XDisplayManager::OnCurrentWorkspaceChanged(
const std::string& new_workspace) {
change_notifier_.NotifyCurrentWorkspaceChanged(new_workspace);
}
void XDisplayManager::UpdateDisplayList() {
std::vector<display::Display> old_displays = displays_;
FetchDisplayList();
change_notifier_.NotifyDisplaysChanged(old_displays, displays_);
}
void XDisplayManager::DispatchDelayedDisplayListUpdate() {
update_task_.Reset(base::BindOnce(&XDisplayManager::UpdateDisplayList,
base::Unretained(this)));
base::SingleThreadTaskRunner::GetCurrentDefault()->PostTask(
FROM_HERE, update_task_.callback());
}
gfx::Point XDisplayManager::GetCursorLocation() const {
if (auto response = connection_->QueryPointer({x_root_window_}).Sync())
return {response->root_x, response->root_y};
return {};
}
std::string XDisplayManager::GetCurrentWorkspace() {
return workspace_handler_.GetCurrentWorkspace();
}
}