4 #include "splitwindow.h" 6 #include "../core/log.h" 7 #include "../core/workspace.h" 8 #include "mainwindow.h" 10 WindowBase::WindowBase() {
11 LOG_DEBUG(
"new split window created");
12 setParent(MainWindow::instance());
14 WindowBase::~WindowBase() { LOG_DEBUG(
"closing split window"); }
15 void WindowBase::changeEvent(QEvent *event) {
16 if (event->type() == QEvent::ParentChange) {
17 LOG_DEBUG(
"parent of " <<
typeid(*this).name() <<
" changed to " 18 << (parent() !=
nullptr ?
typeid(*parent()).name()
23 static void replaceWindow(
const std::shared_ptr<Object> &old_window,
24 const std::shared_ptr<Window> &new_window) {
26 std::shared_ptr<Object> parent;
28 [&](
const std::shared_ptr<Object> &p,
const std::shared_ptr<Object> &c) {
29 if (c == old_window) {
34 for (
auto &property : parent->properties()) {
35 if (property.typeId() ==
typeid(std::shared_ptr<Window>)) {
36 if (property.read().value<std::shared_ptr<Window>>() == old_window) {
37 property.assign(
Variant(new_window));
44 SplitWindowBase::SplitWindowBase(Qt::Orientation orientation)
45 : orientation(orientation) {
46 class Splitter :
public QWidget {
48 bool aggregate =
false;
52 virtual void mousePressEvent(QMouseEvent *event)
override {
56 virtual void mouseMoveEvent(QMouseEvent *event)
override {
58 if (event->buttons() == Qt::LeftButton) {
59 ActionScope ws(
"Drag Splitter",
nullptr, aggregate);
61 QPoint pos = parent->mapFromGlobal(event->globalPos());
63 if (parent->orientation == Qt::Vertical) {
64 p = (pos.x()) * 1.0 / parent->width();
66 p = (pos.y()) * 1.0 / parent->height();
68 parent->position() = std::max(0.05, std::min(0.95, p));
69 LOG_DEBUG(
"splitter position " << parent->position());
73 virtual void mouseReleaseEvent(QMouseEvent *event)
override {
77 QBoxLayout *layout =
nullptr;
78 Splitter *splitter =
new Splitter(
this);
79 if (orientation == Qt::Horizontal) {
80 layout =
new QVBoxLayout(
this);
81 splitter->setCursor(Qt::SplitVCursor);
83 layout =
new QHBoxLayout(
this);
84 splitter->setCursor(Qt::SplitHCursor);
86 layout->setContentsMargins(0, 0, 0, 0);
87 layout->setSpacing(0);
89 a() = std::make_shared<EmptyWindow>();
90 b() = std::make_shared<EmptyWindow>();
91 splitter->setMinimumSize(8, 8);
92 this->splitter = splitter;
94 ws->modified.connect(
this, [
this, layout, splitter]() { sync(); });
95 layout->addWidget(splitter);
98 SplitWindowBase::~SplitWindowBase() {
99 LOG_DEBUG(
"~SplitWindowBase");
100 LOG_DEBUG(
"mainwindow: " << MainWindow::instance());
102 std::dynamic_pointer_cast<
WindowBase>(a())->setParent(MainWindow::instance());
103 std::dynamic_pointer_cast<
WindowBase>(b())->setParent(MainWindow::instance());
105 void SplitWindowBase::sync() {
107 auto *layout = (QBoxLayout *)this->layout();
108 if (!layout->itemAt(0) ||
109 (
dynamic_cast<WindowBase *
>(a().get()) != layout->itemAt(0)->widget())) {
110 if (
auto *item = layout->takeAt(0)) {
111 item->widget()->setParent(MainWindow::instance());
113 layout->insertWidget(0, (
WindowBase *)a().
get());
115 if (layout->count() != 3) {
117 layout->insertWidget(1, splitter);
119 if (!layout->itemAt(2) ||
120 (
dynamic_cast<WindowBase *
>(b().get()) != layout->itemAt(2)->widget())) {
121 if (
auto *item = layout->takeAt(2)) {
122 item->widget()->setParent(MainWindow::instance());
124 layout->insertWidget(2, (
WindowBase *)b().
get());
127 ->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored);
128 splitter->setSizePolicy(QSizePolicy::Preferred, QSizePolicy::Preferred);
130 ->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored);
131 int range = (orientation == Qt::Vertical) ? width() : height();
132 std::array<int, 3> s;
133 s[0] = (int)std::round(position() * range);
136 for (
size_t i = 0; i < s.size(); i++) {
137 if (layout->stretch(i) != s[i]) {
138 layout->setStretch(i, s[i]);
143 void ContentWindowBase::replace(
const std::shared_ptr<Window> &new_window) {
144 LOG_DEBUG(
"replacing window with " <<
typeid(*new_window).name());
146 auto me = std::dynamic_pointer_cast<
Window>(this->shared_from_this());
148 QTimer::singleShot(0,
this, [snapshot, new_window,
this]() {
150 auto me = std::dynamic_pointer_cast<
Window>(this->shared_from_this());
151 replaceWindow(me, new_window);
152 if (
auto split_window =
153 std::dynamic_pointer_cast<SplitWindowBase>(new_window)) {
155 snapshot->apply(split_window->a());
156 snapshot->apply(split_window->b());
157 split_window->b()->assignNewId();
162 ContentWindowBase::ContentWindowBase() {
163 layout =
new QVBoxLayout(
this);
164 layout->setContentsMargins(0, 0, 0, 0);
165 bar =
new QHBoxLayout(
this);
166 layout->addLayout(bar);
168 layout->setSpacing(0);
171 button->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
172 QTimer::singleShot(0,
this, [button,
this]() {
173 QMenu *menu =
new QMenu(
this);
174 auto types = Type::find<Window>()->list();
175 for (
auto &type : types) {
180 if (!type->constructable()) {
183 QString label = type->name().c_str();
184 label = label.replace(
"Window",
"");
185 if (type->typeId() ==
typeid(*this)) {
186 button->setText(label);
188 connect(menu->addAction(label), &QAction::triggered,
this,
189 [type,
this](
bool checked) {
191 replace(type->instantiate<
Window>());
194 button->setMenu(menu);
196 bar->addWidget(button);
199 class Spacer :
public QWidget {
206 virtual void mousePressEvent(QMouseEvent *event)
override {
207 QWidget::mousePressEvent(event);
209 ws->selection() = _parent->shared_from_this();
213 spacer =
new Spacer(
this);
214 spacer->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Minimum);
215 bar->addWidget(spacer);
220 button->setIcon(MATERIAL_ICON(
"border_vertical"));
221 button->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
222 connect(button, &QPushButton::clicked,
this, [
this]() {
224 replace(std::make_shared<SplitWindowVertical>());
226 bar->addWidget(button);
230 button->setIcon(MATERIAL_ICON(
"border_horizontal"));
231 button->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
232 connect(button, &QPushButton::clicked,
this, [
this]() {
234 replace(std::make_shared<SplitWindowHorizontal>());
236 bar->addWidget(button);
242 button->setIcon(style()->standardIcon(QStyle::SP_DockWidgetCloseButton));
243 button->setSizePolicy(QSizePolicy::Minimum, QSizePolicy::Fixed);
244 bar->addWidget(button);
245 connect(button, &QAbstractButton::clicked,
this, [
this]() {
246 QTimer::singleShot(0,
this, [
this]() {
249 auto me = std::dynamic_pointer_cast<
Window>(this->shared_from_this());
250 std::shared_ptr<Object> window_old;
251 std::shared_ptr<Window> window_new;
252 ws->recurse([&](
const std::shared_ptr<Object> &window) {
253 if (
auto split = std::dynamic_pointer_cast<SplitWindowBase>(window)) {
254 if (split->a() == me) {
256 window_new = split->b();
257 split->b() = std::make_shared<EmptyWindow>();
259 if (split->b() == me) {
261 window_new = split->a();
262 split->a() = std::make_shared<EmptyWindow>();
265 if (
auto parent = std::dynamic_pointer_cast<Document>(window)) {
268 window_new = std::make_shared<EmptyWindow>();
274 replaceWindow(window_old, window_new);
281 setContentWidget(
new QWidget(
this));
284 void ContentWindowBase::addToolWidget(QWidget *widget) {
285 widget->setParent(
this);
286 bar->insertWidget(bar->indexOf(spacer), widget);
288 void ContentWindowBase::addToolWidgetRight(QWidget *widget) {
289 widget->setParent(
this);
290 bar->insertWidget(bar->indexOf(spacer) + 1, widget);
293 void ContentWindowBase::setContentWidget(QWidget *widget) {
294 if (content_window) {
295 layout->removeWidget(content_window);
296 delete content_window;
298 content_window = widget;
299 content_window->setParent(
this);
300 content_window->setSizePolicy(QSizePolicy::Ignored, QSizePolicy::Ignored);
301 layout->addWidget(content_window);
302 layout->setStretch(0, 0);
303 layout->setStretch(1, 100);
306 EmptyWindow::EmptyWindow() {
307 auto *view =
new QWidget(
this);
308 view->setStyleSheet(
"background: #000;");
309 setContentWidget(view);
312 void ContentWindowBase::paintAnnotationHUD(
313 QPainter *painter,
const std::shared_ptr<const Type> &type) {
314 if (type !=
nullptr) {
318 if (
auto track = ws->currentAnnotationTrack().resolve(ws())) {
319 color = track->color();
322 QSize size = painter->window().size();
323 size = QSize(size.width(), size.height());
324 if (type->name() != _annotation_hud_string) {
325 static auto font = []() {
327 font.setPixelSize(32);
330 _annotation_hud_string = type->name();
339 auto c = QColor::fromHsvF(color, 1, 0.7, 0.9);
344 QString str = QString::fromStdString(_annotation_hud_string);
346 padding_x * 2 + painter->fontMetrics().boundingRect(str).width();
347 double text_bottom = padding_y * 2 + painter->fontMetrics().height();
348 painter->fillRect(0, 0, text_right, text_bottom, QBrush(c));
349 painter->fillRect(text_right, 0, size.width() - text_right, frame,
351 painter->fillRect(0, text_bottom, frame, size.height() - text_bottom,
353 painter->fillRect(size.width() - frame, frame, frame, size.height() - frame,
355 painter->fillRect(frame, size.height() - frame, size.width() - 2 * frame,
357 painter->setPen(QPen(QBrush(Qt::white), 0));
359 painter->drawText(QRectF(0, 0, text_right, text_bottom), Qt::AlignCenter,