split tags.rs up
This commit is contained in:
parent
1621f2ff26
commit
33d467e688
196
rslib/src/tags/dragdrop.rs
Normal file
196
rslib/src/tags/dragdrop.rs
Normal file
@ -0,0 +1,196 @@
|
|||||||
|
// Copyright: Ankitects Pty Ltd and contributors
|
||||||
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
||||||
|
|
||||||
|
use regex::{NoExpand, Regex, Replacer};
|
||||||
|
|
||||||
|
use super::split_tags;
|
||||||
|
use crate::{notes::TransformNoteOutput, prelude::*};
|
||||||
|
|
||||||
|
impl Collection {
|
||||||
|
pub fn drag_drop_tags(
|
||||||
|
&mut self,
|
||||||
|
source_tags: &[String],
|
||||||
|
target_tag: Option<String>,
|
||||||
|
) -> Result<()> {
|
||||||
|
let source_tags_and_outputs: Vec<_> = source_tags
|
||||||
|
.iter()
|
||||||
|
// generate resulting names and filter out invalid ones
|
||||||
|
.flat_map(|source_tag| {
|
||||||
|
if let Some(output_name) = drag_drop_tag_name(source_tag, target_tag.as_deref()) {
|
||||||
|
Some((source_tag, output_name))
|
||||||
|
} else {
|
||||||
|
// invalid rename, ignore this tag
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
let regexps_and_replacements = source_tags_and_outputs
|
||||||
|
.iter()
|
||||||
|
// convert the names into regexps/replacements
|
||||||
|
.map(|(tag, output)| {
|
||||||
|
regex_matching_tag_and_children_in_single_tag(tag).map(|regex| (regex, output))
|
||||||
|
})
|
||||||
|
.collect::<Result<Vec<_>>>()?;
|
||||||
|
|
||||||
|
// locate notes that match them
|
||||||
|
let mut nids = vec![];
|
||||||
|
self.storage.for_each_note_tags(|nid, tags| {
|
||||||
|
for tag in split_tags(&tags) {
|
||||||
|
for (regex, _) in ®exps_and_replacements {
|
||||||
|
if regex.is_match(&tag) {
|
||||||
|
nids.push(nid);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
})?;
|
||||||
|
|
||||||
|
if nids.is_empty() {
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
|
||||||
|
// update notes
|
||||||
|
self.transact_no_undo(|col| {
|
||||||
|
// clear the existing original tags
|
||||||
|
for (source_tag, _) in &source_tags_and_outputs {
|
||||||
|
col.storage.clear_tag_and_children(source_tag)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
col.transform_notes(&nids, |note, _nt| {
|
||||||
|
let mut changed = false;
|
||||||
|
for (re, repl) in ®exps_and_replacements {
|
||||||
|
if note.replace_tags(re, NoExpand(&repl).by_ref()) {
|
||||||
|
changed = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(TransformNoteOutput {
|
||||||
|
changed,
|
||||||
|
generate_cards: false,
|
||||||
|
mark_modified: true,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})?;
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Arguments are expected in 'human' form with an :: separator.
|
||||||
|
pub(crate) fn drag_drop_tag_name(dragged: &str, dropped: Option<&str>) -> Option<String> {
|
||||||
|
let dragged_base = dragged.rsplit("::").next().unwrap();
|
||||||
|
if let Some(dropped) = dropped {
|
||||||
|
if dropped.starts_with(dragged) {
|
||||||
|
// foo onto foo::bar, or foo onto itself -> no-op
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
// foo::bar onto baz -> baz::bar
|
||||||
|
Some(format!("{}::{}", dropped, dragged_base))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// foo::bar onto top level -> bar
|
||||||
|
Some(dragged_base.into())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// A regex that will match a string tag that has been split from a list.
|
||||||
|
fn regex_matching_tag_and_children_in_single_tag(tag: &str) -> Result<Regex> {
|
||||||
|
Regex::new(&format!(
|
||||||
|
r#"(?ix)
|
||||||
|
^
|
||||||
|
{}
|
||||||
|
# optional children
|
||||||
|
(::.+)?
|
||||||
|
$
|
||||||
|
"#,
|
||||||
|
regex::escape(tag)
|
||||||
|
))
|
||||||
|
.map_err(Into::into)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
use crate::collection::open_test_collection;
|
||||||
|
|
||||||
|
fn alltags(col: &Collection) -> Vec<String> {
|
||||||
|
col.storage
|
||||||
|
.all_tags()
|
||||||
|
.unwrap()
|
||||||
|
.into_iter()
|
||||||
|
.map(|t| t.name)
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn dragdrop() -> Result<()> {
|
||||||
|
let mut col = open_test_collection();
|
||||||
|
let nt = col.get_notetype_by_name("Basic")?.unwrap();
|
||||||
|
for tag in &[
|
||||||
|
"another",
|
||||||
|
"parent1::child1::grandchild1",
|
||||||
|
"parent1::child1",
|
||||||
|
"parent1",
|
||||||
|
"parent2",
|
||||||
|
"yet::another",
|
||||||
|
] {
|
||||||
|
let mut note = nt.new_note();
|
||||||
|
note.tags.push(tag.to_string());
|
||||||
|
col.add_note(&mut note, DeckID(1))?;
|
||||||
|
}
|
||||||
|
|
||||||
|
// two decks with the same base name; they both get mapped
|
||||||
|
// to parent1::another
|
||||||
|
col.drag_drop_tags(
|
||||||
|
&["another".to_string(), "yet::another".to_string()],
|
||||||
|
Some("parent1".to_string()),
|
||||||
|
)?;
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
alltags(&col),
|
||||||
|
&[
|
||||||
|
"parent1",
|
||||||
|
"parent1::another",
|
||||||
|
"parent1::child1",
|
||||||
|
"parent1::child1::grandchild1",
|
||||||
|
"parent2",
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// child and children moved to parent2
|
||||||
|
col.drag_drop_tags(
|
||||||
|
&["parent1::child1".to_string()],
|
||||||
|
Some("parent2".to_string()),
|
||||||
|
)?;
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
alltags(&col),
|
||||||
|
&[
|
||||||
|
"parent1",
|
||||||
|
"parent1::another",
|
||||||
|
"parent2",
|
||||||
|
"parent2::child1",
|
||||||
|
"parent2::child1::grandchild1",
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
// empty target reparents to root
|
||||||
|
col.drag_drop_tags(&["parent1::another".to_string()], None)?;
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
alltags(&col),
|
||||||
|
&[
|
||||||
|
"another",
|
||||||
|
"parent1",
|
||||||
|
"parent2",
|
||||||
|
"parent2::child1",
|
||||||
|
"parent2::child1::grandchild1",
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
@ -1,24 +1,19 @@
|
|||||||
// Copyright: Ankitects Pty Ltd and contributors
|
// Copyright: Ankitects Pty Ltd and contributors
|
||||||
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
||||||
|
|
||||||
|
mod dragdrop;
|
||||||
mod prefix_replacer;
|
mod prefix_replacer;
|
||||||
|
mod register;
|
||||||
|
mod remove;
|
||||||
|
mod rename;
|
||||||
|
mod selectednotes;
|
||||||
|
mod tree;
|
||||||
pub(crate) mod undo;
|
pub(crate) mod undo;
|
||||||
|
|
||||||
use crate::{
|
|
||||||
backend_proto::TagTreeNode,
|
|
||||||
collection::Collection,
|
|
||||||
err::{AnkiError, Result},
|
|
||||||
notes::{NoteID, TransformNoteOutput},
|
|
||||||
prelude::*,
|
|
||||||
text::{normalize_to_nfc, to_re},
|
|
||||||
types::Usn,
|
|
||||||
};
|
|
||||||
|
|
||||||
use prefix_replacer::PrefixReplacer;
|
|
||||||
use regex::{NoExpand, Regex, Replacer};
|
|
||||||
use std::{borrow::Cow, collections::HashSet, iter::Peekable};
|
|
||||||
use unicase::UniCase;
|
use unicase::UniCase;
|
||||||
|
|
||||||
|
use crate::prelude::*;
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq)]
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
pub struct Tag {
|
pub struct Tag {
|
||||||
pub name: String,
|
pub name: String,
|
||||||
@ -52,41 +47,6 @@ fn is_tag_separator(c: char) -> bool {
|
|||||||
c == ' ' || c == '\u{3000}'
|
c == ' ' || c == '\u{3000}'
|
||||||
}
|
}
|
||||||
|
|
||||||
fn invalid_char_for_tag(c: char) -> bool {
|
|
||||||
c.is_ascii_control() || is_tag_separator(c) || c == '"'
|
|
||||||
}
|
|
||||||
|
|
||||||
fn normalized_tag_name_component(comp: &str) -> Cow<str> {
|
|
||||||
let mut out = normalize_to_nfc(comp);
|
|
||||||
if out.contains(invalid_char_for_tag) {
|
|
||||||
out = out.replace(invalid_char_for_tag, "").into();
|
|
||||||
}
|
|
||||||
let trimmed = out.trim();
|
|
||||||
if trimmed.is_empty() {
|
|
||||||
"blank".to_string().into()
|
|
||||||
} else if trimmed.len() != out.len() {
|
|
||||||
trimmed.to_string().into()
|
|
||||||
} else {
|
|
||||||
out
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn normalize_tag_name(name: &str) -> Cow<str> {
|
|
||||||
if name
|
|
||||||
.split("::")
|
|
||||||
.any(|comp| matches!(normalized_tag_name_component(comp), Cow::Owned(_)))
|
|
||||||
{
|
|
||||||
let comps: Vec<_> = name
|
|
||||||
.split("::")
|
|
||||||
.map(normalized_tag_name_component)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
comps.join("::").into()
|
|
||||||
} else {
|
|
||||||
// no changes required
|
|
||||||
name.into()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn immediate_parent_name_unicase(tag_name: UniCase<&str>) -> Option<UniCase<&str>> {
|
fn immediate_parent_name_unicase(tag_name: UniCase<&str>) -> Option<UniCase<&str>> {
|
||||||
tag_name.rsplitn(2, '\x1f').nth(1).map(UniCase::new)
|
tag_name.rsplitn(2, '\x1f').nth(1).map(UniCase::new)
|
||||||
}
|
}
|
||||||
@ -95,224 +55,7 @@ fn immediate_parent_name_str(tag_name: &str) -> Option<&str> {
|
|||||||
tag_name.rsplitn(2, "::").nth(1)
|
tag_name.rsplitn(2, "::").nth(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Arguments are expected in 'human' form with an :: separator.
|
|
||||||
pub(crate) fn drag_drop_tag_name(dragged: &str, dropped: Option<&str>) -> Option<String> {
|
|
||||||
let dragged_base = dragged.rsplit("::").next().unwrap();
|
|
||||||
if let Some(dropped) = dropped {
|
|
||||||
if dropped.starts_with(dragged) {
|
|
||||||
// foo onto foo::bar, or foo onto itself -> no-op
|
|
||||||
None
|
|
||||||
} else {
|
|
||||||
// foo::bar onto baz -> baz::bar
|
|
||||||
Some(format!("{}::{}", dropped, dragged_base))
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// foo::bar onto top level -> bar
|
|
||||||
Some(dragged_base.into())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// For the given tag, check if immediate parent exists. If so, add
|
|
||||||
/// tag and return.
|
|
||||||
/// If the immediate parent is missing, check and add any missing parents.
|
|
||||||
/// This should ensure that if an immediate parent is found, all ancestors
|
|
||||||
/// are guaranteed to already exist.
|
|
||||||
fn add_tag_and_missing_parents<'a, 'b>(
|
|
||||||
all: &'a mut HashSet<UniCase<&'b str>>,
|
|
||||||
missing: &'a mut Vec<UniCase<&'b str>>,
|
|
||||||
tag_name: UniCase<&'b str>,
|
|
||||||
) {
|
|
||||||
if let Some(parent) = immediate_parent_name_unicase(tag_name) {
|
|
||||||
if !all.contains(&parent) {
|
|
||||||
missing.push(parent);
|
|
||||||
add_tag_and_missing_parents(all, missing, parent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// finally, add provided tag
|
|
||||||
all.insert(tag_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Append any missing parents. Caller must sort afterwards.
|
|
||||||
fn add_missing_parents(tags: &mut Vec<Tag>) {
|
|
||||||
let mut all_names: HashSet<UniCase<&str>> = HashSet::new();
|
|
||||||
let mut missing = vec![];
|
|
||||||
for tag in &*tags {
|
|
||||||
add_tag_and_missing_parents(&mut all_names, &mut missing, UniCase::new(&tag.name))
|
|
||||||
}
|
|
||||||
let mut missing: Vec<_> = missing
|
|
||||||
.into_iter()
|
|
||||||
.map(|n| Tag::new(n.to_string(), Usn(0)))
|
|
||||||
.collect();
|
|
||||||
tags.append(&mut missing);
|
|
||||||
}
|
|
||||||
|
|
||||||
fn tags_to_tree(mut tags: Vec<Tag>) -> TagTreeNode {
|
|
||||||
for tag in &mut tags {
|
|
||||||
tag.name = tag.name.replace("::", "\x1f");
|
|
||||||
}
|
|
||||||
add_missing_parents(&mut tags);
|
|
||||||
tags.sort_unstable_by(|a, b| UniCase::new(&a.name).cmp(&UniCase::new(&b.name)));
|
|
||||||
let mut top = TagTreeNode::default();
|
|
||||||
let mut it = tags.into_iter().peekable();
|
|
||||||
add_child_nodes(&mut it, &mut top);
|
|
||||||
|
|
||||||
top
|
|
||||||
}
|
|
||||||
|
|
||||||
fn add_child_nodes(tags: &mut Peekable<impl Iterator<Item = Tag>>, parent: &mut TagTreeNode) {
|
|
||||||
while let Some(tag) = tags.peek() {
|
|
||||||
let split_name: Vec<_> = tag.name.split('\x1f').collect();
|
|
||||||
match split_name.len() as u32 {
|
|
||||||
l if l <= parent.level => {
|
|
||||||
// next item is at a higher level
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
l if l == parent.level + 1 => {
|
|
||||||
// next item is an immediate descendent of parent
|
|
||||||
parent.children.push(TagTreeNode {
|
|
||||||
name: (*split_name.last().unwrap()).into(),
|
|
||||||
children: vec![],
|
|
||||||
level: parent.level + 1,
|
|
||||||
expanded: tag.expanded,
|
|
||||||
});
|
|
||||||
tags.next();
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
// next item is at a lower level
|
|
||||||
if let Some(last_child) = parent.children.last_mut() {
|
|
||||||
add_child_nodes(tags, last_child)
|
|
||||||
} else {
|
|
||||||
// immediate parent is missing
|
|
||||||
tags.next();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Collection {
|
impl Collection {
|
||||||
pub fn tag_tree(&mut self) -> Result<TagTreeNode> {
|
|
||||||
let tags = self.storage.all_tags()?;
|
|
||||||
let tree = tags_to_tree(tags);
|
|
||||||
|
|
||||||
Ok(tree)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Given a list of tags, fix case, ordering and duplicates.
|
|
||||||
/// Returns true if any new tags were added.
|
|
||||||
pub(crate) fn canonify_tags(
|
|
||||||
&mut self,
|
|
||||||
tags: Vec<String>,
|
|
||||||
usn: Usn,
|
|
||||||
) -> Result<(Vec<String>, bool)> {
|
|
||||||
let mut seen = HashSet::new();
|
|
||||||
let mut added = false;
|
|
||||||
|
|
||||||
let tags: Vec<_> = tags.iter().flat_map(|t| split_tags(t)).collect();
|
|
||||||
for tag in tags {
|
|
||||||
let mut tag = Tag::new(tag.to_string(), usn);
|
|
||||||
added |= self.register_tag(&mut tag)?;
|
|
||||||
seen.insert(UniCase::new(tag.name));
|
|
||||||
}
|
|
||||||
|
|
||||||
// exit early if no non-empty tags
|
|
||||||
if seen.is_empty() {
|
|
||||||
return Ok((vec![], added));
|
|
||||||
}
|
|
||||||
|
|
||||||
// return the sorted, canonified tags
|
|
||||||
let mut tags = seen.into_iter().collect::<Vec<_>>();
|
|
||||||
tags.sort_unstable();
|
|
||||||
let tags: Vec<_> = tags.into_iter().map(|s| s.into_inner()).collect();
|
|
||||||
|
|
||||||
Ok((tags, added))
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Adjust tag casing to match any existing parents, and register it if it's not already
|
|
||||||
/// in the tags list. True if the tag was added and not already in tag list.
|
|
||||||
/// In the case the tag is already registered, tag will be mutated to match the existing
|
|
||||||
/// name.
|
|
||||||
pub(crate) fn register_tag(&mut self, tag: &mut Tag) -> Result<bool> {
|
|
||||||
let is_new = self.prepare_tag_for_registering(tag)?;
|
|
||||||
if is_new {
|
|
||||||
self.register_tag_undoable(&tag)?;
|
|
||||||
}
|
|
||||||
Ok(is_new)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn register_tag_string(&mut self, tag: String, usn: Usn) -> Result<bool> {
|
|
||||||
let mut tag = Tag::new(tag, usn);
|
|
||||||
self.register_tag(&mut tag)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Create a tag object, normalize text, and match parents/existing case if available.
|
|
||||||
/// True if tag is new.
|
|
||||||
fn prepare_tag_for_registering(&self, tag: &mut Tag) -> Result<bool> {
|
|
||||||
let normalized_name = normalize_tag_name(&tag.name);
|
|
||||||
if normalized_name.is_empty() {
|
|
||||||
// this should not be possible
|
|
||||||
return Err(AnkiError::invalid_input("blank tag"));
|
|
||||||
}
|
|
||||||
if let Some(existing_tag) = self.storage.get_tag(&normalized_name)? {
|
|
||||||
tag.name = existing_tag.name;
|
|
||||||
Ok(false)
|
|
||||||
} else {
|
|
||||||
if let Some(new_name) = self.adjusted_case_for_parents(&normalized_name)? {
|
|
||||||
tag.name = new_name;
|
|
||||||
} else if let Cow::Owned(new_name) = normalized_name {
|
|
||||||
tag.name = new_name;
|
|
||||||
}
|
|
||||||
Ok(true)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// If parent tag(s) exist and differ in case, return a rewritten tag.
|
|
||||||
fn adjusted_case_for_parents(&self, tag: &str) -> Result<Option<String>> {
|
|
||||||
if let Some(parent_tag) = self.first_existing_parent_tag(&tag)? {
|
|
||||||
let child_split: Vec<_> = tag.split("::").collect();
|
|
||||||
let parent_count = parent_tag.matches("::").count() + 1;
|
|
||||||
Ok(Some(format!(
|
|
||||||
"{}::{}",
|
|
||||||
parent_tag,
|
|
||||||
&child_split[parent_count..].join("::")
|
|
||||||
)))
|
|
||||||
} else {
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn first_existing_parent_tag(&self, mut tag: &str) -> Result<Option<String>> {
|
|
||||||
while let Some(parent_name) = immediate_parent_name_str(tag) {
|
|
||||||
if let Some(parent_tag) = self.storage.preferred_tag_case(parent_name)? {
|
|
||||||
return Ok(Some(parent_tag));
|
|
||||||
}
|
|
||||||
tag = parent_name;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(None)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Remove tags not referenced by notes, returning removed count.
|
|
||||||
pub fn clear_unused_tags(&mut self) -> Result<OpOutput<usize>> {
|
|
||||||
self.transact(Op::ClearUnusedTags, |col| col.clear_unused_tags_inner())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn clear_unused_tags_inner(&mut self) -> Result<usize> {
|
|
||||||
let mut count = 0;
|
|
||||||
let in_notes = self.storage.all_tags_in_notes()?;
|
|
||||||
let need_remove = self
|
|
||||||
.storage
|
|
||||||
.all_tags()?
|
|
||||||
.into_iter()
|
|
||||||
.filter(|tag| !in_notes.contains(&tag.name));
|
|
||||||
for tag in need_remove {
|
|
||||||
self.remove_single_tag_undoable(tag)?;
|
|
||||||
count += 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(count)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub(crate) fn set_tag_expanded(&self, name: &str, expanded: bool) -> Result<()> {
|
pub(crate) fn set_tag_expanded(&self, name: &str, expanded: bool) -> Result<()> {
|
||||||
let mut name = name;
|
let mut name = name;
|
||||||
let tag;
|
let tag;
|
||||||
@ -324,584 +67,4 @@ impl Collection {
|
|||||||
}
|
}
|
||||||
self.storage.set_tag_collapsed(name, !expanded)
|
self.storage.set_tag_collapsed(name, !expanded)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn replace_tags_for_notes_inner<R: Replacer>(
|
|
||||||
&mut self,
|
|
||||||
nids: &[NoteID],
|
|
||||||
tags: &[Regex],
|
|
||||||
mut repl: R,
|
|
||||||
) -> Result<OpOutput<usize>> {
|
|
||||||
self.transact(Op::UpdateTag, |col| {
|
|
||||||
col.transform_notes(nids, |note, _nt| {
|
|
||||||
let mut changed = false;
|
|
||||||
for re in tags {
|
|
||||||
if note.replace_tags(re, repl.by_ref()) {
|
|
||||||
changed = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(TransformNoteOutput {
|
|
||||||
changed,
|
|
||||||
generate_cards: false,
|
|
||||||
mark_modified: true,
|
|
||||||
})
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Apply the provided list of regular expressions to note tags,
|
|
||||||
/// saving any modified notes.
|
|
||||||
pub fn replace_tags_for_notes(
|
|
||||||
&mut self,
|
|
||||||
nids: &[NoteID],
|
|
||||||
tags: &str,
|
|
||||||
repl: &str,
|
|
||||||
regex: bool,
|
|
||||||
) -> Result<OpOutput<usize>> {
|
|
||||||
// generate regexps
|
|
||||||
let tags = split_tags(tags)
|
|
||||||
.map(|tag| {
|
|
||||||
let tag = if regex { tag.into() } else { to_re(tag) };
|
|
||||||
Regex::new(&format!("(?i)^{}(::.*)?$", tag))
|
|
||||||
.map_err(|_| AnkiError::invalid_input("invalid regex"))
|
|
||||||
})
|
|
||||||
.collect::<Result<Vec<Regex>>>()?;
|
|
||||||
if !regex {
|
|
||||||
self.replace_tags_for_notes_inner(nids, &tags, NoExpand(repl))
|
|
||||||
} else {
|
|
||||||
self.replace_tags_for_notes_inner(nids, &tags, repl)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn add_tags_to_notes(&mut self, nids: &[NoteID], tags: &str) -> Result<OpOutput<usize>> {
|
|
||||||
let tags: Vec<_> = split_tags(tags).collect();
|
|
||||||
let matcher = regex::RegexSet::new(
|
|
||||||
tags.iter()
|
|
||||||
.map(|s| regex::escape(s))
|
|
||||||
.map(|s| format!("(?i)^{}$", s)),
|
|
||||||
)
|
|
||||||
.map_err(|_| AnkiError::invalid_input("invalid regex"))?;
|
|
||||||
|
|
||||||
self.transact(Op::UpdateTag, |col| {
|
|
||||||
col.transform_notes(nids, |note, _nt| {
|
|
||||||
let mut need_to_add = true;
|
|
||||||
let mut match_count = 0;
|
|
||||||
for tag in ¬e.tags {
|
|
||||||
if matcher.is_match(tag) {
|
|
||||||
match_count += 1;
|
|
||||||
}
|
|
||||||
if match_count == tags.len() {
|
|
||||||
need_to_add = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if need_to_add {
|
|
||||||
note.tags.extend(tags.iter().map(|&s| s.to_string()))
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(TransformNoteOutput {
|
|
||||||
changed: need_to_add,
|
|
||||||
generate_cards: false,
|
|
||||||
mark_modified: true,
|
|
||||||
})
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn drag_drop_tags(
|
|
||||||
&mut self,
|
|
||||||
source_tags: &[String],
|
|
||||||
target_tag: Option<String>,
|
|
||||||
) -> Result<()> {
|
|
||||||
let source_tags_and_outputs: Vec<_> = source_tags
|
|
||||||
.iter()
|
|
||||||
// generate resulting names and filter out invalid ones
|
|
||||||
.flat_map(|source_tag| {
|
|
||||||
if let Some(output_name) = drag_drop_tag_name(source_tag, target_tag.as_deref()) {
|
|
||||||
Some((source_tag, output_name))
|
|
||||||
} else {
|
|
||||||
// invalid rename, ignore this tag
|
|
||||||
None
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
let regexps_and_replacements = source_tags_and_outputs
|
|
||||||
.iter()
|
|
||||||
// convert the names into regexps/replacements
|
|
||||||
.map(|(tag, output)| {
|
|
||||||
regex_matching_tag_and_children_in_single_tag(tag).map(|regex| (regex, output))
|
|
||||||
})
|
|
||||||
.collect::<Result<Vec<_>>>()?;
|
|
||||||
|
|
||||||
// locate notes that match them
|
|
||||||
let mut nids = vec![];
|
|
||||||
self.storage.for_each_note_tags(|nid, tags| {
|
|
||||||
for tag in split_tags(&tags) {
|
|
||||||
for (regex, _) in ®exps_and_replacements {
|
|
||||||
if regex.is_match(&tag) {
|
|
||||||
nids.push(nid);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
})?;
|
|
||||||
|
|
||||||
if nids.is_empty() {
|
|
||||||
return Ok(());
|
|
||||||
}
|
|
||||||
|
|
||||||
// update notes
|
|
||||||
self.transact_no_undo(|col| {
|
|
||||||
// clear the existing original tags
|
|
||||||
for (source_tag, _) in &source_tags_and_outputs {
|
|
||||||
col.storage.clear_tag_and_children(source_tag)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
col.transform_notes(&nids, |note, _nt| {
|
|
||||||
let mut changed = false;
|
|
||||||
for (re, repl) in ®exps_and_replacements {
|
|
||||||
if note.replace_tags(re, NoExpand(&repl).by_ref()) {
|
|
||||||
changed = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(TransformNoteOutput {
|
|
||||||
changed,
|
|
||||||
generate_cards: false,
|
|
||||||
mark_modified: true,
|
|
||||||
})
|
|
||||||
})
|
|
||||||
})?;
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Rename a given tag and its children on all notes that reference it, returning changed
|
|
||||||
/// note count.
|
|
||||||
pub fn rename_tag(&mut self, old_prefix: &str, new_prefix: &str) -> Result<OpOutput<usize>> {
|
|
||||||
self.transact(Op::RenameTag, |col| {
|
|
||||||
col.rename_tag_inner(old_prefix, new_prefix)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
fn rename_tag_inner(&mut self, old_prefix: &str, new_prefix: &str) -> Result<usize> {
|
|
||||||
if new_prefix.contains(' ') {
|
|
||||||
return Err(AnkiError::invalid_input(
|
|
||||||
"replacement name can not contain a space",
|
|
||||||
));
|
|
||||||
}
|
|
||||||
if new_prefix.trim().is_empty() {
|
|
||||||
return Err(AnkiError::invalid_input(
|
|
||||||
"replacement name must not be empty",
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
let usn = self.usn()?;
|
|
||||||
|
|
||||||
// match existing case if available, and ensure normalized.
|
|
||||||
let mut tag = Tag::new(new_prefix.to_string(), usn);
|
|
||||||
self.prepare_tag_for_registering(&mut tag)?;
|
|
||||||
let new_prefix = &tag.name;
|
|
||||||
|
|
||||||
// gather tags that need replacing
|
|
||||||
let mut re = PrefixReplacer::new(old_prefix)?;
|
|
||||||
let matched_notes = self
|
|
||||||
.storage
|
|
||||||
.get_note_tags_by_predicate(|tags| re.is_match(tags))?;
|
|
||||||
let match_count = matched_notes.len();
|
|
||||||
if match_count == 0 {
|
|
||||||
// no matches; exit early so we don't clobber the empty tag entries
|
|
||||||
return Ok(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
// remove old prefix from the tag list
|
|
||||||
for tag in self.storage.get_tags_by_predicate(|tag| re.is_match(tag))? {
|
|
||||||
self.remove_single_tag_undoable(tag)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
// replace tags
|
|
||||||
for mut note in matched_notes {
|
|
||||||
let original = note.clone();
|
|
||||||
note.tags = re.replace(¬e.tags, new_prefix);
|
|
||||||
note.set_modified(usn);
|
|
||||||
self.update_note_tags_undoable(¬e, original)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
// update tag list
|
|
||||||
for tag in re.into_seen_tags() {
|
|
||||||
self.register_tag_string(tag, usn)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(match_count)
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Take tags as a whitespace-separated string and remove them from all notes and the tag list.
|
|
||||||
pub fn remove_tags(&mut self, tags: &str) -> Result<OpOutput<usize>> {
|
|
||||||
self.transact(Op::RemoveTag, |col| col.remove_tags_inner(tags))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn remove_tags_inner(&mut self, tags: &str) -> Result<usize> {
|
|
||||||
let usn = self.usn()?;
|
|
||||||
|
|
||||||
// gather tags that need removing
|
|
||||||
let mut re = PrefixReplacer::new(tags)?;
|
|
||||||
let matched_notes = self
|
|
||||||
.storage
|
|
||||||
.get_note_tags_by_predicate(|tags| re.is_match(tags))?;
|
|
||||||
let match_count = matched_notes.len();
|
|
||||||
|
|
||||||
// remove from the tag list
|
|
||||||
for tag in self.storage.get_tags_by_predicate(|tag| re.is_match(tag))? {
|
|
||||||
self.remove_single_tag_undoable(tag)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
// replace tags
|
|
||||||
for mut note in matched_notes {
|
|
||||||
let original = note.clone();
|
|
||||||
note.tags = re.remove(¬e.tags);
|
|
||||||
note.set_modified(usn);
|
|
||||||
self.update_note_tags_undoable(¬e, original)?;
|
|
||||||
}
|
|
||||||
|
|
||||||
Ok(match_count)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// fixme: merge with prefixmatcher
|
|
||||||
|
|
||||||
/// A regex that will match a string tag that has been split from a list.
|
|
||||||
fn regex_matching_tag_and_children_in_single_tag(tag: &str) -> Result<Regex> {
|
|
||||||
Regex::new(&format!(
|
|
||||||
r#"(?ix)
|
|
||||||
^
|
|
||||||
{}
|
|
||||||
# optional children
|
|
||||||
(::.+)?
|
|
||||||
$
|
|
||||||
"#,
|
|
||||||
regex::escape(tag)
|
|
||||||
))
|
|
||||||
.map_err(Into::into)
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod test {
|
|
||||||
use super::*;
|
|
||||||
use crate::{collection::open_test_collection, decks::DeckID};
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn tags() -> Result<()> {
|
|
||||||
let mut col = open_test_collection();
|
|
||||||
let nt = col.get_notetype_by_name("Basic")?.unwrap();
|
|
||||||
let mut note = nt.new_note();
|
|
||||||
col.add_note(&mut note, DeckID(1))?;
|
|
||||||
|
|
||||||
let tags: String = col.storage.db_scalar("select tags from notes")?;
|
|
||||||
assert_eq!(tags, "");
|
|
||||||
|
|
||||||
// first instance wins in case of duplicates
|
|
||||||
note.tags = vec!["foo".into(), "FOO".into()];
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
assert_eq!(¬e.tags, &["foo"]);
|
|
||||||
let tags: String = col.storage.db_scalar("select tags from notes")?;
|
|
||||||
assert_eq!(tags, " foo ");
|
|
||||||
|
|
||||||
// existing case is used if in DB
|
|
||||||
note.tags = vec!["FOO".into()];
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
assert_eq!(¬e.tags, &["foo"]);
|
|
||||||
assert_eq!(tags, " foo ");
|
|
||||||
|
|
||||||
// tags are normalized to nfc
|
|
||||||
note.tags = vec!["\u{fa47}".into()];
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
assert_eq!(¬e.tags, &["\u{6f22}"]);
|
|
||||||
|
|
||||||
// if code incorrectly adds a space to a tag, it gets split
|
|
||||||
note.tags = vec!["one two".into()];
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
assert_eq!(¬e.tags, &["one", "two"]);
|
|
||||||
|
|
||||||
// blanks should be handled
|
|
||||||
note.tags = vec![
|
|
||||||
"".into(),
|
|
||||||
"foo".into(),
|
|
||||||
" ".into(),
|
|
||||||
"::".into(),
|
|
||||||
"foo::".into(),
|
|
||||||
];
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
assert_eq!(¬e.tags, &["blank::blank", "foo", "foo::blank"]);
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn bulk() -> Result<()> {
|
|
||||||
let mut col = open_test_collection();
|
|
||||||
let nt = col.get_notetype_by_name("Basic")?.unwrap();
|
|
||||||
let mut note = nt.new_note();
|
|
||||||
note.tags.push("test".into());
|
|
||||||
col.add_note(&mut note, DeckID(1))?;
|
|
||||||
|
|
||||||
col.replace_tags_for_notes(&[note.id], "foo test", "bar", false)?;
|
|
||||||
let note = col.storage.get_note(note.id)?.unwrap();
|
|
||||||
assert_eq!(note.tags[0], "bar");
|
|
||||||
|
|
||||||
col.replace_tags_for_notes(&[note.id], "b.r", "baz", false)?;
|
|
||||||
let note = col.storage.get_note(note.id)?.unwrap();
|
|
||||||
assert_eq!(note.tags[0], "bar");
|
|
||||||
|
|
||||||
col.replace_tags_for_notes(&[note.id], "b*r", "baz", false)?;
|
|
||||||
let note = col.storage.get_note(note.id)?.unwrap();
|
|
||||||
assert_eq!(note.tags[0], "baz");
|
|
||||||
|
|
||||||
col.replace_tags_for_notes(&[note.id], "b.r", "baz", true)?;
|
|
||||||
let note = col.storage.get_note(note.id)?.unwrap();
|
|
||||||
assert_eq!(note.tags[0], "baz");
|
|
||||||
|
|
||||||
let out = col.add_tags_to_notes(&[note.id], "cee aye")?;
|
|
||||||
assert_eq!(out.output, 1);
|
|
||||||
let note = col.storage.get_note(note.id)?.unwrap();
|
|
||||||
assert_eq!(¬e.tags, &["aye", "baz", "cee"]);
|
|
||||||
|
|
||||||
// if all tags already on note, it doesn't get updated
|
|
||||||
let out = col.add_tags_to_notes(&[note.id], "cee aye")?;
|
|
||||||
assert_eq!(out.output, 0);
|
|
||||||
|
|
||||||
// empty replacement deletes tag
|
|
||||||
col.replace_tags_for_notes(&[note.id], "b.* .*ye", "", true)?;
|
|
||||||
let note = col.storage.get_note(note.id)?.unwrap();
|
|
||||||
assert_eq!(¬e.tags, &["cee"]);
|
|
||||||
|
|
||||||
let mut note = col.storage.get_note(note.id)?.unwrap();
|
|
||||||
note.tags = vec![
|
|
||||||
"foo::bar".into(),
|
|
||||||
"foo::bar::foo".into(),
|
|
||||||
"bar::foo".into(),
|
|
||||||
"bar::foo::bar".into(),
|
|
||||||
];
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
col.replace_tags_for_notes(&[note.id], "bar::foo", "foo::bar", false)?;
|
|
||||||
let note = col.storage.get_note(note.id)?.unwrap();
|
|
||||||
assert_eq!(¬e.tags, &["foo::bar", "foo::bar::bar", "foo::bar::foo",]);
|
|
||||||
|
|
||||||
// ensure replacements fully match
|
|
||||||
let mut note = col.storage.get_note(note.id)?.unwrap();
|
|
||||||
note.tags = vec!["foobar".into(), "barfoo".into(), "foo".into()];
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
col.replace_tags_for_notes(&[note.id], "foo", "", false)?;
|
|
||||||
let note = col.storage.get_note(note.id)?.unwrap();
|
|
||||||
assert_eq!(¬e.tags, &["barfoo", "foobar"]);
|
|
||||||
|
|
||||||
// tag children are also cleared when clearing their parent
|
|
||||||
col.storage.clear_all_tags()?;
|
|
||||||
for name in vec!["a", "a::b", "A::b::c"] {
|
|
||||||
col.register_tag(&mut Tag::new(name.to_string(), Usn(0)))?;
|
|
||||||
}
|
|
||||||
col.storage.clear_tag_and_children("a")?;
|
|
||||||
assert_eq!(col.storage.all_tags()?, vec![]);
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn node(name: &str, level: u32, children: Vec<TagTreeNode>) -> TagTreeNode {
|
|
||||||
TagTreeNode {
|
|
||||||
name: name.into(),
|
|
||||||
level,
|
|
||||||
children,
|
|
||||||
|
|
||||||
..Default::default()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn leaf(name: &str, level: u32) -> TagTreeNode {
|
|
||||||
node(name, level, vec![])
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn tree() -> Result<()> {
|
|
||||||
let mut col = open_test_collection();
|
|
||||||
let nt = col.get_notetype_by_name("Basic")?.unwrap();
|
|
||||||
let mut note = nt.new_note();
|
|
||||||
note.tags.push("foo::bar::a".into());
|
|
||||||
note.tags.push("foo::bar::b".into());
|
|
||||||
col.add_note(&mut note, DeckID(1))?;
|
|
||||||
|
|
||||||
// missing parents are added
|
|
||||||
assert_eq!(
|
|
||||||
col.tag_tree()?,
|
|
||||||
node(
|
|
||||||
"",
|
|
||||||
0,
|
|
||||||
vec![node(
|
|
||||||
"foo",
|
|
||||||
1,
|
|
||||||
vec![node("bar", 2, vec![leaf("a", 3), leaf("b", 3)])]
|
|
||||||
)]
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
// differing case should result in only one parent case being added -
|
|
||||||
// the first one
|
|
||||||
col.storage.clear_all_tags()?;
|
|
||||||
note.tags[0] = "foo::BAR::a".into();
|
|
||||||
note.tags[1] = "FOO::bar::b".into();
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
assert_eq!(
|
|
||||||
col.tag_tree()?,
|
|
||||||
node(
|
|
||||||
"",
|
|
||||||
0,
|
|
||||||
vec![node(
|
|
||||||
"foo",
|
|
||||||
1,
|
|
||||||
vec![node("BAR", 2, vec![leaf("a", 3), leaf("b", 3)])]
|
|
||||||
)]
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
// things should work even if the immediate parent is not missing
|
|
||||||
col.storage.clear_all_tags()?;
|
|
||||||
note.tags[0] = "foo::bar::baz".into();
|
|
||||||
note.tags[1] = "foo::bar::baz::quux".into();
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
assert_eq!(
|
|
||||||
col.tag_tree()?,
|
|
||||||
node(
|
|
||||||
"",
|
|
||||||
0,
|
|
||||||
vec![node(
|
|
||||||
"foo",
|
|
||||||
1,
|
|
||||||
vec![node("bar", 2, vec![node("baz", 3, vec![leaf("quux", 4)])])]
|
|
||||||
)]
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
// numbers have a smaller ascii number than ':', so a naive sort on
|
|
||||||
// '::' would result in one::two being nested under one1.
|
|
||||||
col.storage.clear_all_tags()?;
|
|
||||||
note.tags[0] = "one".into();
|
|
||||||
note.tags[1] = "one1".into();
|
|
||||||
note.tags.push("one::two".into());
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
assert_eq!(
|
|
||||||
col.tag_tree()?,
|
|
||||||
node(
|
|
||||||
"",
|
|
||||||
0,
|
|
||||||
vec![node("one", 1, vec![leaf("two", 2)]), leaf("one1", 1)]
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
// children should match the case of their parents
|
|
||||||
col.storage.clear_all_tags()?;
|
|
||||||
note.tags[0] = "FOO".into();
|
|
||||||
note.tags[1] = "foo::BAR".into();
|
|
||||||
note.tags[2] = "foo::bar::baz".into();
|
|
||||||
col.update_note(&mut note)?;
|
|
||||||
assert_eq!(note.tags, vec!["FOO", "FOO::BAR", "FOO::BAR::baz"]);
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn clearing() -> Result<()> {
|
|
||||||
let mut col = open_test_collection();
|
|
||||||
let nt = col.get_notetype_by_name("Basic")?.unwrap();
|
|
||||||
let mut note = nt.new_note();
|
|
||||||
note.tags.push("one".into());
|
|
||||||
note.tags.push("two".into());
|
|
||||||
col.add_note(&mut note, DeckID(1))?;
|
|
||||||
|
|
||||||
col.set_tag_expanded("one", true)?;
|
|
||||||
col.clear_unused_tags()?;
|
|
||||||
assert_eq!(col.storage.get_tag("one")?.unwrap().expanded, true);
|
|
||||||
assert_eq!(col.storage.get_tag("two")?.unwrap().expanded, false);
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
|
|
||||||
fn alltags(col: &Collection) -> Vec<String> {
|
|
||||||
col.storage
|
|
||||||
.all_tags()
|
|
||||||
.unwrap()
|
|
||||||
.into_iter()
|
|
||||||
.map(|t| t.name)
|
|
||||||
.collect()
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn dragdrop() -> Result<()> {
|
|
||||||
let mut col = open_test_collection();
|
|
||||||
let nt = col.get_notetype_by_name("Basic")?.unwrap();
|
|
||||||
for tag in &[
|
|
||||||
"another",
|
|
||||||
"parent1::child1::grandchild1",
|
|
||||||
"parent1::child1",
|
|
||||||
"parent1",
|
|
||||||
"parent2",
|
|
||||||
"yet::another",
|
|
||||||
] {
|
|
||||||
let mut note = nt.new_note();
|
|
||||||
note.tags.push(tag.to_string());
|
|
||||||
col.add_note(&mut note, DeckID(1))?;
|
|
||||||
}
|
|
||||||
|
|
||||||
// two decks with the same base name; they both get mapped
|
|
||||||
// to parent1::another
|
|
||||||
col.drag_drop_tags(
|
|
||||||
&["another".to_string(), "yet::another".to_string()],
|
|
||||||
Some("parent1".to_string()),
|
|
||||||
)?;
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
alltags(&col),
|
|
||||||
&[
|
|
||||||
"parent1",
|
|
||||||
"parent1::another",
|
|
||||||
"parent1::child1",
|
|
||||||
"parent1::child1::grandchild1",
|
|
||||||
"parent2",
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
// child and children moved to parent2
|
|
||||||
col.drag_drop_tags(
|
|
||||||
&["parent1::child1".to_string()],
|
|
||||||
Some("parent2".to_string()),
|
|
||||||
)?;
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
alltags(&col),
|
|
||||||
&[
|
|
||||||
"parent1",
|
|
||||||
"parent1::another",
|
|
||||||
"parent2",
|
|
||||||
"parent2::child1",
|
|
||||||
"parent2::child1::grandchild1",
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
// empty target reparents to root
|
|
||||||
col.drag_drop_tags(&["parent1::another".to_string()], None)?;
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
alltags(&col),
|
|
||||||
&[
|
|
||||||
"another",
|
|
||||||
"parent1",
|
|
||||||
"parent2",
|
|
||||||
"parent2::child1",
|
|
||||||
"parent2::child1::grandchild1",
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
Ok(())
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
194
rslib/src/tags/register.rs
Normal file
194
rslib/src/tags/register.rs
Normal file
@ -0,0 +1,194 @@
|
|||||||
|
// Copyright: Ankitects Pty Ltd and contributors
|
||||||
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
||||||
|
|
||||||
|
use super::{immediate_parent_name_str, is_tag_separator, split_tags, Tag};
|
||||||
|
use crate::{prelude::*, text::normalize_to_nfc, types::Usn};
|
||||||
|
|
||||||
|
use std::{borrow::Cow, collections::HashSet};
|
||||||
|
use unicase::UniCase;
|
||||||
|
|
||||||
|
impl Collection {
|
||||||
|
/// Given a list of tags, fix case, ordering and duplicates.
|
||||||
|
/// Returns true if any new tags were added.
|
||||||
|
pub(crate) fn canonify_tags(
|
||||||
|
&mut self,
|
||||||
|
tags: Vec<String>,
|
||||||
|
usn: Usn,
|
||||||
|
) -> Result<(Vec<String>, bool)> {
|
||||||
|
let mut seen = HashSet::new();
|
||||||
|
let mut added = false;
|
||||||
|
|
||||||
|
let tags: Vec<_> = tags.iter().flat_map(|t| split_tags(t)).collect();
|
||||||
|
for tag in tags {
|
||||||
|
let mut tag = Tag::new(tag.to_string(), usn);
|
||||||
|
added |= self.register_tag(&mut tag)?;
|
||||||
|
seen.insert(UniCase::new(tag.name));
|
||||||
|
}
|
||||||
|
|
||||||
|
// exit early if no non-empty tags
|
||||||
|
if seen.is_empty() {
|
||||||
|
return Ok((vec![], added));
|
||||||
|
}
|
||||||
|
|
||||||
|
// return the sorted, canonified tags
|
||||||
|
let mut tags = seen.into_iter().collect::<Vec<_>>();
|
||||||
|
tags.sort_unstable();
|
||||||
|
let tags: Vec<_> = tags.into_iter().map(|s| s.into_inner()).collect();
|
||||||
|
|
||||||
|
Ok((tags, added))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Adjust tag casing to match any existing parents, and register it if it's not already
|
||||||
|
/// in the tags list. True if the tag was added and not already in tag list.
|
||||||
|
/// In the case the tag is already registered, tag will be mutated to match the existing
|
||||||
|
/// name.
|
||||||
|
pub(crate) fn register_tag(&mut self, tag: &mut Tag) -> Result<bool> {
|
||||||
|
let is_new = self.prepare_tag_for_registering(tag)?;
|
||||||
|
if is_new {
|
||||||
|
self.register_tag_undoable(&tag)?;
|
||||||
|
}
|
||||||
|
Ok(is_new)
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create a tag object, normalize text, and match parents/existing case if available.
|
||||||
|
/// True if tag is new.
|
||||||
|
pub(super) fn prepare_tag_for_registering(&self, tag: &mut Tag) -> Result<bool> {
|
||||||
|
let normalized_name = normalize_tag_name(&tag.name);
|
||||||
|
if normalized_name.is_empty() {
|
||||||
|
// this should not be possible
|
||||||
|
return Err(AnkiError::invalid_input("blank tag"));
|
||||||
|
}
|
||||||
|
if let Some(existing_tag) = self.storage.get_tag(&normalized_name)? {
|
||||||
|
tag.name = existing_tag.name;
|
||||||
|
Ok(false)
|
||||||
|
} else {
|
||||||
|
if let Some(new_name) = self.adjusted_case_for_parents(&normalized_name)? {
|
||||||
|
tag.name = new_name;
|
||||||
|
} else if let Cow::Owned(new_name) = normalized_name {
|
||||||
|
tag.name = new_name;
|
||||||
|
}
|
||||||
|
Ok(true)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(super) fn register_tag_string(&mut self, tag: String, usn: Usn) -> Result<bool> {
|
||||||
|
let mut tag = Tag::new(tag, usn);
|
||||||
|
self.register_tag(&mut tag)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Collection {
|
||||||
|
/// If parent tag(s) exist and differ in case, return a rewritten tag.
|
||||||
|
fn adjusted_case_for_parents(&self, tag: &str) -> Result<Option<String>> {
|
||||||
|
if let Some(parent_tag) = self.first_existing_parent_tag(&tag)? {
|
||||||
|
let child_split: Vec<_> = tag.split("::").collect();
|
||||||
|
let parent_count = parent_tag.matches("::").count() + 1;
|
||||||
|
Ok(Some(format!(
|
||||||
|
"{}::{}",
|
||||||
|
parent_tag,
|
||||||
|
&child_split[parent_count..].join("::")
|
||||||
|
)))
|
||||||
|
} else {
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn first_existing_parent_tag(&self, mut tag: &str) -> Result<Option<String>> {
|
||||||
|
while let Some(parent_name) = immediate_parent_name_str(tag) {
|
||||||
|
if let Some(parent_tag) = self.storage.preferred_tag_case(parent_name)? {
|
||||||
|
return Ok(Some(parent_tag));
|
||||||
|
}
|
||||||
|
tag = parent_name;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(None)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn invalid_char_for_tag(c: char) -> bool {
|
||||||
|
c.is_ascii_control() || is_tag_separator(c) || c == '"'
|
||||||
|
}
|
||||||
|
|
||||||
|
fn normalized_tag_name_component(comp: &str) -> Cow<str> {
|
||||||
|
let mut out = normalize_to_nfc(comp);
|
||||||
|
if out.contains(invalid_char_for_tag) {
|
||||||
|
out = out.replace(invalid_char_for_tag, "").into();
|
||||||
|
}
|
||||||
|
let trimmed = out.trim();
|
||||||
|
if trimmed.is_empty() {
|
||||||
|
"blank".to_string().into()
|
||||||
|
} else if trimmed.len() != out.len() {
|
||||||
|
trimmed.to_string().into()
|
||||||
|
} else {
|
||||||
|
out
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn normalize_tag_name(name: &str) -> Cow<str> {
|
||||||
|
if name
|
||||||
|
.split("::")
|
||||||
|
.any(|comp| matches!(normalized_tag_name_component(comp), Cow::Owned(_)))
|
||||||
|
{
|
||||||
|
let comps: Vec<_> = name
|
||||||
|
.split("::")
|
||||||
|
.map(normalized_tag_name_component)
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
comps.join("::").into()
|
||||||
|
} else {
|
||||||
|
// no changes required
|
||||||
|
name.into()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
use crate::{collection::open_test_collection, decks::DeckID};
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn tags() -> Result<()> {
|
||||||
|
let mut col = open_test_collection();
|
||||||
|
let nt = col.get_notetype_by_name("Basic")?.unwrap();
|
||||||
|
let mut note = nt.new_note();
|
||||||
|
col.add_note(&mut note, DeckID(1))?;
|
||||||
|
|
||||||
|
let tags: String = col.storage.db_scalar("select tags from notes")?;
|
||||||
|
assert_eq!(tags, "");
|
||||||
|
|
||||||
|
// first instance wins in case of duplicates
|
||||||
|
note.tags = vec!["foo".into(), "FOO".into()];
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
assert_eq!(¬e.tags, &["foo"]);
|
||||||
|
let tags: String = col.storage.db_scalar("select tags from notes")?;
|
||||||
|
assert_eq!(tags, " foo ");
|
||||||
|
|
||||||
|
// existing case is used if in DB
|
||||||
|
note.tags = vec!["FOO".into()];
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
assert_eq!(¬e.tags, &["foo"]);
|
||||||
|
assert_eq!(tags, " foo ");
|
||||||
|
|
||||||
|
// tags are normalized to nfc
|
||||||
|
note.tags = vec!["\u{fa47}".into()];
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
assert_eq!(¬e.tags, &["\u{6f22}"]);
|
||||||
|
|
||||||
|
// if code incorrectly adds a space to a tag, it gets split
|
||||||
|
note.tags = vec!["one two".into()];
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
assert_eq!(¬e.tags, &["one", "two"]);
|
||||||
|
|
||||||
|
// blanks should be handled
|
||||||
|
note.tags = vec![
|
||||||
|
"".into(),
|
||||||
|
"foo".into(),
|
||||||
|
" ".into(),
|
||||||
|
"::".into(),
|
||||||
|
"foo::".into(),
|
||||||
|
];
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
assert_eq!(¬e.tags, &["blank::blank", "foo", "foo::blank"]);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
84
rslib/src/tags/remove.rs
Normal file
84
rslib/src/tags/remove.rs
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
// Copyright: Ankitects Pty Ltd and contributors
|
||||||
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
||||||
|
|
||||||
|
use super::prefix_replacer::PrefixReplacer;
|
||||||
|
use crate::prelude::*;
|
||||||
|
|
||||||
|
impl Collection {
|
||||||
|
/// Take tags as a whitespace-separated string and remove them from all notes and the tag list.
|
||||||
|
pub fn remove_tags(&mut self, tags: &str) -> Result<OpOutput<usize>> {
|
||||||
|
self.transact(Op::RemoveTag, |col| col.remove_tags_inner(tags))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Remove tags not referenced by notes, returning removed count.
|
||||||
|
pub fn clear_unused_tags(&mut self) -> Result<OpOutput<usize>> {
|
||||||
|
self.transact(Op::ClearUnusedTags, |col| col.clear_unused_tags_inner())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Collection {
|
||||||
|
fn remove_tags_inner(&mut self, tags: &str) -> Result<usize> {
|
||||||
|
let usn = self.usn()?;
|
||||||
|
|
||||||
|
// gather tags that need removing
|
||||||
|
let mut re = PrefixReplacer::new(tags)?;
|
||||||
|
let matched_notes = self
|
||||||
|
.storage
|
||||||
|
.get_note_tags_by_predicate(|tags| re.is_match(tags))?;
|
||||||
|
let match_count = matched_notes.len();
|
||||||
|
|
||||||
|
// remove from the tag list
|
||||||
|
for tag in self.storage.get_tags_by_predicate(|tag| re.is_match(tag))? {
|
||||||
|
self.remove_single_tag_undoable(tag)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
// replace tags
|
||||||
|
for mut note in matched_notes {
|
||||||
|
let original = note.clone();
|
||||||
|
note.tags = re.remove(¬e.tags);
|
||||||
|
note.set_modified(usn);
|
||||||
|
self.update_note_tags_undoable(¬e, original)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(match_count)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn clear_unused_tags_inner(&mut self) -> Result<usize> {
|
||||||
|
let mut count = 0;
|
||||||
|
let in_notes = self.storage.all_tags_in_notes()?;
|
||||||
|
let need_remove = self
|
||||||
|
.storage
|
||||||
|
.all_tags()?
|
||||||
|
.into_iter()
|
||||||
|
.filter(|tag| !in_notes.contains(&tag.name));
|
||||||
|
for tag in need_remove {
|
||||||
|
self.remove_single_tag_undoable(tag)?;
|
||||||
|
count += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(count)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
use crate::collection::open_test_collection;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn clearing() -> Result<()> {
|
||||||
|
let mut col = open_test_collection();
|
||||||
|
let nt = col.get_notetype_by_name("Basic")?.unwrap();
|
||||||
|
let mut note = nt.new_note();
|
||||||
|
note.tags.push("one".into());
|
||||||
|
note.tags.push("two".into());
|
||||||
|
col.add_note(&mut note, DeckID(1))?;
|
||||||
|
|
||||||
|
col.set_tag_expanded("one", true)?;
|
||||||
|
col.clear_unused_tags()?;
|
||||||
|
assert_eq!(col.storage.get_tag("one")?.unwrap().expanded, true);
|
||||||
|
assert_eq!(col.storage.get_tag("two")?.unwrap().expanded, false);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
68
rslib/src/tags/rename.rs
Normal file
68
rslib/src/tags/rename.rs
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
// Copyright: Ankitects Pty Ltd and contributors
|
||||||
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
||||||
|
|
||||||
|
use super::{prefix_replacer::PrefixReplacer, Tag};
|
||||||
|
use crate::prelude::*;
|
||||||
|
|
||||||
|
impl Collection {
|
||||||
|
/// Rename a given tag and its children on all notes that reference it, returning changed
|
||||||
|
/// note count.
|
||||||
|
pub fn rename_tag(&mut self, old_prefix: &str, new_prefix: &str) -> Result<OpOutput<usize>> {
|
||||||
|
self.transact(Op::RenameTag, |col| {
|
||||||
|
col.rename_tag_inner(old_prefix, new_prefix)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Collection {
|
||||||
|
fn rename_tag_inner(&mut self, old_prefix: &str, new_prefix: &str) -> Result<usize> {
|
||||||
|
if new_prefix.contains(' ') {
|
||||||
|
return Err(AnkiError::invalid_input(
|
||||||
|
"replacement name can not contain a space",
|
||||||
|
));
|
||||||
|
}
|
||||||
|
if new_prefix.trim().is_empty() {
|
||||||
|
return Err(AnkiError::invalid_input(
|
||||||
|
"replacement name must not be empty",
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
let usn = self.usn()?;
|
||||||
|
|
||||||
|
// match existing case if available, and ensure normalized.
|
||||||
|
let mut tag = Tag::new(new_prefix.to_string(), usn);
|
||||||
|
self.prepare_tag_for_registering(&mut tag)?;
|
||||||
|
let new_prefix = &tag.name;
|
||||||
|
|
||||||
|
// gather tags that need replacing
|
||||||
|
let mut re = PrefixReplacer::new(old_prefix)?;
|
||||||
|
let matched_notes = self
|
||||||
|
.storage
|
||||||
|
.get_note_tags_by_predicate(|tags| re.is_match(tags))?;
|
||||||
|
let match_count = matched_notes.len();
|
||||||
|
if match_count == 0 {
|
||||||
|
// no matches; exit early so we don't clobber the empty tag entries
|
||||||
|
return Ok(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
// remove old prefix from the tag list
|
||||||
|
for tag in self.storage.get_tags_by_predicate(|tag| re.is_match(tag))? {
|
||||||
|
self.remove_single_tag_undoable(tag)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
// replace tags
|
||||||
|
for mut note in matched_notes {
|
||||||
|
let original = note.clone();
|
||||||
|
note.tags = re.replace(¬e.tags, new_prefix);
|
||||||
|
note.set_modified(usn);
|
||||||
|
self.update_note_tags_undoable(¬e, original)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
// update tag list
|
||||||
|
for tag in re.into_seen_tags() {
|
||||||
|
self.register_tag_string(tag, usn)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(match_count)
|
||||||
|
}
|
||||||
|
}
|
172
rslib/src/tags/selectednotes.rs
Normal file
172
rslib/src/tags/selectednotes.rs
Normal file
@ -0,0 +1,172 @@
|
|||||||
|
// Copyright: Ankitects Pty Ltd and contributors
|
||||||
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
||||||
|
|
||||||
|
//! Add/update/remove tags on selected notes
|
||||||
|
|
||||||
|
use crate::{notes::TransformNoteOutput, prelude::*, text::to_re};
|
||||||
|
|
||||||
|
use regex::{NoExpand, Regex, Replacer};
|
||||||
|
|
||||||
|
use super::split_tags;
|
||||||
|
|
||||||
|
impl Collection {
|
||||||
|
fn replace_tags_for_notes_inner<R: Replacer>(
|
||||||
|
&mut self,
|
||||||
|
nids: &[NoteID],
|
||||||
|
tags: &[Regex],
|
||||||
|
mut repl: R,
|
||||||
|
) -> Result<OpOutput<usize>> {
|
||||||
|
self.transact(Op::UpdateTag, |col| {
|
||||||
|
col.transform_notes(nids, |note, _nt| {
|
||||||
|
let mut changed = false;
|
||||||
|
for re in tags {
|
||||||
|
if note.replace_tags(re, repl.by_ref()) {
|
||||||
|
changed = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(TransformNoteOutput {
|
||||||
|
changed,
|
||||||
|
generate_cards: false,
|
||||||
|
mark_modified: true,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Apply the provided list of regular expressions to note tags,
|
||||||
|
/// saving any modified notes.
|
||||||
|
pub fn replace_tags_for_notes(
|
||||||
|
&mut self,
|
||||||
|
nids: &[NoteID],
|
||||||
|
tags: &str,
|
||||||
|
repl: &str,
|
||||||
|
regex: bool,
|
||||||
|
) -> Result<OpOutput<usize>> {
|
||||||
|
// generate regexps
|
||||||
|
let tags = split_tags(tags)
|
||||||
|
.map(|tag| {
|
||||||
|
let tag = if regex { tag.into() } else { to_re(tag) };
|
||||||
|
Regex::new(&format!("(?i)^{}(::.*)?$", tag))
|
||||||
|
.map_err(|_| AnkiError::invalid_input("invalid regex"))
|
||||||
|
})
|
||||||
|
.collect::<Result<Vec<Regex>>>()?;
|
||||||
|
if !regex {
|
||||||
|
self.replace_tags_for_notes_inner(nids, &tags, NoExpand(repl))
|
||||||
|
} else {
|
||||||
|
self.replace_tags_for_notes_inner(nids, &tags, repl)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn add_tags_to_notes(&mut self, nids: &[NoteID], tags: &str) -> Result<OpOutput<usize>> {
|
||||||
|
let tags: Vec<_> = split_tags(tags).collect();
|
||||||
|
let matcher = regex::RegexSet::new(
|
||||||
|
tags.iter()
|
||||||
|
.map(|s| regex::escape(s))
|
||||||
|
.map(|s| format!("(?i)^{}$", s)),
|
||||||
|
)
|
||||||
|
.map_err(|_| AnkiError::invalid_input("invalid regex"))?;
|
||||||
|
|
||||||
|
self.transact(Op::UpdateTag, |col| {
|
||||||
|
col.transform_notes(nids, |note, _nt| {
|
||||||
|
let mut need_to_add = true;
|
||||||
|
let mut match_count = 0;
|
||||||
|
for tag in ¬e.tags {
|
||||||
|
if matcher.is_match(tag) {
|
||||||
|
match_count += 1;
|
||||||
|
}
|
||||||
|
if match_count == tags.len() {
|
||||||
|
need_to_add = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if need_to_add {
|
||||||
|
note.tags.extend(tags.iter().map(|&s| s.to_string()))
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(TransformNoteOutput {
|
||||||
|
changed: need_to_add,
|
||||||
|
generate_cards: false,
|
||||||
|
mark_modified: true,
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
use crate::tags::Tag;
|
||||||
|
use crate::{collection::open_test_collection, decks::DeckID};
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn bulk() -> Result<()> {
|
||||||
|
let mut col = open_test_collection();
|
||||||
|
let nt = col.get_notetype_by_name("Basic")?.unwrap();
|
||||||
|
let mut note = nt.new_note();
|
||||||
|
note.tags.push("test".into());
|
||||||
|
col.add_note(&mut note, DeckID(1))?;
|
||||||
|
|
||||||
|
col.replace_tags_for_notes(&[note.id], "foo test", "bar", false)?;
|
||||||
|
let note = col.storage.get_note(note.id)?.unwrap();
|
||||||
|
assert_eq!(note.tags[0], "bar");
|
||||||
|
|
||||||
|
col.replace_tags_for_notes(&[note.id], "b.r", "baz", false)?;
|
||||||
|
let note = col.storage.get_note(note.id)?.unwrap();
|
||||||
|
assert_eq!(note.tags[0], "bar");
|
||||||
|
|
||||||
|
col.replace_tags_for_notes(&[note.id], "b*r", "baz", false)?;
|
||||||
|
let note = col.storage.get_note(note.id)?.unwrap();
|
||||||
|
assert_eq!(note.tags[0], "baz");
|
||||||
|
|
||||||
|
col.replace_tags_for_notes(&[note.id], "b.r", "baz", true)?;
|
||||||
|
let note = col.storage.get_note(note.id)?.unwrap();
|
||||||
|
assert_eq!(note.tags[0], "baz");
|
||||||
|
|
||||||
|
let out = col.add_tags_to_notes(&[note.id], "cee aye")?;
|
||||||
|
assert_eq!(out.output, 1);
|
||||||
|
let note = col.storage.get_note(note.id)?.unwrap();
|
||||||
|
assert_eq!(¬e.tags, &["aye", "baz", "cee"]);
|
||||||
|
|
||||||
|
// if all tags already on note, it doesn't get updated
|
||||||
|
let out = col.add_tags_to_notes(&[note.id], "cee aye")?;
|
||||||
|
assert_eq!(out.output, 0);
|
||||||
|
|
||||||
|
// empty replacement deletes tag
|
||||||
|
col.replace_tags_for_notes(&[note.id], "b.* .*ye", "", true)?;
|
||||||
|
let note = col.storage.get_note(note.id)?.unwrap();
|
||||||
|
assert_eq!(¬e.tags, &["cee"]);
|
||||||
|
|
||||||
|
let mut note = col.storage.get_note(note.id)?.unwrap();
|
||||||
|
note.tags = vec![
|
||||||
|
"foo::bar".into(),
|
||||||
|
"foo::bar::foo".into(),
|
||||||
|
"bar::foo".into(),
|
||||||
|
"bar::foo::bar".into(),
|
||||||
|
];
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
col.replace_tags_for_notes(&[note.id], "bar::foo", "foo::bar", false)?;
|
||||||
|
let note = col.storage.get_note(note.id)?.unwrap();
|
||||||
|
assert_eq!(¬e.tags, &["foo::bar", "foo::bar::bar", "foo::bar::foo",]);
|
||||||
|
|
||||||
|
// ensure replacements fully match
|
||||||
|
let mut note = col.storage.get_note(note.id)?.unwrap();
|
||||||
|
note.tags = vec!["foobar".into(), "barfoo".into(), "foo".into()];
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
col.replace_tags_for_notes(&[note.id], "foo", "", false)?;
|
||||||
|
let note = col.storage.get_note(note.id)?.unwrap();
|
||||||
|
assert_eq!(¬e.tags, &["barfoo", "foobar"]);
|
||||||
|
|
||||||
|
// tag children are also cleared when clearing their parent
|
||||||
|
col.storage.clear_all_tags()?;
|
||||||
|
for name in vec!["a", "a::b", "A::b::c"] {
|
||||||
|
col.register_tag(&mut Tag::new(name.to_string(), Usn(0)))?;
|
||||||
|
}
|
||||||
|
col.storage.clear_tag_and_children("a")?;
|
||||||
|
assert_eq!(col.storage.all_tags()?, vec![]);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
203
rslib/src/tags/tree.rs
Normal file
203
rslib/src/tags/tree.rs
Normal file
@ -0,0 +1,203 @@
|
|||||||
|
// Copyright: Ankitects Pty Ltd and contributors
|
||||||
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
||||||
|
|
||||||
|
use std::{collections::HashSet, iter::Peekable};
|
||||||
|
|
||||||
|
use unicase::UniCase;
|
||||||
|
|
||||||
|
use super::{immediate_parent_name_unicase, Tag};
|
||||||
|
use crate::{backend_proto::TagTreeNode, prelude::*};
|
||||||
|
|
||||||
|
impl Collection {
|
||||||
|
pub fn tag_tree(&mut self) -> Result<TagTreeNode> {
|
||||||
|
let tags = self.storage.all_tags()?;
|
||||||
|
let tree = tags_to_tree(tags);
|
||||||
|
|
||||||
|
Ok(tree)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Append any missing parents. Caller must sort afterwards.
|
||||||
|
fn add_missing_parents(tags: &mut Vec<Tag>) {
|
||||||
|
let mut all_names: HashSet<UniCase<&str>> = HashSet::new();
|
||||||
|
let mut missing = vec![];
|
||||||
|
for tag in &*tags {
|
||||||
|
add_tag_and_missing_parents(&mut all_names, &mut missing, UniCase::new(&tag.name))
|
||||||
|
}
|
||||||
|
let mut missing: Vec<_> = missing
|
||||||
|
.into_iter()
|
||||||
|
.map(|n| Tag::new(n.to_string(), Usn(0)))
|
||||||
|
.collect();
|
||||||
|
tags.append(&mut missing);
|
||||||
|
}
|
||||||
|
|
||||||
|
fn tags_to_tree(mut tags: Vec<Tag>) -> TagTreeNode {
|
||||||
|
for tag in &mut tags {
|
||||||
|
tag.name = tag.name.replace("::", "\x1f");
|
||||||
|
}
|
||||||
|
add_missing_parents(&mut tags);
|
||||||
|
tags.sort_unstable_by(|a, b| UniCase::new(&a.name).cmp(&UniCase::new(&b.name)));
|
||||||
|
let mut top = TagTreeNode::default();
|
||||||
|
let mut it = tags.into_iter().peekable();
|
||||||
|
add_child_nodes(&mut it, &mut top);
|
||||||
|
|
||||||
|
top
|
||||||
|
}
|
||||||
|
|
||||||
|
fn add_child_nodes(tags: &mut Peekable<impl Iterator<Item = Tag>>, parent: &mut TagTreeNode) {
|
||||||
|
while let Some(tag) = tags.peek() {
|
||||||
|
let split_name: Vec<_> = tag.name.split('\x1f').collect();
|
||||||
|
match split_name.len() as u32 {
|
||||||
|
l if l <= parent.level => {
|
||||||
|
// next item is at a higher level
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
l if l == parent.level + 1 => {
|
||||||
|
// next item is an immediate descendent of parent
|
||||||
|
parent.children.push(TagTreeNode {
|
||||||
|
name: (*split_name.last().unwrap()).into(),
|
||||||
|
children: vec![],
|
||||||
|
level: parent.level + 1,
|
||||||
|
expanded: tag.expanded,
|
||||||
|
});
|
||||||
|
tags.next();
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
// next item is at a lower level
|
||||||
|
if let Some(last_child) = parent.children.last_mut() {
|
||||||
|
add_child_nodes(tags, last_child)
|
||||||
|
} else {
|
||||||
|
// immediate parent is missing
|
||||||
|
tags.next();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// For the given tag, check if immediate parent exists. If so, add
|
||||||
|
/// tag and return.
|
||||||
|
/// If the immediate parent is missing, check and add any missing parents.
|
||||||
|
/// This should ensure that if an immediate parent is found, all ancestors
|
||||||
|
/// are guaranteed to already exist.
|
||||||
|
fn add_tag_and_missing_parents<'a, 'b>(
|
||||||
|
all: &'a mut HashSet<UniCase<&'b str>>,
|
||||||
|
missing: &'a mut Vec<UniCase<&'b str>>,
|
||||||
|
tag_name: UniCase<&'b str>,
|
||||||
|
) {
|
||||||
|
if let Some(parent) = immediate_parent_name_unicase(tag_name) {
|
||||||
|
if !all.contains(&parent) {
|
||||||
|
missing.push(parent);
|
||||||
|
add_tag_and_missing_parents(all, missing, parent);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// finally, add provided tag
|
||||||
|
all.insert(tag_name);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use super::*;
|
||||||
|
use crate::collection::open_test_collection;
|
||||||
|
|
||||||
|
fn node(name: &str, level: u32, children: Vec<TagTreeNode>) -> TagTreeNode {
|
||||||
|
TagTreeNode {
|
||||||
|
name: name.into(),
|
||||||
|
level,
|
||||||
|
children,
|
||||||
|
|
||||||
|
..Default::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn leaf(name: &str, level: u32) -> TagTreeNode {
|
||||||
|
node(name, level, vec![])
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn tree() -> Result<()> {
|
||||||
|
let mut col = open_test_collection();
|
||||||
|
let nt = col.get_notetype_by_name("Basic")?.unwrap();
|
||||||
|
let mut note = nt.new_note();
|
||||||
|
note.tags.push("foo::bar::a".into());
|
||||||
|
note.tags.push("foo::bar::b".into());
|
||||||
|
col.add_note(&mut note, DeckID(1))?;
|
||||||
|
|
||||||
|
// missing parents are added
|
||||||
|
assert_eq!(
|
||||||
|
col.tag_tree()?,
|
||||||
|
node(
|
||||||
|
"",
|
||||||
|
0,
|
||||||
|
vec![node(
|
||||||
|
"foo",
|
||||||
|
1,
|
||||||
|
vec![node("bar", 2, vec![leaf("a", 3), leaf("b", 3)])]
|
||||||
|
)]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
// differing case should result in only one parent case being added -
|
||||||
|
// the first one
|
||||||
|
col.storage.clear_all_tags()?;
|
||||||
|
note.tags[0] = "foo::BAR::a".into();
|
||||||
|
note.tags[1] = "FOO::bar::b".into();
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
assert_eq!(
|
||||||
|
col.tag_tree()?,
|
||||||
|
node(
|
||||||
|
"",
|
||||||
|
0,
|
||||||
|
vec![node(
|
||||||
|
"foo",
|
||||||
|
1,
|
||||||
|
vec![node("BAR", 2, vec![leaf("a", 3), leaf("b", 3)])]
|
||||||
|
)]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
// things should work even if the immediate parent is not missing
|
||||||
|
col.storage.clear_all_tags()?;
|
||||||
|
note.tags[0] = "foo::bar::baz".into();
|
||||||
|
note.tags[1] = "foo::bar::baz::quux".into();
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
assert_eq!(
|
||||||
|
col.tag_tree()?,
|
||||||
|
node(
|
||||||
|
"",
|
||||||
|
0,
|
||||||
|
vec![node(
|
||||||
|
"foo",
|
||||||
|
1,
|
||||||
|
vec![node("bar", 2, vec![node("baz", 3, vec![leaf("quux", 4)])])]
|
||||||
|
)]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
// numbers have a smaller ascii number than ':', so a naive sort on
|
||||||
|
// '::' would result in one::two being nested under one1.
|
||||||
|
col.storage.clear_all_tags()?;
|
||||||
|
note.tags[0] = "one".into();
|
||||||
|
note.tags[1] = "one1".into();
|
||||||
|
note.tags.push("one::two".into());
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
assert_eq!(
|
||||||
|
col.tag_tree()?,
|
||||||
|
node(
|
||||||
|
"",
|
||||||
|
0,
|
||||||
|
vec![node("one", 1, vec![leaf("two", 2)]), leaf("one1", 1)]
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
// children should match the case of their parents
|
||||||
|
col.storage.clear_all_tags()?;
|
||||||
|
note.tags[0] = "FOO".into();
|
||||||
|
note.tags[1] = "foo::BAR".into();
|
||||||
|
note.tags[2] = "foo::bar::baz".into();
|
||||||
|
col.update_note(&mut note)?;
|
||||||
|
assert_eq!(note.tags, vec!["FOO", "FOO::BAR", "FOO::BAR::baz"]);
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user