1
0
Fork 0
mirror of synced 2024-06-22 04:20:25 +12:00
Rare/rare/components/tabs/settings/widgets/wrapper.py

335 lines
12 KiB
Python
Raw Normal View History

2022-02-27 12:14:47 +13:00
import re
2022-03-10 08:47:52 +13:00
import shutil
2022-02-27 12:14:47 +13:00
from logging import getLogger
from typing import Dict, List
2022-02-27 12:14:47 +13:00
2023-02-19 06:36:49 +13:00
from PyQt5.QtCore import pyqtSignal, QSettings, QSize, Qt, QMimeData, pyqtSlot, QCoreApplication
from PyQt5.QtGui import QDrag, QDropEvent, QDragEnterEvent, QDragMoveEvent, QFont, QMouseEvent
2023-02-19 06:36:49 +13:00
from PyQt5.QtWidgets import (
QHBoxLayout,
QLabel,
QPushButton,
QInputDialog,
QFrame,
QMessageBox,
QSizePolicy,
QWidget,
QScrollArea,
)
from rare.shared import RareCore
from rare.ui.components.tabs.settings.widgets.wrapper import Ui_WrapperSettings
2022-02-27 12:14:47 +13:00
from rare.utils import config_helper
from rare.utils.misc import icon
2022-02-27 12:14:47 +13:00
2023-02-19 06:36:49 +13:00
logger = getLogger("WrapperSettings")
2022-02-27 12:14:47 +13:00
extra_wrapper_regex = {
"proton": "\".*proton\" run", # proton
"mangohud": "mangohud" # mangohud
}
class Wrapper:
pass
class WrapperWidget(QFrame):
update_wrapper = pyqtSignal(str, str)
2022-02-27 12:14:47 +13:00
delete_wrapper = pyqtSignal(str)
def __init__(self, text: str, show_text=None, parent=None):
2022-03-12 11:08:36 +13:00
super(WrapperWidget, self).__init__(parent=parent)
if not show_text:
show_text = text.split()[0]
self.setFrameShape(QFrame.StyledPanel)
self.setSizePolicy(QSizePolicy.Maximum, QSizePolicy.Fixed)
2022-02-27 12:14:47 +13:00
self.text = text
self.text_lbl = QLabel(show_text, parent=self)
2022-03-10 08:47:52 +13:00
self.text_lbl.setFont(QFont("monospace"))
self.image_lbl = QLabel(parent=self)
2022-03-10 08:47:52 +13:00
self.image_lbl.setPixmap(icon("mdi.drag-vertical").pixmap(QSize(20, 20)))
self.setToolTip(text)
2022-02-27 12:14:47 +13:00
self.delete_button = QPushButton(icon("ei.remove"), "", parent=self)
self.unmanaged = show_text in extra_wrapper_regex.keys()
self.delete_button.setDisabled(self.unmanaged)
self.text_lbl.setDisabled(self.unmanaged)
if self.unmanaged:
self.delete_button.setToolTip(self.tr("Disable in settings"))
else:
self.delete_button.setToolTip(self.tr("Remove"))
2022-02-27 12:14:47 +13:00
self.delete_button.clicked.connect(self.delete)
layout = QHBoxLayout(self)
layout.setContentsMargins(0, 0, 0, 0)
layout.addWidget(self.image_lbl)
layout.addWidget(self.text_lbl)
layout.addWidget(self.delete_button)
self.setLayout(layout)
# lk: set object names for the stylesheet
self.setObjectName(type(self).__name__)
self.delete_button.setObjectName(f"{self.objectName()}Button")
2022-03-10 08:47:52 +13:00
2022-02-27 12:14:47 +13:00
def delete(self):
self.delete_wrapper.emit(self.text)
def mouseMoveEvent(self, a0: QMouseEvent) -> None:
if a0.buttons() == Qt.LeftButton:
a0.accept()
2022-03-10 08:47:52 +13:00
drag = QDrag(self)
mime = QMimeData()
drag.setMimeData(mime)
drag.exec_(Qt.MoveAction)
def mouseDoubleClickEvent(self, a0: QMouseEvent) -> None:
if a0.button() == Qt.LeftButton:
a0.accept()
if self.unmanaged:
return
dialog = QInputDialog(self)
dialog.setWindowTitle(f"{self.tr('Edit wrapper')} - {QCoreApplication.instance().applicationName()}")
dialog.setLabelText(self.tr("Edit wrapper command"))
dialog.setTextValue(self.text)
accepted = dialog.exec()
wrapper = dialog.textValue()
dialog.deleteLater()
if accepted:
self.update_wrapper.emit(self.text, wrapper)
2022-02-27 12:14:47 +13:00
class WrapperSettings(QWidget, Ui_WrapperSettings):
2022-02-27 12:14:47 +13:00
def __init__(self):
2022-03-10 08:47:52 +13:00
super(WrapperSettings, self).__init__()
2022-02-27 12:14:47 +13:00
self.setupUi(self)
self.wrappers: Dict[str, WrapperWidget] = {}
self.app_name: str
self.wrapper_scroll = QScrollArea(self.widget_stack)
self.wrapper_scroll.setWidgetResizable(True)
self.wrapper_scroll.setSizeAdjustPolicy(QScrollArea.AdjustToContents)
self.wrapper_scroll.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
self.wrapper_scroll.setProperty("no_kinetic_scroll", True)
2022-03-12 11:08:36 +13:00
self.scroll_content = WrapperContainer(
save_cb=self.save, parent=self.wrapper_scroll
)
self.wrapper_scroll.setWidget(self.scroll_content)
self.widget_stack.insertWidget(0, self.wrapper_scroll)
2022-03-10 08:47:52 +13:00
2023-02-19 06:36:49 +13:00
self.core = RareCore.instance().core()
2022-02-27 12:14:47 +13:00
self.add_button.clicked.connect(self.add_button_pressed)
2022-03-07 07:48:35 +13:00
self.settings = QSettings()
2022-02-27 12:14:47 +13:00
self.wrapper_scroll.horizontalScrollBar().rangeChanged.connect(self.adjust_scrollarea)
# lk: set object names for the stylesheet
self.setObjectName(type(self).__name__)
self.no_wrapper_label.setObjectName(f"{self.objectName()}Label")
self.wrapper_scroll.setObjectName(f"{self.objectName()}Scroll")
self.wrapper_scroll.horizontalScrollBar().setObjectName(
f"{self.wrapper_scroll.objectName()}Bar")
self.wrapper_scroll.verticalScrollBar().setObjectName(
f"{self.wrapper_scroll.objectName()}Bar")
@pyqtSlot(int, int)
def adjust_scrollarea(self, min: int, max: int):
wrapper_widget = self.scroll_content.findChild(WrapperWidget)
if not wrapper_widget:
return
# lk: when the scrollbar is not visible, min and max are 0
if max > min:
self.wrapper_scroll.setMaximumHeight(
wrapper_widget.sizeHint().height()
+ self.wrapper_scroll.rect().height() // 2
- self.wrapper_scroll.contentsRect().height() // 2
+ self.scroll_content.layout().spacing()
+ self.wrapper_scroll.horizontalScrollBar().sizeHint().height()
)
else:
self.wrapper_scroll.setMaximumHeight(
wrapper_widget.sizeHint().height()
+ self.wrapper_scroll.rect().height()
- self.wrapper_scroll.contentsRect().height()
)
2022-03-10 08:47:52 +13:00
2022-02-27 12:14:47 +13:00
def get_wrapper_string(self):
2022-03-07 07:48:35 +13:00
return " ".join(self.get_wrapper_list())
def get_wrapper_list(self):
wrappers = list(self.wrappers.values())
wrappers.sort(key=lambda x: self.scroll_content.layout().indexOf(x))
return [w.text for w in wrappers]
2022-02-27 12:14:47 +13:00
def add_button_pressed(self):
dialog = QInputDialog(self)
dialog.setWindowTitle(f"{self.tr('Add wrapper')} - {QCoreApplication.instance().applicationName()}")
dialog.setLabelText(self.tr("Enter wrapper command"))
accepted = dialog.exec()
wrapper = dialog.textValue()
dialog.deleteLater()
if accepted:
self.add_wrapper(wrapper)
def add_wrapper(self, text: str, position: int = -1, from_load: bool = False):
if text == "mangohud" and self.wrappers.get("mangohud"):
return
show_text = ""
2022-02-27 12:14:47 +13:00
for key, extra_wrapper in extra_wrapper_regex.items():
if re.match(extra_wrapper, text):
show_text = key
if not show_text:
show_text = text.split()[0]
2022-03-10 08:47:52 +13:00
# validate
if not text.strip(): # is empty
return
2022-03-10 08:47:52 +13:00
if not from_load:
if self.wrappers.get(text):
QMessageBox.warning(
self, self.tr("Warning"), self.tr("Wrapper is already in the list")
)
2022-03-10 08:47:52 +13:00
return
if show_text != "proton" and not shutil.which(text.split()[0]):
if QMessageBox.question(
self, self.tr("Warning"), self.tr("Wrapper is not in $PATH. Ignore? "),
QMessageBox.Yes | QMessageBox.No, QMessageBox.No
) == QMessageBox.No:
2022-03-10 08:47:52 +13:00
return
if text == "proton":
QMessageBox.warning(
self, self.tr("Warning"), self.tr("Do not insert proton manually. Add it through Proton settings")
)
return
self.widget_stack.setCurrentIndex(0)
if widget := self.wrappers.get(show_text, None):
widget.deleteLater()
widget = WrapperWidget(text, show_text, self.scroll_content)
if position < 0:
self.scroll_content.layout().addWidget(widget)
else:
self.scroll_content.layout().insertWidget(position, widget)
self.adjust_scrollarea(
self.wrapper_scroll.horizontalScrollBar().minimum(),
self.wrapper_scroll.horizontalScrollBar().maximum()
)
widget.update_wrapper.connect(self.update_wrapper)
2022-02-27 12:14:47 +13:00
widget.delete_wrapper.connect(self.delete_wrapper)
self.wrappers[show_text] = widget
2022-03-10 08:47:52 +13:00
if not from_load:
self.save()
2022-02-27 12:14:47 +13:00
@pyqtSlot(str)
2022-02-27 12:14:47 +13:00
def delete_wrapper(self, text: str):
text = text.split()[0]
2022-02-27 12:14:47 +13:00
widget = self.wrappers.get(text, None)
if widget:
2022-02-27 12:14:47 +13:00
self.wrappers.pop(text)
widget.deleteLater()
2022-02-27 12:14:47 +13:00
if not self.wrappers:
self.wrapper_scroll.setMaximumHeight(self.label_page.sizeHint().height())
2022-02-27 12:14:47 +13:00
self.widget_stack.setCurrentIndex(1)
2022-02-27 12:14:47 +13:00
self.save()
@pyqtSlot(str, str)
def update_wrapper(self, old: str, new: str):
key = old.split()[0]
idx = self.scroll_content.layout().indexOf(self.wrappers[key])
self.delete_wrapper(key)
self.add_wrapper(new, position=idx)
2022-02-27 12:14:47 +13:00
def save(self):
2022-03-07 07:48:35 +13:00
# save wrappers twice, to support wrappers with spaces
if len(self.wrappers) == 0:
config_helper.remove_option(self.app_name, "wrapper")
2022-03-07 07:48:35 +13:00
self.settings.remove(f"{self.app_name}/wrapper")
else:
config_helper.add_option(self.app_name, "wrapper", self.get_wrapper_string())
2022-03-07 07:48:35 +13:00
self.settings.setValue(f"{self.app_name}/wrapper", self.get_wrapper_list())
2022-02-27 12:14:47 +13:00
2022-03-07 07:48:35 +13:00
def load_settings(self, app_name: str):
2022-02-27 12:14:47 +13:00
self.app_name = app_name
for i in self.wrappers.values():
i.deleteLater()
self.wrappers.clear()
2022-03-07 07:48:35 +13:00
wrappers = self.settings.value(f"{self.app_name}/wrapper", [], str)
if not wrappers and (cfg := self.core.lgd.config.get(self.app_name, "wrapper", fallback="")):
logger.info("Loading wrappers from legendary config")
# no qt wrapper, but legendary wrapper, to have backward compatibility
pattern = re.compile(r'''((?:[^ "']|"[^"]*"|'[^']*')+)''')
wrappers = pattern.split(cfg)[1::2]
for wrapper in wrappers:
self.add_wrapper(wrapper, from_load=True)
2022-03-07 07:48:35 +13:00
if not self.wrappers:
self.wrapper_scroll.setMaximumHeight(self.label_page.sizeHint().height())
2022-02-27 12:14:47 +13:00
self.widget_stack.setCurrentIndex(1)
else:
self.widget_stack.setCurrentIndex(0)
2022-03-10 08:47:52 +13:00
self.save()
2022-03-12 11:08:36 +13:00
class WrapperContainer(QWidget):
drag_widget: QWidget
def __init__(self, save_cb, parent=None):
super(WrapperContainer, self).__init__(parent=parent)
self.setAcceptDrops(True)
self.save = save_cb
layout = QHBoxLayout()
layout.setContentsMargins(0, 0, 0, 0)
layout.setAlignment(Qt.AlignLeft | Qt.AlignTop)
self.setLayout(layout)
# lk: set object names for the stylesheet
self.setObjectName(type(self).__name__)
2022-03-12 11:08:36 +13:00
2022-03-10 08:47:52 +13:00
def dragEnterEvent(self, e: QDragEnterEvent):
widget = e.source()
self.drag_widget = widget
e.accept()
def _get_drop_index(self, x):
2022-03-12 11:08:36 +13:00
drag_idx = self.layout().indexOf(self.drag_widget)
if drag_idx > 0:
prev_widget = self.layout().itemAt(drag_idx - 1).widget()
if x < self.drag_widget.x() - prev_widget.width() // 2:
return drag_idx - 1
if drag_idx < self.layout().count() - 1:
next_widget = self.layout().itemAt(drag_idx + 1).widget()
if x > self.drag_widget.x() + self.drag_widget.width() + next_widget.width() // 2:
return drag_idx + 1
return drag_idx
2022-03-10 08:47:52 +13:00
def dragMoveEvent(self, e: QDragMoveEvent) -> None:
i = self._get_drop_index(e.pos().x())
2022-03-12 11:08:36 +13:00
self.layout().insertWidget(i, self.drag_widget)
2022-03-10 08:47:52 +13:00
def dropEvent(self, e: QDropEvent):
pos = e.pos()
widget = e.source()
index = self._get_drop_index(pos.x())
2022-03-12 11:08:36 +13:00
self.layout().insertWidget(index, widget)
2022-03-10 08:47:52 +13:00
self.drag_widget = None
e.accept()
self.save()