nheko/src/Splitter.cpp

177 lines
5.2 KiB
C++
Raw Normal View History

2017-05-19 02:28:15 +02:00
/*
* nheko Copyright (C) 2017 Konstantinos Sideris <siderisk@auth.gr>
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <QSettings>
2017-05-19 02:28:15 +02:00
2020-01-31 06:12:02 +01:00
#include "Logging.h"
2017-05-19 02:28:15 +02:00
#include "Splitter.h"
constexpr auto MaxWidth = (1 << 24) - 1;
2017-05-19 18:55:38 +02:00
Splitter::Splitter(QWidget *parent)
2017-08-20 12:47:22 +02:00
: QSplitter(parent)
2020-01-31 06:12:02 +01:00
, sz_{splitter::calculateSidebarSizes(QFont{})}
2017-05-19 02:28:15 +02:00
{
2017-09-10 11:59:21 +02:00
connect(this, &QSplitter::splitterMoved, this, &Splitter::onSplitterMoved);
setChildrenCollapsible(false);
2017-05-19 02:28:15 +02:00
}
void
Splitter::restoreSizes(int fallback)
{
QSettings settings;
int savedWidth = settings.value("sidebar/width").toInt();
auto left = widget(0);
if (savedWidth == 0) {
hideSidebar();
return;
2018-10-07 13:09:47 +02:00
} else if (savedWidth == sz_.small) {
if (left) {
2018-10-07 13:09:47 +02:00
left->setMinimumWidth(sz_.small);
left->setMaximumWidth(sz_.small);
return;
}
}
2018-10-07 13:09:47 +02:00
left->setMinimumWidth(sz_.normal);
left->setMaximumWidth(2 * sz_.normal);
setSizes({sz_.normal, fallback - sz_.normal});
setStretchFactor(0, 0);
setStretchFactor(1, 1);
}
Splitter::~Splitter()
{
auto left = widget(0);
if (left) {
QSettings settings;
settings.setValue("sidebar/width", left->width());
}
}
2017-08-20 12:47:22 +02:00
void
Splitter::onSplitterMoved(int pos, int index)
2017-05-19 02:28:15 +02:00
{
2017-09-10 11:59:21 +02:00
Q_UNUSED(pos);
Q_UNUSED(index);
2017-05-19 02:28:15 +02:00
2017-09-10 11:59:21 +02:00
auto s = sizes();
2017-05-19 02:28:15 +02:00
2017-09-10 11:59:21 +02:00
if (s.count() < 2) {
2020-01-31 06:12:02 +01:00
nhlog::ui()->warn("Splitter needs at least two children");
2017-09-10 11:59:21 +02:00
return;
}
2017-05-19 02:28:15 +02:00
2018-10-07 13:09:47 +02:00
if (s[0] == sz_.normal) {
2017-09-10 11:59:21 +02:00
rightMoveCount_ += 1;
2017-05-19 02:28:15 +02:00
2017-09-10 11:59:21 +02:00
if (rightMoveCount_ > moveEventLimit_) {
2018-07-08 21:10:40 +02:00
auto left = widget(0);
auto cursorPosition = left->mapFromGlobal(QCursor::pos());
2017-05-19 02:28:15 +02:00
2017-09-10 11:59:21 +02:00
// if we are coming from the right, the cursor should
// end up on the first widget.
2018-07-08 21:10:40 +02:00
if (left->rect().contains(cursorPosition)) {
2018-10-07 13:09:47 +02:00
left->setMinimumWidth(sz_.small);
left->setMaximumWidth(sz_.small);
2017-05-19 02:28:15 +02:00
2017-09-10 11:59:21 +02:00
rightMoveCount_ = 0;
}
}
2018-10-07 13:09:47 +02:00
} else if (s[0] == sz_.small) {
2017-09-10 11:59:21 +02:00
leftMoveCount_ += 1;
2017-05-19 02:28:15 +02:00
2017-09-10 11:59:21 +02:00
if (leftMoveCount_ > moveEventLimit_) {
2018-07-08 21:10:40 +02:00
auto left = widget(0);
auto right = widget(1);
auto cursorPosition = right->mapFromGlobal(QCursor::pos());
2017-05-19 02:28:15 +02:00
2017-09-10 11:59:21 +02:00
// We move the start a little further so the transition isn't so abrupt.
auto extended = right->rect();
extended.translate(100, 0);
2017-05-19 02:28:15 +02:00
2017-09-10 11:59:21 +02:00
// if we are coming from the left, the cursor should
// end up on the second widget.
2018-07-08 21:10:40 +02:00
if (extended.contains(cursorPosition) &&
2018-10-07 13:09:47 +02:00
right->size().width() >= sz_.collapsePoint + sz_.normal) {
left->setMinimumWidth(sz_.normal);
left->setMaximumWidth(2 * sz_.normal);
2017-05-19 02:28:15 +02:00
2017-09-10 11:59:21 +02:00
leftMoveCount_ = 0;
}
}
}
2017-05-19 02:28:15 +02:00
}
void
Splitter::hideSidebar()
{
auto left = widget(0);
if (left)
left->hide();
}
void
Splitter::showChatView()
{
auto left = widget(0);
auto right = widget(1);
if (right->isHidden()) {
left->hide();
right->show();
// Restore previous size.
2018-10-07 13:09:47 +02:00
if (left->minimumWidth() == sz_.small) {
left->setMinimumWidth(sz_.small);
left->setMaximumWidth(sz_.small);
} else {
2018-10-07 13:09:47 +02:00
left->setMinimumWidth(sz_.normal);
left->setMaximumWidth(2 * sz_.normal);
}
}
}
void
Splitter::showFullRoomList()
{
auto left = widget(0);
auto right = widget(1);
right->hide();
left->show();
left->setMaximumWidth(MaxWidth);
}
2020-01-31 06:12:02 +01:00
splitter::SideBarSizes
splitter::calculateSidebarSizes(const QFont &f)
{
const auto height = static_cast<double>(QFontMetrics{f}.lineSpacing());
SideBarSizes sz;
2020-02-28 19:56:08 +01:00
sz.small = std::ceil(3.8 * height);
2020-01-31 06:12:02 +01:00
sz.normal = std::ceil(16 * height);
sz.groups = std::ceil(3 * height);
sz.collapsePoint = 2 * sz.normal;
return sz;
}