import json
### Custom util functions
from .utils.modules import *
### Logger for debuging
from .utils.logger import logger
from typing import Any, List, Optional
from PyQt5.QtWidgets import QWidget
from platform import system, version, release
from ctypes import *
# import the main window object (mw) from aqt
from aqt import AnkiQt, DialogManager, mw
from aqt.theme import theme_manager, colors
from aqt import gui_hooks
## QT dialog windows
# Browser import legacy check (2.1.22)
if module_exists("aqt.browser.browser"):
from aqt.browser.browser import Browser
else:
from aqt.browser import Browser
# DeckStats import legacy check
if module_has_attribute("aqt.stats", "NewDeckStats"):
from aqt.stats import DeckStats, NewDeckStats
else:
from aqt.stats import DeckStats
from aqt.addcards import AddCards
from aqt.editcurrent import EditCurrent
from aqt.about import ClosableQDialog
from aqt.preferences import Preferences
from aqt.addons import AddonsDialog
# FilteredDeckConfigDialog import non-legacy check
if module_exists("aqt.filtered_deck"):
from aqt.filtered_deck import FilteredDeckConfigDialog
# QT page views
from aqt.toolbar import Toolbar, TopToolbar
from aqt.deckbrowser import DeckBrowser, DeckBrowserBottomBar
from aqt.overview import Overview, OverviewBottomBar
from aqt.editor import Editor
from aqt.reviewer import Reviewer, ReviewerBottomBar
from aqt.webview import AnkiWebView, WebContent
### Load config data here
from .config import config, write_config, get_config
## Addon compatibility fixes
# More Overview Stats 2.1 addon compatibility fix
addon_more_overview_stats_fix = config['addon_more_overview_stats']
addon_advanced_review_bottom_bar = config['addon_advanced_review_bottom_bar']
## Customization
theme = config['theme']
# Init script/file path
from .utils.css_files import css_files_dir
from .utils.themes import themes, write_theme, get_theme
logger.debug(css_files_dir)
logger.debug(themes)
themes_parsed = get_theme(theme)
color_mode = 2 if theme_manager.get_night_mode() else 1 # 1 = light and 2 = dark
dwmapi = None
## Darkmode windows titlebar thanks to miere43
def set_dark_titlebar(window, dwmapi) -> None:
if dwmapi:
handler_window = c_void_p(int(window.winId()))
DWMWA_USE_IMMERSIVE_DARK_MODE_BEFORE_20H1 = c_int(19)
DWMWA_USE_IMMERSIVE_DARK_MODE = c_int(20)
windows_version = int(version().split('.')[2])
attribute = DWMWA_USE_IMMERSIVE_DARK_MODE if windows_version >= 18985 else DWMWA_USE_IMMERSIVE_DARK_MODE_BEFORE_20H1
if windows_version >= 17763 and int(release()) >= 10:
dwmapi.DwmSetWindowAttribute(handler_window, attribute, byref(c_int(1)), c_size_t(4))
def set_dark_titlebar_qt(obj, dwmapi, fix=True) -> None:
if dwmapi and theme_manager.get_night_mode():
set_dark_titlebar(obj, dwmapi)
# Trick to refresh the titlebar after dark titlebar is set
if fix:
obj.showFullScreen()
obj.showNormal()
if system() == "Windows" and theme_manager.get_night_mode():
dwmapi = WinDLL("dwmapi")
if dwmapi:
dwmapi.DwmSetWindowAttribute.argtypes = [c_void_p, c_int, c_void_p, c_size_t]
dwmapi.DwmSetWindowAttribute.restype = c_int
set_dark_titlebar(mw, dwmapi)
logger.debug(dwmapi)
### CSS injections
def load_custom_style():
theme_colors = ""
for color_name in themes_parsed.get("colors"):
color = themes_parsed.get("colors").get(color_name)
if color[3]:
theme_colors += f"{color[3]}: {color[color_mode]};\n "
else:
theme_colors += f"--{color_name.lower().replace('_','-')}: {color[color_mode]};\n "
custom_style = """
""" % (theme_colors, config["font"], config["font_size"])
return custom_style
## Adds styling on the different webview contents, before the content is set
def on_webview_will_set_content(web_content: WebContent, context: Optional[Any]) -> None:
logger.debug(context) # Logs content being loaded, find out the instance
web_content.css.append(css_files_dir['global']) # Global css
web_content.head += load_custom_style() # Custom styling
# Deckbrowser
if isinstance(context, DeckBrowser):
web_content.css.append(css_files_dir['DeckBrowser'])
# TopToolbar
elif isinstance(context, TopToolbar):
web_content.css.append(css_files_dir['TopToolbar'])
# BottomToolbar (Buttons)
elif isinstance(context, DeckBrowserBottomBar) or isinstance(context, OverviewBottomBar):
web_content.css.append(css_files_dir['BottomBar'])
# Overview
elif isinstance(context, Overview):
if addon_more_overview_stats_fix:
web_content.head += ""
web_content.css.append(css_files_dir['Overview'])
# Editor
elif isinstance(context, Editor):
web_content.css.append(css_files_dir['Editor'])
# Reviewer
elif isinstance(context, Reviewer):
web_content.css.append(css_files_dir['Reviewer'])
elif isinstance(context, ReviewerBottomBar):
if addon_advanced_review_bottom_bar:
#web_content.head += ""
web_content.body += ""
else:
web_content.css.append(css_files_dir['BottomBar'])
web_content.css.append(css_files_dir['ReviewerBottomBar'])
# Button padding bottom
web_content.body += "
"
web_content.body += ""
mw.bottomWeb.adjustHeightToFit()
# CardLayout
elif context_name_includes(context, "aqt.clayout.CardLayout"):
web_content.css.append(css_files_dir['CardLayout'])
## Legacy webviews
# ResetRequired on card edit (legacy)
elif context_name_includes(context, "aqt.main.ResetRequired"):
web_content.css.append(css_files_dir['legacy'])
gui_hooks.webview_will_set_content.append(on_webview_will_set_content)
# TopToolbar styling fix through height change by adding
tag
def redraw_toolbar() -> None:
# Reload the webview content with added
tag, making the bar larger in height
mw.toolbar.web.setFixedHeight(60)
mw.toolbar.web.eval("""
var br = document.createElement("br");
document.body.appendChild(br);
""")
if 'Qt6' in QPalette.ColorRole.__module__:
mw.toolbar.web.eval("""
var div = document.createElement("div");
div.style.width = "5px";
div.style.height = "10px";
document.body.appendChild(div);
""")
# Auto adjust the height, then redraw the toolbar
mw.toolbar.web.adjustHeightToFit()
mw.toolbar.redraw()
def redraw_toolbar_legacy(links: List[str], _: Toolbar) -> None:
# Utilizing the link hook, we inject
tag through javascript
inject_br = """
"""
mw.toolbar.web.setFixedHeight(60)
links.append(inject_br)
mw.toolbar.web.adjustHeightToFit()
if attribute_exists(gui_hooks, "main_window_did_init"):
pass
#gui_hooks.main_window_did_init.append(redraw_toolbar)
elif attribute_exists(gui_hooks, "top_toolbar_did_init_links"):
gui_hooks.top_toolbar_did_init_links.append(redraw_toolbar_legacy)
pass
# Dialog window styling
def on_dialog_manager_did_open_dialog(dialog_manager: DialogManager, dialog_name: str, dialog_instance: QWidget) -> None:
logger.debug(dialog_name)
dialog: AnkiQt = dialog_manager._dialogs[dialog_name][1]
# If dwmapi found and nightmode is enabled, set dark titlebar to dialog window
set_dark_titlebar_qt(dialog, dwmapi)
# AddCards
if dialog_name == "AddCards":
context: AddCards = dialog_manager._dialogs[dialog_name][1]
context.setStyleSheet(open(css_files_dir['QAddCards'], encoding='utf-8').read())
# Addons popup
elif dialog_name == "AddonsDialog":
context: AddonsDialog = dialog_manager._dialogs[dialog_name][1]
context.setStyleSheet(open(css_files_dir['QAddonsDialog'], encoding='utf-8').read())
# Browser
elif dialog_name == "Browser":
context: Browser = dialog_manager._dialogs[dialog_name][1]
context.setStyleSheet(open(css_files_dir['QBrowser'], encoding='utf-8').read())
pass
# EditCurrent
elif dialog_name == "EditCurrent":
context: EditCurrent = dialog_manager._dialogs[dialog_name][1]
context.setStyleSheet(open(css_files_dir['QEditCurrent'], encoding='utf-8').read())
# FilteredDeckConfigDialog
elif module_exists("aqt.filtered_deck") and dialog_name == "FilteredDeckConfigDialog":
context: FilteredDeckConfigDialog = dialog_manager._dialogs[dialog_name][1]
context.setStyleSheet(open(css_files_dir['QFilteredDeckConfigDialog'], encoding='utf-8').read())
# Statistics / NewDeckStats
elif dialog_name == "NewDeckStats":
context: NewDeckStats = dialog_manager._dialogs[dialog_name][1]
context.setStyleSheet(open(css_files_dir['QNewDeckStats'], encoding='utf-8').read())
# About
elif dialog_name == "About":
context: ClosableQDialog = dialog_manager._dialogs[dialog_name][1]
context.setStyleSheet(open(css_files_dir['QAbout'], encoding='utf-8').read())
# Preferences
elif dialog_name == "Preferences":
context: Preferences = dialog_manager._dialogs[dialog_name][1]
context.setStyleSheet(open(css_files_dir['QPreferences'], encoding='utf-8').read())
# sync_log - kore ha nani desu???
elif dialog_name == "sync_log":
pass
if attribute_exists(gui_hooks, "dialog_manager_did_open_dialog"):
gui_hooks.dialog_manager_did_open_dialog.append(on_dialog_manager_did_open_dialog)
else:
## Legacy dialog window styling
# Sad monkey patch, instead of hooks :c
# setupDialogGC is being called on almost all dialog windows, utilizing this, the
# function is used as a type of hook to inject CSS styling on the QT instances
def monkey_setup_dialog_gc(obj: Any) -> None:
obj.finished.connect(lambda: mw.gcWindow(obj))
logger.debug(obj)
set_dark_titlebar_qt(obj, dwmapi)
# AddCards
if isinstance(obj, AddCards):
obj.setStyleSheet(open(css_files_dir['QAddCards'], encoding='utf-8').read())
# EditCurrent
elif isinstance(obj, EditCurrent):
obj.setStyleSheet(open(css_files_dir['QEditCurrent'], encoding='utf-8').read())
# Statistics / DeckStats
elif isinstance(obj, DeckStats):
obj.setStyleSheet(open(css_files_dir['QNewDeckStats'], encoding='utf-8').read())
# About
elif isinstance(obj, ClosableQDialog):
obj.setStyleSheet(open(css_files_dir['QAbout'], encoding='utf-8').read())
# Preferences
## Haven't found a solution for preferences yet :c
mw.setupDialogGC = monkey_setup_dialog_gc # Should be rare enough for other addons to also patch this I hope.
# Addons popup
if attribute_exists(gui_hooks, "addons_dialog_will_show"):
def on_addons_dialog_will_show(dialog: AddonsDialog) -> None:
logger.debug(dialog)
set_dark_titlebar_qt(dialog, dwmapi)
dialog.setStyleSheet(open(css_files_dir['QAddonsDialog'], encoding='utf-8').read())
gui_hooks.addons_dialog_will_show.append(on_addons_dialog_will_show)
# Browser
if attribute_exists(gui_hooks, "browser_will_show"):
def on_browser_will_show(browser: Browser) -> None:
logger.debug(browser)
set_dark_titlebar_qt(browser, dwmapi)
browser.setStyleSheet(open(css_files_dir['QBrowser'], encoding='utf-8').read())
gui_hooks.browser_will_show.append(on_browser_will_show)
## CONFIG DIALOG
from aqt import mw, gui_hooks
from aqt.qt import *
from .config import config
from aqt.utils import showInfo
from aqt.webview import AnkiWebView
class ThemeEditor(QDialog):
def __init__(self, parent, *args, **kwargs):
super().__init__(parent=parent or mw, *args, **kwargs)
self.config_editor = parent
self.setWindowModality(Qt.ApplicationModal)
self.setWindowTitle(f'Anki-redesign Advanced Editor')
self.setSizePolicy(self.make_size_policy())
self.setMinimumSize(420, 420)
self.setAttribute(Qt.WidgetAttribute.WA_DeleteOnClose)
set_dark_titlebar_qt(self, dwmapi, fix=False)
# Root layout
self.root_layout = QVBoxLayout(self)
# Main layout
self.layout = QVBoxLayout()
self.textedit = QTextEdit()
themes_plaintext = open(themes[theme], encoding='utf-8').read()
self.textedit.setPlainText(themes_plaintext)
self.layout.addWidget(self.textedit)
self.root_layout.addLayout(self.layout)
self.root_layout.addLayout(self.make_button_box())
def save_edit(self) -> None:
themes_parsed = json.loads(self.textedit.toPlainText())
write_theme(themes[theme], themes_parsed)
self.config_editor.update()
self.accept()
def make_button_box(self) -> QWidget:
def cancel():
button = QPushButton('Cancel')
button.setCursor(QCursor(Qt.CursorShape.PointingHandCursor))
qconnect(button.clicked, self.accept)
return button
def save():
button = QPushButton('Save')
button.setCursor(QCursor(Qt.CursorShape.PointingHandCursor))
button.setDefault(True)
button.setShortcut("Ctrl+Return")
button.clicked.connect(lambda _: self.save_edit())
return button
button_box = QHBoxLayout()
button_box.addStretch()
button_box.addWidget(cancel())
button_box.addWidget(save())
return button_box
def make_size_policy(self) -> QSizePolicy:
size_policy = QSizePolicy(QSizePolicy.Preferred, QSizePolicy.Minimum)
size_policy.setHorizontalStretch(0)
size_policy.setVerticalStretch(0)
size_policy.setHeightForWidth(self.sizePolicy().hasHeightForWidth())
return size_policy
class ConfigDialog(QDialog):
def __init__(self, parent: QWidget, *args, **kwargs):
super().__init__(parent=parent or mw, *args, **kwargs)
self.setWindowModality(Qt.ApplicationModal)
self.setWindowTitle(f'Anki-redesign Configuration')
self.setSizePolicy(self.make_size_policy())
self.setMinimumSize(420, 580)
self.setAttribute(Qt.WidgetAttribute.WA_DeleteOnClose)
set_dark_titlebar_qt(self, dwmapi, fix=False)
# Color/theme
## Loads theme color
self.theme_colors = themes_parsed.get("colors")
self.updates = []
self.theme_general = ["TEXT_FG", "WINDOW_BG", "FRAME_BG", "BUTTON_BG", "TOOLTIP_BG", "BORDER", "MEDIUM_BORDER", "FAINT_BORDER", "HIGHLIGHT_BG", "HIGHLIGHT_FG" , "LINK", "DISABLED", "SLIGHTLY_GREY_TEXT", "PRIMARY_COLOR"]
self.theme_decks = ["CURRENT_DECK", "NEW_COUNT", "LEARN_COUNT", "REVIEW_COUNT", "ZERO_COUNT"]
self.theme_browse = ["BURIED_FG", "SUSPENDED_FG", "MARKED_BG", "FLAG1_BG", "FLAG1_FG", "FLAG2_BG", "FLAG2_FG", "FLAG3_BG", "FLAG3_FG", "FLAG4_BG", "FLAG4_FG", "FLAG5_BG", "FLAG5_FG", "FLAG6_BG", "FLAG6_FG", "FLAG7_BG", "FLAG7_FG"]
self.theme_extra = []
# Root layout
self.root_layout = QVBoxLayout(self)
# Main layout
self.layout = QVBoxLayout()
## Initialize tab screen
self.tabs = QTabWidget(objectName="tabs")
self.tabs.setFocusPolicy(Qt.FocusPolicy.StrongFocus)
self.tab_general = QWidget(objectName="general")
self.tab_general.setLayout(self.create_color_picker_layout(self.theme_general))
self.tab_decks = QWidget(objectName="decks")
self.tab_decks.setLayout(self.create_color_picker_layout(self.theme_decks))
self.tab_browse = QWidget(objectName="browse")
self.tab_browse.setLayout(self.create_color_picker_layout(self.theme_browse))
self.tab_extra = QWidget(objectName="extra")
self.tab_extra.setLayout(self.create_color_picker_layout(self.theme_extra))
self.tab_settings = QWidget(objectName="settings")
self.settings_layout = QFormLayout()
self.theme_label = QLabel("Theme:")
self.theme_label.setStyleSheet('QLabel { font-size: 14px; font-weight: bold }')
self.settings_layout.addRow(self.theme_label)
for key in themes:
self.radio = self.radio_button(key)
self.settings_layout.addRow(key, self.radio)
self.settings_layout.addRow(QLabel())
self.font_label = QLabel("Font:")
self.font_label.setStyleSheet('QLabel { font-size: 14px; font-weight: bold }')
self.settings_layout.addRow(self.font_label)
self.interface_font = QFontComboBox()
self.interface_font.setFixedWidth(200)
self.interface_font.setCurrentFont(QFont(config["font"]))
self.settings_layout.addRow(self.interface_font)
self.font_size = QSpinBox()
self.font_size.setFixedWidth(200)
self.font_size.setValue(config["font_size"])
self.font_size.setSuffix("px")
self.settings_layout.addRow(self.font_size)
self.settings_layout.addRow(QLabel())
self.fix_label = QLabel("Addon-Compatibility Fixes: ")
self.fix_label.setStyleSheet('QLabel { font-size: 14px; font-weight: bold }')
self.settings_layout.addRow(self.fix_label)
self.reload_theme = self.checkbox("theme_reload")
self.settings_layout.addRow("QT6 theme on-start reload fix", self.reload_theme)
self.addon_more_overview_stats_check = self.checkbox("addon_more_overview_stats")
self.settings_layout.addRow("More Overview Stats 21", self.addon_more_overview_stats_check)
self.addon_advanced_review_bottom_bar_check = self.checkbox("addon_advanced_review_bottom_bar")
self.settings_layout.addRow("Advanced Review Bottom Bar", self.addon_advanced_review_bottom_bar_check)
self.tab_settings.setLayout(self.settings_layout)
## Add tabs
self.tabs.resize(300,200)
self.tabs.addTab(self.tab_settings,"Settings")
self.tabs.addTab(self.tab_general,"General")
self.tabs.addTab(self.tab_decks,"Decks")
self.tabs.addTab(self.tab_browse,"Browse")
#self.tabs.addTab(self.tab_extra,"Extra")
## Add tabs to widget
self.layout.addWidget(self.tabs)
self.root_layout.addLayout(self.layout)
self.root_layout.addLayout(self.make_button_box())
self.setLayout(self.root_layout)
self.show()
def update(self) -> None:
global themes_parsed
themes_parsed = get_theme(theme)
self.theme_colors = themes_parsed.get("colors")
for update in self.updates:
update()
def checkbox(self, key: str) -> QCheckBox:
checkbox = QCheckBox()
def update() -> None:
value = config[key]
checkbox.setChecked(value)
self.updates.append(update)
update()
return checkbox
def radio_button(self, key: str) -> QRadioButton:
radio = QRadioButton()
def update() -> None:
if theme == key:
radio.setChecked(True)
elif radio.isChecked():
radio.setChecked(False)
def toggle(checked) -> None:
global theme
if checked:
theme = key
self.update()
self.updates.append(update)
radio.toggled.connect(lambda checked: toggle(checked))
update()
return radio
def color_input(self, key: str) -> QPushButton:
button = QPushButton()
button.setCursor(QCursor(Qt.CursorShape.PointingHandCursor))
button.setFixedWidth(25)
button.setFixedHeight(25)
color_dialog = QColorDialog(self)
def set_color(rgb: str) -> None:
# Check for valid color
color = QColor()
color.setNamedColor(rgb)
if not color.isValid():
return
# Update color
color_dialog.setCurrentColor(color)
button.setStyleSheet('QPushButton{ background-color: "%s"; border: none; border-radius: 2px}' % rgb)
def update() -> None:
rgb = self.theme_colors.get(key)[color_mode]
set_color(rgb)
def save(color: QColor) -> None:
rgb = color.name(QColor.NameFormat.HexRgb)
self.theme_colors[key][color_mode] = rgb
set_color(rgb)
self.updates.append(update)
color_dialog.colorSelected.connect(lambda color: save(color))
button.clicked.connect(lambda _: color_dialog.exec())
return button
def create_color_picker_layout(self, colors) -> None:
layout = QFormLayout()
for key in colors:
self.test = self.color_input(key)
layout.addRow(self.theme_colors.get(key)[0], self.test)
return layout
def theme_file_editor(self) -> None:
diag = ThemeEditor(self)
diag.show()
def make_button_box(self) -> QWidget:
def advanced():
button = QPushButton('Advanced')
button.setCursor(QCursor(Qt.CursorShape.PointingHandCursor))
qconnect(button.clicked, self.theme_file_editor)
return button
def cancel():
button = QPushButton('Cancel')
button.setCursor(QCursor(Qt.CursorShape.PointingHandCursor))
qconnect(button.clicked, self.accept)
return button
def save():
button = QPushButton('Save')
button.setCursor(QCursor(Qt.CursorShape.PointingHandCursor))
button.setDefault(True)
button.setShortcut("Ctrl+Return")
button.clicked.connect(lambda _: self.save())
return button
button_box = QHBoxLayout()
button_box.addWidget(advanced())
button_box.addStretch()
button_box.addWidget(cancel())
button_box.addWidget(save())
return button_box
def make_size_policy(self) -> QSizePolicy:
size_policy = QSizePolicy(QSizePolicy.Preferred, QSizePolicy.Minimum)
size_policy.setHorizontalStretch(0)
size_policy.setVerticalStretch(0)
size_policy.setHeightForWidth(self.sizePolicy().hasHeightForWidth())
return size_policy
def save(self) -> None:
# Save settings and update config
global config, color_mode
config["font"] = self.interface_font.currentFont().family()
config["font_size"] = self.font_size.value()
config['addon_more_overview_stats'] = self.addon_more_overview_stats_check.isChecked()
config['addon_advanced_review_bottom_bar'] = self.addon_advanced_review_bottom_bar_check.isChecked()
config['theme_reload'] = self.reload_theme.isChecked()
config["theme"] = theme
write_config(config)
config = get_config()
# Write and update theme
color_mode = 2 if theme_manager.get_night_mode() else 1 # 1 = light and 2 = dark
themes_parsed["colors"] = self.theme_colors
write_theme(themes[theme], themes_parsed)
update_theme()
# mw.reset()
# ShowInfo for both new and legacy support
showInfo(_("Changes will take effect when you restart Anki."))
#showInfo(tr.preferences_changes_will_take_effect_when_you())
self.accept()
def check_legacy_colors() -> None:
try:
_ = colors.items()
except:
return False
return True
def refresh_all_windows() -> None:
# Redraw top toolbar
mw.toolbar.draw()
if attribute_exists(gui_hooks, "top_toolbar_did_init_links"):
gui_hooks.top_toolbar_did_init_links.append(lambda a,b: [redraw_toolbar_legacy(a,b), gui_hooks.top_toolbar_did_init_links.remove(print)])
# Redraw main body
if mw.state == "review":
mw.reviewer._initWeb()
if mw.reviewer.state == "question":
mw.reviewer._showQuestion()
else:
mw.reviewer._showAnswer()
elif mw.state == "overview":
mw.overview.refresh()
elif mw.state == "deckBrowser":
mw.deckBrowser.show()
if attribute_exists(gui_hooks, "top_toolbar_did_init_links"):
gui_hooks.top_toolbar_did_init_links.remove(redraw_toolbar)
def update_theme() -> None:
themes_parsed = get_theme(theme)
theme_colors = themes_parsed.get("colors")
# Apply theme on colors
ncolors = {}
# Legacy color check
logger.debug(dir(colors))
legacy = check_legacy_colors()
for color_name in theme_colors:
c = theme_colors.get(color_name)
ncolors[color_name] = c[color_mode]
if legacy:
colors[f"day{c[3].replace('--','-')}"] = c[1]
colors[f"night{c[3].replace('--','-')}"] = c[2]
# Potentially add fusion fixes too?
else:
logger.debug(getattr(colors, color_name, False))
if getattr(colors, color_name, False):
setattr(colors, color_name, (c[1], c[2]))
# Apply theme on palette
apply_theme(ncolors)
refresh_all_windows()
def apply_theme(colors) -> None:
# Load palette
palette = QPalette()
# QT mappings
#logger.debug(QPalette.ColorRole.__dict__)
color_map = {
QPalette.ColorRole.Window: "WINDOW_BG",
QPalette.ColorRole.WindowText: "TEXT_FG",
QPalette.ColorRole.Base: "FRAME_BG",
QPalette.ColorRole.AlternateBase: "WINDOW_BG",
QPalette.ColorRole.ToolTipBase: "TOOLTIP_BG",
QPalette.ColorRole.ToolTipText: "TEXT_FG",
QPalette.ColorRole.Text: "TEXT_FG",
QPalette.ColorRole.Button: "BUTTON_BG",
QPalette.ColorRole.ButtonText: "TEXT_FG",
QPalette.ColorRole.BrightText: "HIGHLIGHT_FG",
QPalette.ColorRole.HighlightedText: "HIGHLIGHT_FG",
QPalette.ColorRole.Link: "LINK",
QPalette.ColorRole.NoRole: "WINDOW_BG",
}
for color_role in color_map:
palette.setColor(color_role, QColor(colors[color_map[color_role]]))
highlight_bg = QColor(colors["HIGHLIGHT_BG"])
highlight_bg.setAlpha(64)
palette.setColor(QPalette.ColorRole.Highlight, highlight_bg)
disabled_color = QColor(colors["DISABLED"])
palette.setColor(QPalette.ColorRole.PlaceholderText, disabled_color)
palette.setColor(QPalette.ColorGroup.Disabled, QPalette.ColorRole.Text, disabled_color)
palette.setColor(QPalette.ColorGroup.Disabled, QPalette.ColorRole.ButtonText, disabled_color)
palette.setColor(QPalette.ColorGroup.Disabled, QPalette.ColorRole.HighlightedText, disabled_color)
# Update palette
mw.app.setPalette(palette)
theme_manager.default_palette = palette
theme_manager._apply_style(mw.app)
# Update webview bg
AnkiWebView._getWindowColor = lambda *args: QColor(colors["WINDOW_BG"])
AnkiWebView.get_window_bg_color = lambda *args: QColor(colors["WINDOW_BG"])
# Create menu actions
def create_menu_action(parent: QWidget, dialog_class: QDialog, dialog_name: str) -> QAction:
def open_dialog():
dialog = dialog_class(mw)
return dialog.exec()
action = QAction(dialog_name, parent)
#qconnect(action.triggered, open_dialog)
action.triggered.connect(open_dialog)
return action
# Load in the Anki-redesign menu
if not hasattr(mw, 'anki_redesign'):
mw.anki_redesign = QMenu("&Anki-redesign", mw)
mw.form.menubar.insertMenu(mw.form.menuHelp.menuAction(), mw.anki_redesign)
mw.anki_redesign.addAction(create_menu_action(mw.anki_redesign, ConfigDialog, "&Config"))
mw.anki_redesign.addSeparator()
mw.reset()
update_theme()
mw.reset()
# Rereload view to fix theme change on startup
if 'Qt6' in QPalette.ColorRole.__module__:
logger.debug('QT6 DETECTED....')
config = get_config()
if config["theme_reload"]:
update_theme()
def on_theme_did_change() -> None:
global color_mode
color_mode = 2 if theme_manager.get_night_mode() else 1 # 1 = light and 2 = dark
update_theme()
logger.debug("THEME CHANGEEEED")
refresh_all_windows()
mw.reset()
update_theme()
if attribute_exists(gui_hooks, "theme_did_change"):
gui_hooks.theme_did_change.append(on_theme_did_change)
logger.debug("YEP")