Merge pull request #1050 from BlueGreenMagick/choose-addon-to-update

Choose addon to update
This commit is contained in:
Damien Elmes 2021-03-10 11:41:50 +10:00 committed by GitHub
commit 1e7405296f
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 187 additions and 17 deletions

View File

@ -64,3 +64,4 @@ addons-delete-the-numd-selected-addon =
[one] Delete the { $count } selected add-on? [one] Delete the { $count } selected add-on?
*[other] Delete the { $count } selected add-ons? *[other] Delete the { $count } selected add-ons?
} }
addons-choose-update-window-title = Update Add-ons

View File

@ -10,6 +10,7 @@ import zipfile
from collections import defaultdict from collections import defaultdict
from concurrent.futures import Future from concurrent.futures import Future
from dataclasses import dataclass from dataclasses import dataclass
from datetime import datetime
from typing import IO, Any, Callable, Dict, Iterable, List, Optional, Tuple, Union from typing import IO, Any, Callable, Dict, Iterable, List, Optional, Tuple, Union
from urllib.parse import parse_qs, urlparse from urllib.parse import parse_qs, urlparse
from zipfile import ZipFile from zipfile import ZipFile
@ -104,6 +105,7 @@ class AddonMeta:
max_point_version: int max_point_version: int
branch_index: int branch_index: int
human_version: Optional[str] human_version: Optional[str]
update_enabled: bool
def human_name(self) -> str: def human_name(self) -> str:
return self.provided_name or self.dir_name return self.provided_name or self.dir_name
@ -139,6 +141,7 @@ class AddonMeta:
max_point_version=json_meta.get("max_point_version", 0) or 0, max_point_version=json_meta.get("max_point_version", 0) or 0,
branch_index=json_meta.get("branch_index", 0) or 0, branch_index=json_meta.get("branch_index", 0) or 0,
human_version=json_meta.get("human_version"), human_version=json_meta.get("human_version"),
update_enabled=json_meta.get("update_enabled", True),
) )
@ -242,6 +245,7 @@ class AddonManager:
json_obj["branch_index"] = addon.branch_index json_obj["branch_index"] = addon.branch_index
if addon.human_version is not None: if addon.human_version is not None:
json_obj["human_version"] = addon.human_version json_obj["human_version"] = addon.human_version
json_obj["update_enabled"] = addon.update_enabled
self.writeAddonMeta(addon.dir_name, json_obj) self.writeAddonMeta(addon.dir_name, json_obj)
@ -551,19 +555,19 @@ class AddonManager:
if updated: if updated:
self.write_addon_meta(addon) self.write_addon_meta(addon)
def updates_required(self, items: List[UpdateInfo]) -> List[int]: def updates_required(self, items: List[UpdateInfo]) -> List[UpdateInfo]:
"""Return ids of add-ons requiring an update.""" """Return ids of add-ons requiring an update."""
need_update = [] need_update = []
for item in items: for item in items:
addon = self.addon_meta(str(item.id)) addon = self.addon_meta(str(item.id))
# update if server mtime is newer # update if server mtime is newer
if not addon.is_latest(item.suitable_branch_last_modified): if not addon.is_latest(item.suitable_branch_last_modified):
need_update.append(item.id) need_update.append(item)
elif not addon.compatible() and item.suitable_branch_last_modified > 0: elif not addon.compatible() and item.suitable_branch_last_modified > 0:
# Addon is currently disabled, and a suitable branch was found on the # Addon is currently disabled, and a suitable branch was found on the
# server. Ignore our stored mtime (which may have been set incorrectly # server. Ignore our stored mtime (which may have been set incorrectly
# in the past) and require an update. # in the past) and require an update.
need_update.append(item.id) need_update.append(item)
return need_update return need_update
@ -1132,6 +1136,163 @@ def download_addons(
###################################################################### ######################################################################
class ChooseAddonsToUpdateList(QListWidget):
ADDON_ID_ROLE = 101
def __init__(
self,
parent: QWidget,
mgr: AddonManager,
updated_addons: List[UpdateInfo],
) -> None:
QListWidget.__init__(self, parent)
self.mgr = mgr
self.updated_addons = sorted(
updated_addons, key=lambda addon: addon.suitable_branch_last_modified
)
self.ignore_check_evt = False
self.setup()
self.setContextMenuPolicy(Qt.CustomContextMenu)
qconnect(self.itemClicked, self.on_click)
qconnect(self.itemChanged, self.on_check)
qconnect(self.itemDoubleClicked, self.on_double_click)
qconnect(self.customContextMenuRequested, self.on_context_menu)
def setup(self) -> None:
header_item = QListWidgetItem("", self)
header_item.setFlags(Qt.ItemFlag(Qt.ItemIsUserCheckable | Qt.ItemIsEnabled))
header_item.setBackground(Qt.lightGray)
self.header_item = header_item
for update_info in self.updated_addons:
addon_id = update_info.id
addon_meta = self.mgr.addon_meta(str(addon_id))
update_enabled = addon_meta.update_enabled
addon_name = addon_meta.human_name()
update_timestamp = update_info.suitable_branch_last_modified
update_time = datetime.fromtimestamp(update_timestamp)
addon_label = f"{update_time:%Y-%m-%d} {addon_name}"
item = QListWidgetItem(addon_label, self)
# Not user checkable because it overlaps with itemClicked signal
item.setFlags(Qt.ItemFlag(Qt.ItemIsEnabled))
if update_enabled:
item.setCheckState(Qt.Checked)
else:
item.setCheckState(Qt.Unchecked)
item.setData(self.ADDON_ID_ROLE, addon_id)
self.refresh_header_check_state()
def bool_to_check(self, check_bool: bool) -> Qt.CheckState:
if check_bool:
return Qt.Checked
else:
return Qt.Unchecked
def checked(self, item: QListWidgetItem) -> bool:
return item.checkState() == Qt.Checked
def on_click(self, item: QListWidgetItem) -> None:
if item == self.header_item:
return
checked = self.checked(item)
self.check_item(item, self.bool_to_check(not checked))
self.refresh_header_check_state()
def on_check(self, item: QListWidgetItem) -> None:
if self.ignore_check_evt:
return
if item == self.header_item:
self.header_checked(item.checkState())
def on_double_click(self, item: QListWidgetItem) -> None:
if item == self.header_item:
checked = self.checked(item)
self.check_item(self.header_item, self.bool_to_check(not checked))
self.header_checked(self.bool_to_check(not checked))
def on_context_menu(self, point: QPoint) -> None:
item = self.itemAt(point)
addon_id = item.data(self.ADDON_ID_ROLE)
m = QMenu()
a = m.addAction(tr(TR.ADDONS_VIEW_ADDON_PAGE))
qconnect(a.triggered, lambda _: openLink(f"{aqt.appShared}info/{addon_id}"))
m.exec_(QCursor.pos())
def check_item(self, item: QListWidgetItem, check: Qt.CheckState) -> None:
"call item.setCheckState without triggering on_check"
self.ignore_check_evt = True
item.setCheckState(check)
self.ignore_check_evt = False
def header_checked(self, check: Qt.CheckState) -> None:
for i in range(1, self.count()):
self.check_item(self.item(i), check)
def refresh_header_check_state(self) -> None:
for i in range(1, self.count()):
item = self.item(i)
if not self.checked(item):
self.check_item(self.header_item, Qt.Unchecked)
return
self.check_item(self.header_item, Qt.Checked)
def get_selected_addon_ids(self) -> List[int]:
addon_ids = []
for i in range(1, self.count()):
item = self.item(i)
if self.checked(item):
addon_id = item.data(self.ADDON_ID_ROLE)
addon_ids.append(addon_id)
return addon_ids
def save_check_state(self) -> None:
for i in range(1, self.count()):
item = self.item(i)
addon_id = item.data(self.ADDON_ID_ROLE)
addon_meta = self.mgr.addon_meta(str(addon_id))
addon_meta.update_enabled = self.checked(item)
self.mgr.write_addon_meta(addon_meta)
class ChooseAddonsToUpdateDialog(QDialog):
def __init__(
self, parent: QWidget, mgr: AddonManager, updated_addons: List[UpdateInfo]
) -> None:
QDialog.__init__(self, parent)
self.setWindowTitle(tr(TR.ADDONS_CHOOSE_UPDATE_WINDOW_TITLE))
self.setWindowModality(Qt.WindowModal)
self.mgr = mgr
self.updated_addons = updated_addons
self.setup()
restoreGeom(self, "addonsChooseUpdate")
def setup(self) -> None:
layout = QVBoxLayout()
label = QLabel(tr(TR.ADDONS_THE_FOLLOWING_ADDONS_HAVE_UPDATES_AVAILABLE))
layout.addWidget(label)
addons_list_widget = ChooseAddonsToUpdateList(
self, self.mgr, self.updated_addons
)
layout.addWidget(addons_list_widget)
self.addons_list_widget = addons_list_widget
button_box = QDialogButtonBox(QDialogButtonBox.Ok | QDialogButtonBox.Cancel) # type: ignore
qconnect(button_box.button(QDialogButtonBox.Ok).clicked, self.accept)
qconnect(button_box.button(QDialogButtonBox.Cancel).clicked, self.reject)
layout.addWidget(button_box)
self.setLayout(layout)
def ask(self) -> List[int]:
"Returns a list of selected addons' ids"
ret = self.exec_()
saveGeom(self, "addonsChooseUpdate")
self.addons_list_widget.save_check_state()
if ret == QDialog.Accepted:
return self.addons_list_widget.get_selected_addon_ids()
else:
return []
def fetch_update_info(client: HttpClient, ids: List[int]) -> List[Dict]: def fetch_update_info(client: HttpClient, ids: List[int]) -> List[Dict]:
"""Fetch update info from AnkiWeb in one or more batches.""" """Fetch update info from AnkiWeb in one or more batches."""
all_info: List[Dict] = [] all_info: List[Dict] = []
@ -1164,9 +1325,10 @@ def check_and_prompt_for_updates(
parent: QWidget, parent: QWidget,
mgr: AddonManager, mgr: AddonManager,
on_done: Callable[[List[DownloadLogEntry]], None], on_done: Callable[[List[DownloadLogEntry]], None],
requested_by_user: bool = True,
) -> None: ) -> None:
def on_updates_received(client: HttpClient, items: List[Dict]) -> None: def on_updates_received(client: HttpClient, items: List[Dict]) -> None:
handle_update_info(parent, mgr, client, items, on_done) handle_update_info(parent, mgr, client, items, on_done, requested_by_user)
check_for_updates(mgr, on_updates_received) check_for_updates(mgr, on_updates_received)
@ -1235,35 +1397,39 @@ def handle_update_info(
client: HttpClient, client: HttpClient,
items: List[Dict], items: List[Dict],
on_done: Callable[[List[DownloadLogEntry]], None], on_done: Callable[[List[DownloadLogEntry]], None],
requested_by_user: bool = True,
) -> None: ) -> None:
update_info = mgr.extract_update_info(items) update_info = mgr.extract_update_info(items)
mgr.update_supported_versions(update_info) mgr.update_supported_versions(update_info)
updated_ids = mgr.updates_required(update_info) updated_addons = mgr.updates_required(update_info)
if not updated_ids: if not updated_addons:
on_done([]) on_done([])
return return
prompt_to_update(parent, mgr, client, updated_ids, on_done) prompt_to_update(parent, mgr, client, updated_addons, on_done, requested_by_user)
def prompt_to_update( def prompt_to_update(
parent: QWidget, parent: QWidget,
mgr: AddonManager, mgr: AddonManager,
client: HttpClient, client: HttpClient,
ids: List[int], updated_addons: List[UpdateInfo],
on_done: Callable[[List[DownloadLogEntry]], None], on_done: Callable[[List[DownloadLogEntry]], None],
requested_by_user: bool = True,
) -> None: ) -> None:
names = map(lambda x: mgr.addonName(str(x)), ids) if not requested_by_user:
if not askUser( prompt_update = False
tr(TR.ADDONS_THE_FOLLOWING_ADDONS_HAVE_UPDATES_AVAILABLE) for addon in updated_addons:
+ "\n\n" if mgr.addon_meta(str(addon.id)).update_enabled:
+ "\n".join(names) prompt_update = True
): if not prompt_update:
# on_done is not called if the user cancels return
return
ids = ChooseAddonsToUpdateDialog(parent, mgr, updated_addons).ask()
if not ids:
return
download_addons(parent, mgr, ids, on_done, client) download_addons(parent, mgr, ids, on_done, client)

View File

@ -848,7 +848,10 @@ title="%s" %s>%s</button>""" % (
if elap > 86_400: if elap > 86_400:
check_and_prompt_for_updates( check_and_prompt_for_updates(
self, self.addonManager, self.on_updates_installed self,
self.addonManager,
self.on_updates_installed,
requested_by_user=False,
) )
self.pm.set_last_addon_update_check(intTime()) self.pm.set_last_addon_update_check(intTime())