Blob Blame Raw


#include "tdockwindows.h"
#include <QBoxLayout>
#include <QVBoxLayout>

#include <QRegion>

#include <QPainter>
#include <QStyleOption>

#include <QMouseEvent>

//----------------------------------------

//-------------------
//    Decorations
//-------------------

class TDockDecoAllocator final : public DockDecoAllocator {
  DockSeparator *newSeparator(DockLayout *owner, bool orientation,
                              Region *parentRegion) override;
  DockPlaceholder *newPlaceholder(DockWidget *owner, Region *r, int idx,
                                  int attributes) override;
};

//========================================================================

//-----------------------
//    TMainWindow
//-----------------------

TMainWindow::TMainWindow(QWidget *parent, Qt::WindowFlags flags)
    : QWidget(parent, flags) {
  // Delete on close
  setAttribute(Qt::WidgetAttribute(Qt::WA_DeleteOnClose));

  // Set a vertical layout to include menu bars
  QVBoxLayout *vlayout = new QVBoxLayout;
  vlayout->setMargin(0);
  vlayout->setSpacing(4);
  setLayout(vlayout);

  // Allocate the dock layout
  m_layout = new DockLayout;
  m_layout->setContentsMargins(0, 0, 0, 0);
  m_layout->setSpacing(4);
  m_layout->setDecoAllocator(new TDockDecoAllocator);
  vlayout->addLayout(m_layout);
  vlayout->setAlignment(m_layout, Qt::AlignTop);

  show();  // NOTA: E' NECESSARIO MOSTRARE LA FINESTRA, prima di dockare
           // qualcosa (altrimenti non viene fatto
  // l'update della geometria della main window, e il contentsRect del layout
  // viene sballato!!).
}

//----------------------------------------

TMainWindow::~TMainWindow() {}

//----------------------------------------

//! Adds input \b item to this TMainWindow. If item was already
//! assigned to \b this TMainWindow, nothing happens.
void TMainWindow::addDockWidget(TDockWidget *item) {
  if (!m_layout->find(item)) m_layout->addWidget(item);
}

//----------------------------------------

void TMainWindow::removeDockWidget(TDockWidget *item) {
  m_layout->removeWidget(item);
}

//----------------------------------------

// NOTE: Unlike QMainWindow::addToolBar, we only allow one
// fixed-size undockable menu bar at top of the dock layout.
void TMainWindow::setMenuWidget(QWidget *menubar) {
  if (menubar) {
    QVBoxLayout *vlayout = static_cast<QVBoxLayout *>(layout());

    // If necessary, remove current menu bar
    if (m_menu && m_menu != menubar) vlayout->removeWidget(m_menu);

    vlayout->insertWidget(0, menubar);
  }
}

//----------------------------------------

void TMainWindow::setDecoAllocator(DockDecoAllocator *allocator) {
  m_layout->setDecoAllocator(allocator);
}

//----------------------------------------

//! Sets global thickness of separators between dock widget.
void TMainWindow::setSeparatorsThickness(int thick) {
  if (thick > 0) {
    m_layout->setSpacing(thick);
    m_layout->redistribute();
  }
}

//----------------------------------------

void TMainWindow::resizeEvent(QResizeEvent *event) { m_layout->redistribute(); }

//========================================================================

//-------------------
//    TDockWidget
//-------------------

//! Constructs a TDockWidget with given parent and window flags. If parent is
//! a TMainWindow, then the constructed dock widget is assigned to it
//! (addDockWidget'd).
//! TDockWidgets are always floating at construction.
TDockWidget::TDockWidget(QWidget *parent, Qt::WindowFlags flags)
    : DockWidget(parent, flags), m_widget(0), m_titlebar(0), m_margin(5) {
  setAutoFillBackground(false);
  // setFrameStyle(QFrame::StyledPanel);

  QBoxLayout *layout = new QBoxLayout(QBoxLayout::TopToBottom);
  layout->setSpacing(0);
  setLayout(layout);

  // Check if parent is a TMainWindow class
  TMainWindow *parentMain = qobject_cast<TMainWindow *>(parent);
  if (parentMain) parentMain->addDockWidget(this);

  setDecoAllocator(new TDockDecoAllocator);
}

//----------------------------------------

TDockWidget::TDockWidget(const QString &title, QWidget *parent,
                         Qt::WindowFlags flags)
    : DockWidget(parent, flags), m_widget(0), m_titlebar(0), m_margin(5) {
  setWindowTitle(title);

  QBoxLayout *layout = new QBoxLayout(QBoxLayout::TopToBottom);
  layout->setSpacing(0);
  setLayout(layout);
}

//----------------------------------------

void TDockWidget::setTitleBarWidget(QWidget *titlebar) {
  if (titlebar) {
    QBoxLayout *boxLayout = static_cast<QBoxLayout *>(layout());

    if (m_titlebar && m_titlebar != titlebar)
      boxLayout->removeWidget(m_titlebar);

    boxLayout->insertWidget(0, titlebar);
    // Set top/left-aligned
    boxLayout->setAlignment(
        titlebar, getOrientation() == vertical ? Qt::AlignTop : Qt::AlignLeft);

    m_titlebar = titlebar;
    if (m_floating) setFloatingAppearance();
  }
}

//----------------------------------------

void TDockWidget::windowTitleEvent(QEvent *e) {
  if (m_titlebar) m_titlebar->update();
}

//----------------------------------------

void TDockWidget::setWidget(QWidget *widget) {
  if (widget) {
    QBoxLayout *boxLayout = static_cast<QBoxLayout *>(layout());

    if (m_widget && m_widget != widget) boxLayout->removeWidget(m_widget);

    boxLayout->insertWidget(1, widget);
    widget->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);

    m_widget = widget;
  }
}

//----------------------------------------

void TDockWidget::setFloatingAppearance() {
  if (m_titlebar) {
    // If has a custom title bar, impose a margin to the layout
    // to provide a frame.
    layout()->setMargin(m_margin);

    if (!m_floating)  // was docked
    {
      // Adding margin to extremal sizes
      int addition = 2 * m_margin;
      setMinimumSize(
          QSize(minimumWidth() + addition, minimumHeight() + addition));
      setMaximumSize(
          QSize(maximumWidth() + addition, maximumHeight() + addition));
    }
  }
  // else
  //  setWindowFlags(Qt::Tool);
}

//----------------------------------------

void TDockWidget::setDockedAppearance() {
  // No layout margin is visible when docked
  layout()->setMargin(0);

  if (m_floating)  // was floating
  {
    // Removing margin from extremal sizes
    int addition = 2 * m_margin;
    setMinimumSize(
        QSize(minimumWidth() - addition, minimumHeight() - addition));
    setMaximumSize(
        QSize(maximumWidth() - addition, maximumHeight() - addition));
  }
}

//----------------------------------------

bool TDockWidget::isDragGrip(QPoint p) {
  if (!m_titlebar) return DockWidget::isDragGrip(p);

  return m_titlebar->geometry().contains(p);
}

//----------------------------------------

int TDockWidget::isResizeGrip(QPoint p) {
  if (m_dragging || (!m_titlebar && m_floating)) return 0;

  int marginType = 0;
  QRect geom(QPoint(0, 0), QPoint(width(), height()));
  int margin = layout()->margin();
  QRect contGeom(geom.adjusted(margin, margin, -margin, -margin));

  if (geom.contains(p) && !contGeom.contains(p)) {
    if (p.x() < 15) marginType |= leftMargin;
    if (p.y() < 15) marginType |= topMargin;
    if (p.x() > width() - 15) marginType |= rightMargin;
    if (p.y() > height() - 15) marginType |= bottomMargin;
  }

  return marginType;
}

//----------------------------------------

//! Currently working only for \b status = true. If you need to
//! dock a TDockWidget, you \b must specify a dock location by either
//! choosing a placeholder or identifying the Region and insertion index,
//! and then running 'parentLayout()->dockItem(..)'.
void TDockWidget::setFloating(bool status) {
  if (status) {
    setWindowFlags(Qt::Tool | Qt::FramelessWindowHint);
    if (!m_floating) parentLayout()->undockItem(this);
  }
}

//----------------------------------------

//! Specifies the orientation of the dock widget. It can
//! be either \b vertical (default) or \b horizontal, meaning that
//! the titlebar is laid respectively at the top or left side of
//! content widget. Directly speaking, it is equivalent to setting the
//! Qt's QDockWidget::DockWidgetVerticalTitleBar feature.
void TDockWidget::setOrientation(bool direction) {
  QBoxLayout *boxLayout = static_cast<QBoxLayout *>(layout());
  QBoxLayout::Direction boxDirection;

  if (direction == vertical) {
    boxLayout->setAlignment(m_titlebar, Qt::AlignTop);
    boxDirection = QBoxLayout::TopToBottom;
  } else {
    boxLayout->setAlignment(m_titlebar, Qt::AlignLeft);
    boxDirection = QBoxLayout::LeftToRight;
  }

  boxLayout->setDirection(boxDirection);
}

//----------------------------------------

bool TDockWidget::getOrientation() const {
  QBoxLayout *boxLayout = static_cast<QBoxLayout *>(layout());

  return (boxLayout->direction() == QBoxLayout::TopToBottom) ? vertical
                                                             : horizontal;
}

//----------------------------------------

//! Maximizes \b this TDockWidget, if docked.
void TDockWidget::setMaximized(bool status) {
  parentLayout()->setMaximized(this, status);
}

//----------------------------------------

QSize TDockWidget::getDockedMinimumSize() {
  int addedSize = 2 * m_margin;
  return m_floating ? minimumSize() -= QSize(addedSize, addedSize)
                    : minimumSize();
}

//----------------------------------------

QSize TDockWidget::getDockedMaximumSize() {
  int addedSize = 2 * m_margin;
  return m_floating ? maximumSize() -= QSize(addedSize, addedSize)
                    : maximumSize();
}

//========================================================================

//--------------------------
//    Custom Decorations
//--------------------------

class TDockSeparator final : public DockSeparator {
public:
  TDockSeparator(DockLayout *owner, bool orientation, Region *parentRegion)
      : DockSeparator(owner, orientation, parentRegion) {}

  void paintEvent(QPaintEvent *pe) override;
};

//----------------------------------------

class TDockPlaceholder final : public DockPlaceholder {
  QWidget *m_associated[3];

public:
  TDockPlaceholder(DockWidget *owner, Region *r, int idx, int attributes);
  ~TDockPlaceholder();

  void buildGeometry() override;

  void showEvent(QShowEvent *se) override;
  void hideEvent(QHideEvent *he) override;
};

//----------------------------------------

TDockPlaceholder::TDockPlaceholder(DockWidget *owner, Region *r, int idx,
                                   int attributes)
    : DockPlaceholder(owner, r, idx, attributes) {
  setAutoFillBackground(true);

  setObjectName("TDockPlaceholder");

  setWindowOpacity(0.8);
}

//----------------------------------------

TDockPlaceholder::~TDockPlaceholder() {
  if (isRoot()) {
    delete m_associated[0];
    delete m_associated[1];
    delete m_associated[2];
  }
}

//----------------------------------------

inline void TDockPlaceholder::buildGeometry() {
  DockPlaceholder::buildGeometry();

  if (isRoot()) {
    // Solution 2: Set associated widgets
    QRect geom(geometry());
    QSize horSize(geom.width(), 6);
    QSize vertSize(6, geom.height() + 12);

    setGeometry(QRect(geom.topLeft() - QPoint(6, 6), vertSize));

    m_associated[0] = new TDockPlaceholder(0, 0, 0, 0);
    m_associated[0]->setGeometry(QRect(geom.topLeft() - QPoint(0, 6), horSize));

    m_associated[1] = new TDockPlaceholder(0, 0, 0, 0);
    m_associated[1]->setGeometry(
        QRect(geom.topRight() + QPoint(1, -6), vertSize));

    m_associated[2] = new TDockPlaceholder(0, 0, 0, 0);
    m_associated[2]->setGeometry(
        QRect(geom.bottomLeft() + QPoint(0, 1), horSize));
  }
}

//-------------------------------------

void TDockPlaceholder::showEvent(QShowEvent *se) {
  if (isRoot()) {
    // Show associated widgets
    m_associated[0]->show();
    m_associated[1]->show();
    m_associated[2]->show();
  }
}

//-------------------------------------

void TDockPlaceholder::hideEvent(QHideEvent *he) {
  if (isRoot()) {
    // Show associated widgets
    m_associated[0]->hide();
    m_associated[1]->hide();
    m_associated[2]->hide();
  }
}

//-------------------------------------

void TDockSeparator::paintEvent(QPaintEvent *pe) {
  QPainter p(this);
  QStyleOption opt(0);
  opt.state = (getOrientation() == Region::horizontal)
                  ? QStyle::State_None
                  : QStyle::State_Horizontal;

  /*if (w->isEnabled())
opt.state |= QStyle::State_Enabled;
if (o != Qt::Horizontal)
opt.state |= QStyle::State_Horizontal;
if (mouse_over)
opt.state |= QStyle::State_MouseOver;*/

  opt.rect    = QRect(QPoint(0, 0), QSize(geometry().size()));
  opt.palette = palette();

  style()->drawPrimitive(QStyle::PE_IndicatorDockWidgetResizeHandle, &opt, &p,
                         this);

  p.end();
}

//-------------------------------------

DockSeparator *TDockDecoAllocator::newSeparator(DockLayout *owner,
                                                bool orientation,
                                                Region *parentRegion) {
  return new TDockSeparator(owner, orientation, parentRegion);
}

//-------------------------------------

DockPlaceholder *TDockDecoAllocator::newPlaceholder(DockWidget *owner,
                                                    Region *r, int idx,
                                                    int attributes) {
  return new TDockPlaceholder(owner, r, idx, attributes);
}

//-------------------------------------

void TDockWidget::selectDockPlaceholder(QMouseEvent *me) {
  if (m_placeholders.size() && m_placeholders[0]->isRoot()) {
    DockPlaceholder *selected = 0;

    QPoint pos = parentWidget()->mapFromGlobal(me->globalPos());
    if (parentLayout()->contentsRect().contains(pos))
      selected = m_placeholders[0];

    if (m_selectedPlace != selected) {
      if (m_selectedPlace) m_selectedPlace->hide();
      if (selected) selected->show();
    }

    m_selectedPlace = selected;
  } else
    DockWidget::selectDockPlaceholder(me);
}