mirror of
https://github.com/amnezia-vpn/amnezia-client.git
synced 2026-05-08 14:33:23 +00:00
* refactor: move business logic from servers model * refactor: move containersModel initialization * refactor: added protocol ui controller and removed settings class from protocols model * refactor: moved cli management to separate controller * refactor: moved app split to separate controller * refactor: moved site split to separate controller * refactor: moved allowed dns to separate controller * refactor: moved language logic to separate ui controller * refactor: removed Settings from devices model * refactor: moved configs and services api logit to separate core controller * refactor: added a layer with a repository between the storage and controllers * refactor: use child parent system instead of smart pointers for controllers and models initialization * refactor: moved install functions from server controller to install controller * refactor: install controller refactoring * chore: renamed exportController to exportUiController * refactor: separate export controller * refactor: removed VpnConfigurationsController * chore: renamed ServerController to SshSession * refactor: replaced ServerController to SshSession * chore: moved qml controllers to separate folder * chore: include fixes * chore: moved utils from core root to core/utils * chore: include fixes * chore: rename core/utils files to camelCase foramt * chore: include fixes * chore: moved some utils to api and selfhosted folders * chore: include fixes * chore: remove unused file * chore: moved serialization folder to core/utils * chore: include fixes * chore: moved some files from client root to core/utils * chore: include fixes * chore: moved ui utils to ui/utils folder * chore: include fixes * chore: move utils from root to ui/utils * chore: include fixes * chore: moved configurators to core/configurators * chore: include fixes * refactor: moved iap logic from ui controller to core * refactor: moved remaining core logic from ApiConfigsController to SubscriptionController * chore: rename apiNewsController to apiNewsUiController * refactor: moved core logic from news ui controller to core * chore: renamed apiConfigsController to subscriptionUiController * chore: include fixes * refactor: merge ApiSettingsController with SubscriptionUiController * chore: moved ui selfhosted controllers to separate folder * chore: include fixes * chore: rename connectionController to connectiomUiController * refactor: moved core logic from connectionUiController * chore: rename settingsController to settingsUiController * refactor: move core logic from settingsUiController * refactor: moved core controller signal/slot connections to separate class * fix: newsController fixes after refactoring * chore: rename model to camelCase * chore: include fixes * chore: remove unused code * chore: move selfhosted core to separate folder * chore: include fixes * chore: rename importController to importUiController * refactor: move core logic from importUiController * chore: minor fixes * chore: remove prem v1 migration * refactor: remove openvpn over cloak and openvpn over shadowsocks * refactor: removed protocolsForContainer function * refactor: add core models * refactor: replace json with c++ structs for server config * refactor: move getDnsPair to ServerConfigUtils * feat: add admin selfhosted config export test * feat: add multi import test * refactor: use coreController for tests * feat: add few simple tests * chore: qrepos in all core controllers * feat: add test for settings * refactor: remove repo dependency from configurators * chore: moved protocols to core folder * chore: include fixes * refactor: moved containersDefs, defs, apiDefs, protocolsDefs to different places * chore: include fixes * chore: build fixes * chore: build fixes * refactor: remove q repo and interface repo * feat: add test for ui servers model and controller * chore: renamed to camelCase * chore: include fixes * refactor: moved core logic from sites ui controller * fix: fixed api config processing * fix: fixed processed server index processing * refactor: protocol models now use c++ structs instead of json configs * refactor: servers model now use c++ struct instead of json config * fix: fixed default server index processing * fix: fix logs init * fix: fix secure settings load keys * chore: build fixes * fix: fixed clear settings * fix: fixed restore backup * fix: sshSession usage * fix: fixed export functions signatures * fix: return missing part from buildContainerWorker * fix: fixed server description on page home * refactor: add container config helpers functions * refactor: c++ structs instead of json * chore: add dns protocol config struct * refactor: move config utils functions to config structs * feat: add test for selfhosted server setup * refactor: separate resources.qrc * fix: fixed server rename * chore: return nameOverriddenByUser * fix: build fixes * fix: fixed models init * refactor: cleanup models usage * fix: fixed models init * chore: cleanup connections and functions signatures * chore: cleanup updateModel calls * feat: added cache to servers repo * chore: cleanup unused functions * chore: ssxray processing * chore: remove transportProtoWithDefault and portWithDefault functions * chore: removed proto types any and l2tp * refactor: moved some constants * fix: fixed native configs export * refactor: remove json from processConfigWith functions * fix: fixed processed server index usage * fix: qml warning fixes * chore: merge fixes * chore: update tests * fix: fixed xray config processing * fix: fixed split tunneling processing * chore: rename sites controllers and model * chore: rename fixes * chore: minor fixes * chore: remove ability to load backup from "file with connection settings" button * fix: fixed api device revoke * fix: remove full model update when renaming a user * fix: fixed premium/free server rename * fix: fixed selfhosted new server install * fix: fixed updateContainer function * fix: fixed revoke for external premium configs * feat: add native configs qr processing * chore: codestyle fixes * fix: fixed admin config create * chore: again remove ability to load backup from "file with connection settings" button * chore: minor fixes * fix: fixed variables initialization * fix: fixed qml imports * fix: minor fixes * fix: fix vpnConnection function calls * feat: add buckup error handling * fix: fixed admin config revok * fix: fixed selfhosted awg installation * fix: ad visability * feat: add empty check for primary dns * chore: minor fixes
314 lines
8.1 KiB
C++
314 lines
8.1 KiB
C++
#include "listViewFocusController.h"
|
|
#include "ui/utils/qmlUtils.h"
|
|
|
|
#include <QQuickWindow>
|
|
|
|
ListViewFocusController::ListViewFocusController(QQuickItem *listView, QObject *parent)
|
|
: QObject { parent },
|
|
m_listView { listView },
|
|
m_focusChain {},
|
|
m_currentSection { Section::Default },
|
|
m_header { nullptr },
|
|
m_footer { nullptr },
|
|
m_focusedItem { nullptr },
|
|
m_focusedItemIndex { -1 },
|
|
m_delegateIndex { 0 },
|
|
m_isReturnNeeded { false },
|
|
m_currentSectionString { "Default", "Header", "Delegate", "Footer" }
|
|
{
|
|
QVariant headerItemProperty = m_listView->property("headerItem");
|
|
m_header = headerItemProperty.canConvert<QQuickItem *>() ? headerItemProperty.value<QQuickItem *>() : nullptr;
|
|
|
|
QVariant footerItemProperty = m_listView->property("footerItem");
|
|
m_footer = footerItemProperty.canConvert<QQuickItem *>() ? footerItemProperty.value<QQuickItem *>() : nullptr;
|
|
}
|
|
|
|
ListViewFocusController::~ListViewFocusController()
|
|
{
|
|
}
|
|
|
|
void ListViewFocusController::viewAtCurrentIndex() const
|
|
{
|
|
switch (m_currentSection) {
|
|
case Section::Default: [[fallthrough]];
|
|
case Section::Header: {
|
|
QMetaObject::invokeMethod(m_listView, "positionViewAtBeginning");
|
|
break;
|
|
}
|
|
case Section::Delegate: {
|
|
QMetaObject::invokeMethod(m_listView, "positionViewAtIndex", Q_ARG(int, m_delegateIndex), // Index
|
|
Q_ARG(int, 6)); // PositionMode (0 = Beginning; 1 = Center; 2 = End; 3 = Visible; 4 = Contain; 5 = SnapPosition)
|
|
break;
|
|
}
|
|
case Section::Footer: {
|
|
QMetaObject::invokeMethod(m_listView, "positionViewAtEnd");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
int ListViewFocusController::size() const
|
|
{
|
|
return m_listView->property("count").toInt();
|
|
}
|
|
|
|
int ListViewFocusController::currentIndex() const
|
|
{
|
|
return m_delegateIndex;
|
|
}
|
|
|
|
void ListViewFocusController::setDelegateIndex(int index)
|
|
{
|
|
m_delegateIndex = index;
|
|
m_listView->setProperty("currentIndex", index);
|
|
}
|
|
|
|
void ListViewFocusController::nextDelegate()
|
|
{
|
|
switch (m_currentSection) {
|
|
case Section::Default: {
|
|
if (hasHeader()) {
|
|
m_currentSection = Section::Header;
|
|
viewAtCurrentIndex();
|
|
break;
|
|
}
|
|
[[fallthrough]];
|
|
}
|
|
case Section::Header: {
|
|
if (size() > 0) {
|
|
m_currentSection = Section::Delegate;
|
|
viewAtCurrentIndex();
|
|
break;
|
|
}
|
|
[[fallthrough]];
|
|
}
|
|
case Section::Delegate:
|
|
if (m_delegateIndex < (size() - 1)) {
|
|
setDelegateIndex(m_delegateIndex + 1);
|
|
viewAtCurrentIndex();
|
|
break;
|
|
} else if (hasFooter()) {
|
|
m_currentSection = Section::Footer;
|
|
viewAtCurrentIndex();
|
|
break;
|
|
}
|
|
[[fallthrough]];
|
|
case Section::Footer: {
|
|
m_isReturnNeeded = true;
|
|
m_currentSection = Section::Default;
|
|
viewAtCurrentIndex();
|
|
break;
|
|
}
|
|
default: {
|
|
qCritical() << "Current section is invalid!";
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void ListViewFocusController::previousDelegate()
|
|
{
|
|
switch (m_currentSection) {
|
|
case Section::Default: {
|
|
if (hasFooter()) {
|
|
m_currentSection = Section::Footer;
|
|
viewAtCurrentIndex();
|
|
break;
|
|
}
|
|
[[fallthrough]];
|
|
}
|
|
case Section::Footer: {
|
|
if (size() > 0) {
|
|
m_currentSection = Section::Delegate;
|
|
setDelegateIndex(size() - 1);
|
|
break;
|
|
}
|
|
[[fallthrough]];
|
|
}
|
|
case Section::Delegate: {
|
|
if (m_delegateIndex > 0) {
|
|
setDelegateIndex(m_delegateIndex - 1);
|
|
viewAtCurrentIndex();
|
|
break;
|
|
} else if (hasHeader()) {
|
|
m_currentSection = Section::Header;
|
|
viewAtCurrentIndex();
|
|
break;
|
|
}
|
|
[[fallthrough]];
|
|
}
|
|
case Section::Header: {
|
|
m_isReturnNeeded = true;
|
|
m_currentSection = Section::Default;
|
|
viewAtCurrentIndex();
|
|
break;
|
|
}
|
|
default: {
|
|
qCritical() << "Current section is invalid!";
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void ListViewFocusController::decrementIndex()
|
|
{
|
|
m_delegateIndex--;
|
|
}
|
|
|
|
QQuickItem *ListViewFocusController::itemAtIndex(const int index) const
|
|
{
|
|
QQuickItem *item { nullptr };
|
|
|
|
QMetaObject::invokeMethod(m_listView, "itemAtIndex", Q_RETURN_ARG(QQuickItem *, item), Q_ARG(int, index));
|
|
|
|
return item;
|
|
}
|
|
|
|
QQuickItem *ListViewFocusController::currentDelegate() const
|
|
{
|
|
QQuickItem *result { nullptr };
|
|
|
|
switch (m_currentSection) {
|
|
case Section::Default: {
|
|
qWarning() << "No elements...";
|
|
break;
|
|
}
|
|
case Section::Header: {
|
|
result = m_header;
|
|
break;
|
|
}
|
|
case Section::Delegate: {
|
|
result = itemAtIndex(m_delegateIndex);
|
|
break;
|
|
}
|
|
case Section::Footer: {
|
|
result = m_footer;
|
|
break;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
QQuickItem *ListViewFocusController::focusedItem() const
|
|
{
|
|
return m_focusedItem;
|
|
}
|
|
|
|
void ListViewFocusController::focusNextItem()
|
|
{
|
|
if (m_isReturnNeeded) {
|
|
return;
|
|
}
|
|
|
|
reloadFocusChain();
|
|
|
|
if (m_focusChain.empty()) {
|
|
qWarning() << "No elements found in the delegate. Going to next delegate...";
|
|
nextDelegate();
|
|
focusNextItem();
|
|
return;
|
|
}
|
|
m_focusedItemIndex++;
|
|
m_focusedItem = qobject_cast<QQuickItem *>(m_focusChain.at(m_focusedItemIndex));
|
|
m_focusedItem->forceActiveFocus(Qt::TabFocusReason);
|
|
}
|
|
|
|
void ListViewFocusController::focusPreviousItem()
|
|
{
|
|
if (m_isReturnNeeded) {
|
|
return;
|
|
}
|
|
|
|
if (m_focusChain.empty()) {
|
|
qInfo() << "Empty focusChain with current delegate: " << currentDelegate() << "Scanning for elements...";
|
|
reloadFocusChain();
|
|
}
|
|
if (m_focusChain.empty()) {
|
|
qWarning() << "No elements found in the delegate. Going to next delegate...";
|
|
previousDelegate();
|
|
focusPreviousItem();
|
|
return;
|
|
}
|
|
if (m_focusedItemIndex == -1) {
|
|
m_focusedItemIndex = m_focusChain.size();
|
|
}
|
|
m_focusedItemIndex--;
|
|
m_focusedItem = qobject_cast<QQuickItem *>(m_focusChain.at(m_focusedItemIndex));
|
|
m_focusedItem->forceActiveFocus(Qt::TabFocusReason);
|
|
}
|
|
|
|
void ListViewFocusController::resetFocusChain()
|
|
{
|
|
m_focusChain.clear();
|
|
m_focusedItem = nullptr;
|
|
m_focusedItemIndex = -1;
|
|
}
|
|
|
|
void ListViewFocusController::reloadFocusChain()
|
|
{
|
|
m_focusChain = FocusControl::getItemsChain(currentDelegate());
|
|
}
|
|
|
|
bool ListViewFocusController::isFirstFocusItemInDelegate() const
|
|
{
|
|
return m_focusedItem && (m_focusedItem == m_focusChain.first());
|
|
}
|
|
|
|
bool ListViewFocusController::isLastFocusItemInDelegate() const
|
|
{
|
|
return m_focusedItem && (m_focusedItem == m_focusChain.last());
|
|
}
|
|
|
|
bool ListViewFocusController::hasHeader() const
|
|
{
|
|
return m_header && !FocusControl::getItemsChain(m_header).isEmpty();
|
|
}
|
|
|
|
bool ListViewFocusController::hasFooter() const
|
|
{
|
|
return m_footer && !FocusControl::getItemsChain(m_footer).isEmpty();
|
|
}
|
|
|
|
bool ListViewFocusController::isFirstFocusItemInListView() const
|
|
{
|
|
switch (m_currentSection) {
|
|
case Section::Footer: {
|
|
return isFirstFocusItemInDelegate() && !hasHeader() && (size() == 0);
|
|
}
|
|
case Section::Delegate: {
|
|
return isFirstFocusItemInDelegate() && (m_delegateIndex == 0) && !hasHeader();
|
|
}
|
|
case Section::Header: {
|
|
return isFirstFocusItemInDelegate();
|
|
}
|
|
case Section::Default: {
|
|
return true;
|
|
}
|
|
default: qWarning() << "Wrong section"; return true;
|
|
}
|
|
}
|
|
|
|
bool ListViewFocusController::isLastFocusItemInListView() const
|
|
{
|
|
switch (m_currentSection) {
|
|
case Section::Default: {
|
|
return !hasHeader() && (size() == 0) && !hasFooter();
|
|
}
|
|
case Section::Header: {
|
|
return isLastFocusItemInDelegate() && (size() == 0) && !hasFooter();
|
|
}
|
|
case Section::Delegate: {
|
|
return isLastFocusItemInDelegate() && (m_delegateIndex == size() - 1) && !hasFooter();
|
|
}
|
|
case Section::Footer: {
|
|
return isLastFocusItemInDelegate();
|
|
}
|
|
default: qWarning() << "Wrong section"; return true;
|
|
}
|
|
}
|
|
|
|
bool ListViewFocusController::isReturnNeeded() const
|
|
{
|
|
return m_isReturnNeeded;
|
|
}
|