mirror-chatterino2/src/widgets/splitcontainer.cpp

555 lines
14 KiB
C++
Raw Normal View History

2017-11-12 17:21:50 +01:00
#include "widgets/splitcontainer.hpp"
2017-06-11 09:31:45 +02:00
#include "colorscheme.hpp"
#include "common.hpp"
#include "util/helpers.hpp"
2017-11-12 17:21:50 +01:00
#include "widgets/helper/notebooktab.hpp"
#include "widgets/notebook.hpp"
2017-11-12 17:21:50 +01:00
#include "widgets/split.hpp"
2016-12-29 17:31:07 +01:00
#include <QApplication>
#include <QDebug>
2017-01-18 04:52:47 +01:00
#include <QHBoxLayout>
#include <QMimeData>
2017-01-29 12:26:22 +01:00
#include <QObject>
2017-01-18 04:52:47 +01:00
#include <QPainter>
#include <QVBoxLayout>
#include <QWidget>
#include <boost/foreach.hpp>
2017-01-18 04:52:47 +01:00
#include <algorithm>
2017-04-14 17:52:22 +02:00
namespace chatterino {
namespace widgets {
2017-01-18 21:30:23 +01:00
2017-11-12 17:21:50 +01:00
bool SplitContainer::isDraggingSplit = false;
Split *SplitContainer::draggingSplit = nullptr;
std::pair<int, int> SplitContainer::dropPosition = std::pair<int, int>(-1, -1);
2017-01-01 02:30:42 +01:00
SplitContainer::SplitContainer(ChannelManager &_channelManager, Notebook *parent, NotebookTab *_tab,
const std::string &_settingPrefix)
: BaseWidget(parent->colorScheme, parent)
, settingPrefix(_settingPrefix)
, settingRoot(fS("{}", this->settingPrefix))
, chats(fS("{}/chats", this->settingRoot))
, channelManager(_channelManager)
, tab(_tab)
, dropPreview(this)
2017-11-12 17:21:50 +01:00
, chatWidgets()
2017-01-01 02:30:42 +01:00
{
this->tab->page = this;
2017-04-12 17:46:44 +02:00
this->setLayout(&this->ui.parentLayout);
2017-04-12 17:46:44 +02:00
this->setHidden(true);
this->setAcceptDrops(true);
2017-05-29 21:26:55 +02:00
this->ui.parentLayout.addSpacing(2);
this->ui.parentLayout.addLayout(&this->ui.hbox);
this->ui.parentLayout.setMargin(0);
2017-05-29 21:26:55 +02:00
this->ui.hbox.setSpacing(1);
this->ui.hbox.setMargin(0);
this->loadSplits();
this->refreshTitle();
2017-05-29 21:26:55 +02:00
}
2017-11-12 17:21:50 +01:00
void SplitContainer::updateFlexValues()
{
for (int i = 0; i < this->ui.hbox.count(); i++) {
QVBoxLayout *vbox = (QVBoxLayout *)ui.hbox.itemAt(i)->layout();
if (vbox->count() != 0) {
ui.hbox.setStretch(i, (int)(1000 * ((Split *)vbox->itemAt(0))->getFlexSizeX()));
}
}
}
std::pair<int, int> SplitContainer::removeFromLayout(Split *widget)
2017-04-12 17:46:44 +02:00
{
// remove reference to chat widget from chatWidgets vector
auto it = std::find(std::begin(this->chatWidgets), std::end(this->chatWidgets), widget);
if (it != std::end(this->chatWidgets)) {
this->chatWidgets.erase(it);
this->refreshTitle();
}
2017-04-12 17:46:44 +02:00
// remove from box and return location
for (int i = 0; i < this->ui.hbox.count(); ++i) {
auto vbox = static_cast<QVBoxLayout *>(this->ui.hbox.itemAt(i));
2017-01-01 02:30:42 +01:00
2017-01-11 18:52:09 +01:00
for (int j = 0; j < vbox->count(); ++j) {
2017-04-12 17:46:44 +02:00
if (vbox->itemAt(j)->widget() != widget) {
2017-01-11 18:52:09 +01:00
continue;
2017-04-12 17:46:44 +02:00
}
2017-01-01 02:30:42 +01:00
2017-06-10 22:48:28 +02:00
widget->setParent(nullptr);
2017-01-01 02:30:42 +01:00
bool isLastItem = vbox->count() == 0;
2017-01-11 18:52:09 +01:00
if (isLastItem) {
this->ui.hbox.removeItem(vbox);
2017-01-01 02:30:42 +01:00
delete vbox;
}
2017-01-11 18:52:09 +01:00
return std::pair<int, int>(i, isLastItem ? -1 : j);
2017-01-01 02:30:42 +01:00
}
}
return std::pair<int, int>(-1, -1);
}
2017-11-12 17:21:50 +01:00
void SplitContainer::addToLayout(Split *widget, std::pair<int, int> position)
2017-01-01 02:30:42 +01:00
{
this->chatWidgets.push_back(widget);
this->refreshTitle();
widget->giveFocus(Qt::MouseFocusReason);
2017-01-01 02:30:42 +01:00
// add vbox at the end
if (position.first < 0 || position.first >= this->ui.hbox.count()) {
2017-01-01 02:30:42 +01:00
auto vbox = new QVBoxLayout();
vbox->addWidget(widget);
this->ui.hbox.addLayout(vbox, 1);
this->refreshCurrentFocusCoordinates();
2017-01-01 02:30:42 +01:00
return;
}
// insert vbox
2017-01-11 18:52:09 +01:00
if (position.second == -1) {
2017-01-01 02:30:42 +01:00
auto vbox = new QVBoxLayout();
vbox->addWidget(widget);
this->ui.hbox.insertLayout(position.first, vbox, 1);
this->refreshCurrentFocusCoordinates();
2017-01-01 02:30:42 +01:00
return;
}
// add to existing vbox
auto vbox = static_cast<QVBoxLayout *>(this->ui.hbox.itemAt(position.first));
2017-01-01 02:30:42 +01:00
2017-04-12 17:46:44 +02:00
vbox->insertWidget(std::max(0, std::min(vbox->count(), position.second)), widget);
this->refreshCurrentFocusCoordinates();
2017-01-01 02:30:42 +01:00
}
2017-11-12 17:21:50 +01:00
const std::vector<Split *> &SplitContainer::getChatWidgets() const
{
return this->chatWidgets;
}
2017-11-12 17:21:50 +01:00
NotebookTab *SplitContainer::getTab() const
{
return this->tab;
}
void SplitContainer::addChat(bool openChannelNameDialog, std::string chatUUID)
{
if (chatUUID.empty()) {
chatUUID = CreateUUID().toStdString();
}
Split *w = this->createChatWidget(chatUUID);
if (openChannelNameDialog) {
bool ret = w->showChangeChannelPopup("Open channel", true);
if (!ret) {
delete w;
return;
}
}
this->addToLayout(w, std::pair<int, int>(-1, -1));
}
2017-11-12 17:21:50 +01:00
void SplitContainer::refreshCurrentFocusCoordinates(bool alsoSetLastRequested)
{
int setX = -1;
int setY = -1;
bool doBreak = false;
for (int x = 0; x < this->ui.hbox.count(); ++x) {
QLayoutItem *item = this->ui.hbox.itemAt(x);
if (item->isEmpty()) {
setX = x;
break;
}
QVBoxLayout *vbox = static_cast<QVBoxLayout *>(item->layout());
for (int y = 0; y < vbox->count(); ++y) {
QLayoutItem *innerItem = vbox->itemAt(y);
if (innerItem->isEmpty()) {
setX = x;
setY = y;
doBreak = true;
break;
}
QWidget *w = innerItem->widget();
if (w) {
2017-11-12 17:21:50 +01:00
Split *chatWidget = static_cast<Split *>(w);
if (chatWidget->hasFocus()) {
setX = x;
setY = y;
doBreak = true;
break;
}
}
}
if (doBreak) {
break;
}
}
if (setX != -1) {
this->currentX = setX;
if (setY != -1) {
this->currentY = setY;
if (alsoSetLastRequested) {
this->lastRequestedY[setX] = setY;
}
}
}
}
2017-11-12 17:21:50 +01:00
void SplitContainer::requestFocus(int requestedX, int requestedY)
{
// XXX: Perhaps if we request an Y coordinate out of bounds, we shuold set all previously set
// requestedYs to 0 (if -1 is requested) or that x-coordinates vbox count (if requestedY >=
// currentvbox.count() is requested)
if (requestedX < 0 || requestedX >= this->ui.hbox.count()) {
return;
}
QLayoutItem *item = this->ui.hbox.itemAt(requestedX);
QWidget *xW = item->widget();
if (item->isEmpty()) {
qDebug() << "Requested hbox item " << requestedX << "is empty";
if (xW) {
qDebug() << "but xW is not null";
// TODO: figure out what to do here
}
return;
}
QVBoxLayout *vbox = static_cast<QVBoxLayout *>(item->layout());
if (requestedY < 0) {
requestedY = 0;
} else if (requestedY >= vbox->count()) {
requestedY = vbox->count() - 1;
}
this->lastRequestedY[requestedX] = requestedY;
QLayoutItem *innerItem = vbox->itemAt(requestedY);
if (innerItem->isEmpty()) {
qDebug() << "Requested vbox item " << requestedY << "is empty";
return;
}
QWidget *w = innerItem->widget();
if (w) {
2017-11-12 17:21:50 +01:00
Split *chatWidget = static_cast<Split *>(w);
chatWidget->giveFocus(Qt::OtherFocusReason);
}
}
2017-11-12 17:21:50 +01:00
void SplitContainer::enterEvent(QEvent *)
{
if (this->ui.hbox.count() == 0) {
this->setCursor(QCursor(Qt::PointingHandCursor));
2017-01-11 18:52:09 +01:00
} else {
this->setCursor(QCursor(Qt::ArrowCursor));
}
}
2017-11-12 17:21:50 +01:00
void SplitContainer::leaveEvent(QEvent *)
{
}
2017-11-12 17:21:50 +01:00
void SplitContainer::mouseReleaseEvent(QMouseEvent *event)
{
if (this->ui.hbox.count() == 0 && event->button() == Qt::LeftButton) {
2017-01-29 11:38:00 +01:00
// "Add Chat" was clicked
this->addChat(true);
this->setCursor(QCursor(Qt::ArrowCursor));
}
}
2017-11-12 17:21:50 +01:00
void SplitContainer::dragEnterEvent(QDragEnterEvent *event)
2017-01-01 02:30:42 +01:00
{
2017-01-11 18:52:09 +01:00
if (!event->mimeData()->hasFormat("chatterino/split"))
return;
2017-01-01 02:30:42 +01:00
2017-08-05 22:53:21 +02:00
if (!isDraggingSplit) {
2017-04-12 17:46:44 +02:00
return;
}
this->dropRegions.clear();
2017-04-12 17:46:44 +02:00
if (this->ui.hbox.count() == 0) {
this->dropRegions.push_back(DropRegion(rect(), std::pair<int, int>(-1, -1)));
2017-04-12 17:46:44 +02:00
} else {
for (int i = 0; i < this->ui.hbox.count() + 1; ++i) {
this->dropRegions.push_back(
DropRegion(QRect(((i * 4 - 1) * width() / this->ui.hbox.count()) / 4, 0,
width() / this->ui.hbox.count() / 2 + 1, height() + 1),
std::pair<int, int>(i, -1)));
2017-04-12 17:46:44 +02:00
}
2017-01-01 02:30:42 +01:00
for (int i = 0; i < this->ui.hbox.count(); ++i) {
auto vbox = static_cast<QVBoxLayout *>(this->ui.hbox.itemAt(i));
2017-04-12 17:46:44 +02:00
for (int j = 0; j < vbox->count() + 1; ++j) {
this->dropRegions.push_back(DropRegion(
QRect(i * width() / this->ui.hbox.count(),
((j * 2 - 1) * height() / vbox->count()) / 2,
width() / this->ui.hbox.count() + 1, height() / vbox->count() + 1),
2017-01-24 19:51:57 +01:00
2017-04-12 17:46:44 +02:00
std::pair<int, int>(i, j)));
2017-01-01 02:30:42 +01:00
}
}
2017-04-12 17:46:44 +02:00
}
2017-01-01 02:30:42 +01:00
2017-04-12 17:46:44 +02:00
setPreviewRect(event->pos());
2017-01-01 02:30:42 +01:00
2017-04-12 17:46:44 +02:00
event->acceptProposedAction();
2017-01-01 02:30:42 +01:00
}
2017-11-12 17:21:50 +01:00
void SplitContainer::dragMoveEvent(QDragMoveEvent *event)
2017-01-01 02:30:42 +01:00
{
setPreviewRect(event->pos());
}
2017-11-12 17:21:50 +01:00
void SplitContainer::setPreviewRect(QPoint mousePos)
2017-01-01 02:30:42 +01:00
{
for (DropRegion region : this->dropRegions) {
2017-01-11 18:52:09 +01:00
if (region.rect.contains(mousePos)) {
this->dropPreview.setBounds(region.rect);
2017-01-24 19:51:57 +01:00
if (!this->dropPreview.isVisible()) {
this->dropPreview.show();
this->dropPreview.raise();
2017-01-24 19:51:57 +01:00
}
2017-01-01 02:30:42 +01:00
dropPosition = region.position;
return;
}
}
2017-01-24 19:51:57 +01:00
this->dropPreview.hide();
2017-01-01 02:30:42 +01:00
}
2017-11-12 17:21:50 +01:00
void SplitContainer::dragLeaveEvent(QDragLeaveEvent *event)
2017-01-01 02:30:42 +01:00
{
this->dropPreview.hide();
2017-01-01 02:30:42 +01:00
}
2017-11-12 17:21:50 +01:00
void SplitContainer::dropEvent(QDropEvent *event)
2017-01-01 02:30:42 +01:00
{
2017-01-11 18:52:09 +01:00
if (isDraggingSplit) {
2017-01-01 02:30:42 +01:00
event->acceptProposedAction();
2017-11-12 17:21:50 +01:00
SplitContainer::draggingSplit->setParent(this);
2017-01-01 02:30:42 +01:00
2017-11-12 17:21:50 +01:00
addToLayout(SplitContainer::draggingSplit, dropPosition);
2017-01-01 02:30:42 +01:00
}
this->dropPreview.hide();
2016-12-30 18:00:25 +01:00
}
2017-11-12 17:21:50 +01:00
bool SplitContainer::eventFilter(QObject *object, QEvent *event)
{
if (event->type() == QEvent::FocusIn) {
QFocusEvent *focusEvent = static_cast<QFocusEvent *>(event);
this->refreshCurrentFocusCoordinates((focusEvent->reason() == Qt::MouseFocusReason));
}
return false;
}
2017-11-12 17:21:50 +01:00
void SplitContainer::paintEvent(QPaintEvent *)
2016-12-30 18:00:25 +01:00
{
QPainter painter(this);
if (this->ui.hbox.count() == 0) {
painter.fillRect(rect(), this->colorScheme.ChatBackground);
2017-01-01 02:30:42 +01:00
painter.setPen(this->colorScheme.Text);
painter.drawText(rect(), "Add Chat", QTextOption(Qt::AlignCenter));
2017-01-11 18:52:09 +01:00
} else {
2017-08-17 14:52:41 +02:00
painter.fillRect(rect(), this->colorScheme.ChatSeperator);
}
QColor accentColor = (QApplication::activeWindow() == this->window()
? this->colorScheme.TabSelectedBackground
: this->colorScheme.TabSelectedUnfocusedBackground);
painter.fillRect(0, 0, width(), 2, accentColor);
2016-12-29 17:31:07 +01:00
}
2017-11-12 17:21:50 +01:00
void SplitContainer::showEvent(QShowEvent *event)
{
// Whenever this notebook page is shown, give focus to the last focused chat widget
// If this is the first time this notebook page is shown, it will give focus to the top-left
// chat widget
this->requestFocus(this->currentX, this->currentY);
}
2017-11-12 17:21:50 +01:00
static std::pair<int, int> getWidgetPositionInLayout(QLayout *layout, const Split *chatWidget)
2017-01-29 12:26:22 +01:00
{
for (int i = 0; i < layout->count(); ++i) {
printf("xD\n");
}
return std::make_pair(-1, -1);
}
2017-11-12 17:21:50 +01:00
std::pair<int, int> SplitContainer::getChatPosition(const Split *chatWidget)
2017-01-29 12:26:22 +01:00
{
auto layout = this->ui.hbox.layout();
2017-01-29 12:26:22 +01:00
if (layout == nullptr) {
return std::make_pair(-1, -1);
}
return getWidgetPositionInLayout(layout, chatWidget);
}
Split *SplitContainer::createChatWidget(const std::string &uuid)
{
auto split = new Split(this->channelManager, this, uuid);
split->getChannelView().highlightedMessageReceived.connect([this] {
this->tab->setHighlightState(HighlightState::Highlighted); //
});
return split;
}
2017-11-12 17:21:50 +01:00
void SplitContainer::refreshTitle()
{
if (!this->tab->useDefaultBehaviour) {
return;
}
QString newTitle = "";
bool first = true;
for (const auto &chatWidget : this->chatWidgets) {
auto channelName = QString::fromStdString(chatWidget->channelName.getValue());
if (channelName.isEmpty()) {
continue;
}
if (!first) {
newTitle += ", ";
}
newTitle += channelName;
first = false;
}
if (newTitle.isEmpty()) {
newTitle = "empty";
}
this->tab->setTitle(newTitle);
}
void SplitContainer::loadSplits()
{
const auto hboxes = this->chats.getValue();
int column = 0;
for (const std::vector<std::string> &hbox : hboxes) {
int row = 0;
for (const std::string &chatUUID : hbox) {
Split *split = this->createChatWidget(chatUUID);
this->addToLayout(split, std::pair<int, int>(column, row));
++row;
2017-01-29 11:38:00 +01:00
}
++column;
2017-01-29 11:38:00 +01:00
}
}
template <typename Container>
static void saveFromLayout(QLayout *layout, Container &container)
2017-01-29 12:26:22 +01:00
{
for (int i = 0; i < layout->count(); ++i) {
auto item = layout->itemAt(i);
auto innerLayout = item->layout();
if (innerLayout != nullptr) {
std::vector<std::string> vbox;
for (int j = 0; j < innerLayout->count(); ++j) {
auto innerItem = innerLayout->itemAt(j);
auto innerWidget = innerItem->widget();
if (innerWidget == nullptr) {
assert(false);
continue;
}
Split *innerSplit = qobject_cast<Split *>(innerWidget);
vbox.push_back(innerSplit->getUUID());
2017-01-29 12:26:22 +01:00
}
container.push_back(vbox);
2017-01-29 12:26:22 +01:00
continue;
}
}
}
void SplitContainer::save()
2017-01-29 11:38:00 +01:00
{
auto layout = this->ui.hbox.layout();
2017-01-29 12:26:22 +01:00
std::vector<std::vector<std::string>> _chats;
2017-01-29 12:26:22 +01:00
for (int i = 0; i < layout->count(); ++i) {
auto item = layout->itemAt(i);
2017-01-29 12:26:22 +01:00
auto innerLayout = item->layout();
if (innerLayout != nullptr) {
std::vector<std::string> vbox;
for (int j = 0; j < innerLayout->count(); ++j) {
auto innerItem = innerLayout->itemAt(j);
auto innerWidget = innerItem->widget();
if (innerWidget == nullptr) {
assert(false);
continue;
}
Split *innerSplit = qobject_cast<Split *>(innerWidget);
vbox.push_back(innerSplit->getUUID());
}
2017-01-29 12:26:22 +01:00
_chats.push_back(vbox);
continue;
}
2017-01-29 11:38:00 +01:00
}
this->chats = _chats;
2017-01-18 21:30:23 +01:00
}
2017-04-14 17:52:22 +02:00
} // namespace widgets
} // namespace chatterino