update find&replace in browser
sadly the UI still stutters on large selections - the calls to get the selected rows from Qt are really slow.
This commit is contained in:
parent
8b557ec382
commit
1852e32183
@ -111,6 +111,10 @@ class DeckIsFilteredError(Exception):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class InvalidInput(StringError):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
def proto_exception_to_native(err: pb.BackendError) -> Exception:
|
def proto_exception_to_native(err: pb.BackendError) -> Exception:
|
||||||
val = err.WhichOneof("value")
|
val = err.WhichOneof("value")
|
||||||
if val == "interrupted":
|
if val == "interrupted":
|
||||||
@ -126,7 +130,7 @@ def proto_exception_to_native(err: pb.BackendError) -> Exception:
|
|||||||
elif val == "template_parse":
|
elif val == "template_parse":
|
||||||
return TemplateError(err.localized)
|
return TemplateError(err.localized)
|
||||||
elif val == "invalid_input":
|
elif val == "invalid_input":
|
||||||
return StringError(err.localized)
|
return InvalidInput(err.localized)
|
||||||
elif val == "json_error":
|
elif val == "json_error":
|
||||||
return StringError(err.localized)
|
return StringError(err.localized)
|
||||||
elif val == "not_found_error":
|
elif val == "not_found_error":
|
||||||
@ -747,7 +751,8 @@ class RustBackend:
|
|||||||
|
|
||||||
def field_names_for_note_ids(self, nids: List[int]) -> Sequence[str]:
|
def field_names_for_note_ids(self, nids: List[int]) -> Sequence[str]:
|
||||||
return self._run_command(
|
return self._run_command(
|
||||||
pb.BackendInput(field_names_for_notes=pb.FieldNamesForNotesIn(nids=nids))
|
pb.BackendInput(field_names_for_notes=pb.FieldNamesForNotesIn(nids=nids)),
|
||||||
|
release_gil=True,
|
||||||
).field_names_for_notes.fields
|
).field_names_for_notes.fields
|
||||||
|
|
||||||
def find_and_replace(
|
def find_and_replace(
|
||||||
@ -769,7 +774,8 @@ class RustBackend:
|
|||||||
match_case=not nocase,
|
match_case=not nocase,
|
||||||
field_name=field_name,
|
field_name=field_name,
|
||||||
)
|
)
|
||||||
)
|
),
|
||||||
|
release_gil=True,
|
||||||
).find_and_replace
|
).find_and_replace
|
||||||
|
|
||||||
|
|
||||||
|
@ -24,7 +24,7 @@ from anki.decks import DeckManager
|
|||||||
from anki.lang import _, ngettext
|
from anki.lang import _, ngettext
|
||||||
from anki.models import NoteType
|
from anki.models import NoteType
|
||||||
from anki.notes import Note
|
from anki.notes import Note
|
||||||
from anki.rsbackend import TR, DeckTreeNode
|
from anki.rsbackend import TR, DeckTreeNode, InvalidInput
|
||||||
from anki.utils import htmlToTextLine, ids2str, intTime, isMac, isWin
|
from anki.utils import htmlToTextLine, ids2str, intTime, isMac, isWin
|
||||||
from aqt import AnkiQt, gui_hooks
|
from aqt import AnkiQt, gui_hooks
|
||||||
from aqt.editor import Editor
|
from aqt.editor import Editor
|
||||||
@ -1926,13 +1926,21 @@ update cards set usn=?, mod=?, did=? where id in """
|
|||||||
def onFindReplace(self):
|
def onFindReplace(self):
|
||||||
self.editor.saveNow(self._onFindReplace)
|
self.editor.saveNow(self._onFindReplace)
|
||||||
|
|
||||||
def _onFindReplace(self):
|
def _onFindReplace(self) -> None:
|
||||||
sf = self.selectedNotes()
|
nids = self.selectedNotes()
|
||||||
if not sf:
|
if not nids:
|
||||||
return
|
return
|
||||||
import anki.find
|
import anki.find
|
||||||
|
|
||||||
fields = anki.find.fieldNamesForNotes(self.mw.col, sf)
|
def find():
|
||||||
|
return anki.find.fieldNamesForNotes(self.mw.col, nids)
|
||||||
|
|
||||||
|
def on_done(fut):
|
||||||
|
self._on_find_replace_diag(fut.result(), nids)
|
||||||
|
|
||||||
|
self.mw.taskman.with_progress(find, on_done, self)
|
||||||
|
|
||||||
|
def _on_find_replace_diag(self, fields: List[str], nids: List[int]) -> None:
|
||||||
d = QDialog(self)
|
d = QDialog(self)
|
||||||
frm = aqt.forms.findreplace.Ui_Dialog()
|
frm = aqt.forms.findreplace.Ui_Dialog()
|
||||||
frm.setupUi(d)
|
frm.setupUi(d)
|
||||||
@ -1948,34 +1956,38 @@ update cards set usn=?, mod=?, did=? where id in """
|
|||||||
field = None
|
field = None
|
||||||
else:
|
else:
|
||||||
field = fields[frm.field.currentIndex() - 1]
|
field = fields[frm.field.currentIndex() - 1]
|
||||||
|
|
||||||
|
search = frm.find.text()
|
||||||
|
replace = frm.replace.text()
|
||||||
|
regex = frm.re.isChecked()
|
||||||
|
nocase = frm.ignoreCase.isChecked()
|
||||||
|
|
||||||
self.mw.checkpoint(_("Find and Replace"))
|
self.mw.checkpoint(_("Find and Replace"))
|
||||||
self.mw.progress.start()
|
# starts progress dialog as well
|
||||||
self.model.beginReset()
|
self.model.beginReset()
|
||||||
try:
|
|
||||||
changed = self.col.findReplace(
|
def do_search():
|
||||||
sf,
|
return self.col.findReplace(nids, search, replace, regex, field, nocase)
|
||||||
str(frm.find.text()),
|
|
||||||
str(frm.replace.text()),
|
def on_done(fut):
|
||||||
frm.re.isChecked(),
|
|
||||||
field,
|
|
||||||
frm.ignoreCase.isChecked(),
|
|
||||||
)
|
|
||||||
except sre_constants.error:
|
|
||||||
showInfo(_("Invalid regular expression."), parent=self)
|
|
||||||
return
|
|
||||||
else:
|
|
||||||
self.search()
|
self.search()
|
||||||
self.mw.requireReset()
|
self.mw.requireReset()
|
||||||
finally:
|
|
||||||
self.model.endReset()
|
self.model.endReset()
|
||||||
self.mw.progress.finish()
|
|
||||||
showInfo(
|
total = len(nids)
|
||||||
ngettext(
|
try:
|
||||||
"%(a)d of %(b)d note updated", "%(a)d of %(b)d notes updated", len(sf)
|
changed = fut.result()
|
||||||
|
except InvalidInput as e:
|
||||||
|
# failed regex
|
||||||
|
showWarning(str(e))
|
||||||
|
return
|
||||||
|
|
||||||
|
showInfo(
|
||||||
|
tr(TR.FINDREPLACE_NOTES_UPDATED, changed=changed, total=total),
|
||||||
|
parent=self,
|
||||||
)
|
)
|
||||||
% {"a": changed, "b": len(sf),},
|
|
||||||
parent=self,
|
self.mw.taskman.run_in_background(do_search, on_done)
|
||||||
)
|
|
||||||
|
|
||||||
def onFindReplaceHelp(self):
|
def onFindReplaceHelp(self):
|
||||||
openHelp("findreplace")
|
openHelp("findreplace")
|
||||||
|
4
rslib/ftl/findreplace.ftl
Normal file
4
rslib/ftl/findreplace.ftl
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
findreplace-notes-updated = { $total ->
|
||||||
|
[one] {$changed} of {$total} note updated
|
||||||
|
*[other] {$changed} of {$total} notes updated
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user