« get me outta code hell

mtui - Music Text User Interface - user-friendly command line music player
about summary refs log tree commit diff
diff options
context:
space:
mode:
-rw-r--r--playlist-utils.js124
1 files changed, 84 insertions, 40 deletions
diff --git a/playlist-utils.js b/playlist-utils.js
index 5fbfff8..317ff84 100644
--- a/playlist-utils.js
+++ b/playlist-utils.js
@@ -831,6 +831,54 @@ function getPathScore(path1, path2) {
   return scores.reduce((a, b) => a < b ? a : b)
 }
 
+function getNameScore(name1, name2) {
+  // Pretty simple algorithm here: we're looking for the longest continuous
+  // series of words which is shared between both names. The score is the
+  // length of that series, so a higher score is better (and a zero score
+  // means no overlap).
+
+  // Split into chunks of word characters, taking out any non-word (\W)
+  // characters between.
+  const toWords = name => name.split(/\W+/)
+
+  const words1 = toWords(name1)
+  const words2 = toWords(name2)
+
+  const getLongestMatch = (parse, against) => {
+    let longestMatch = 0
+
+    for (let i = 0; i < parse.length; i++) {
+      const word = parse[i]
+
+      for (let j = 0; j < against.length; j++) {
+        if (against[j] !== word) {
+          continue
+        }
+
+        let offset = 1
+        while (
+          parse[i + offset] &&
+          against[i + offset] &&
+          parse[i + offset] === against[j + offset]
+        ) {
+          offset++
+        }
+
+        if (offset > longestMatch) {
+          longestMatch = offset
+        }
+      }
+    }
+
+    return longestMatch
+  }
+
+  return Math.max(
+    getLongestMatch(words1, words2),
+    getLongestMatch(words2, words1)
+  )
+}
+
 function findItemObject(referenceData, possibleChoices) {
   // Finds the item object in the provided choices which most closely resembles
   // the provided reference data. This is used for maintaining the identity of
@@ -838,10 +886,9 @@ function findItemObject(referenceData, possibleChoices) {
   // also usable in synchronizing the identity of items across linked clients
   // (see socket.js).
 
-  // Reference data includes item NAME, item SOURCE (downloaderArg), and item
-  // PATH (names of parent groups). Specifics of how existing item  objects are
-  // determined to resemble this data are laid out next to the relevant
-  // implementation code.
+  // Reference data includes item NAME and item PATH (names of parent groups).
+  // Specifics of how existing item objects are determined to resemble this
+  // data are laid out next to the relevant implementation code.
   //
   // TODO: Should track number be considered here?
   // TODO: Should track "metadata" (duration, md5?) be considered too?
@@ -859,11 +906,18 @@ function findItemObject(referenceData, possibleChoices) {
     return getPathScore(path1, path2)
   }
 
-  // The only items which will be considered at all are those which match at
-  // least one of the reference name/source.
-  const baselineResemble = possibleChoices.filter(item =>
-    item.name === referenceData.name ||
-    item.downloaderArg && item.downloaderArg === referenceData.downloaderArg)
+  function getItemNameScore(item) {
+    const name1 = referenceData.name
+    const name2 = item.name
+    return getNameScore(name1, name2)
+  }
+
+  // The only items which will be considered at all are those which at least
+  // partially match the reference name.
+  const baselineResemble = possibleChoices.map(item => ({
+    item,
+    nameScore: getItemNameScore(item)
+  })).filter(item => item.nameScore > 0)
 
   // If no item matches the baseline conditions for resemblance at all,
   // return null. It's up to the caller to decide what to do in this case,
@@ -875,40 +929,27 @@ function findItemObject(referenceData, possibleChoices) {
 
   // Find the "reasons" these items resemble the reference data; these will
   // be used as the factors in calculating which item resembles closest.
-  const reasons = baselineResemble.map(item => ({
+  const reasons = baselineResemble.map(({item, nameScore}) => ({
     item,
-    nameMatches: item.name === referenceData.name,
-    sourceMatches: item.downloaderArg && item.downloaderArg === referenceData.downloaderArg,
-    pathScore: getItemPathScore(item)
+    pathScore: getItemPathScore(item),
+    nameScore
   }))
 
-  // TODO: The algorithm for determining which track matches closest is
-  //       rudimentary at best right now. It would be well improved with
-  //       better-detailed reasoning! That said, here's the current code
-  //       laid out explicitly:
-  //
-  // Initial sort by matches is (NAME & SOURCE), SOURCE, NAME. Track which
-  // most closely matches path is returned thereafter, with ties broken by
-  // the initial sort. (If name, source, *and* path all are equal, first
-  // track as ordered in the source playlist/parent group is returned.)
-  // If no tracks have any path overlap, the first item in the sorted list
-  // is returned (since it is the closest match). (Again, ties here are
-  // broken according to source ordering.)
-
-  reasons.sort((a, b) =>
-    a.sourceMatches && !b.sourceMatches ? -1 :
-    !a.sourceMatches && b.sourceMatches ? 1 :
-    a.nameMatches && !b.nameMatches ? -1 :
-    !a.nameMatches && b.nameMatches ? 1 :
-    0)
-
-  let mostResembles
-  const sharePath = reasons.filter(({ pathScore }) => pathScore >= 0)
-  if (sharePath.length) {
-    mostResembles = sharePath.reduce((a, b) => a.pathScore < b.pathScore ? a : b)
-  } else {
-    mostResembles = reasons[0]
-  }
+  // TODO: Are there circumstances in which a strong path score should be
+  //       prioritized in spite of weaker name score?
+
+  // Sort by closest matching filenames first.
+  reasons.sort((a, b) => b.nameScore - a.nameScore)
+
+  // Filter only the best name matches.
+  const bestNameScore = reasons[0].nameScore
+  const bestName = reasons.filter(({ nameScore }) => nameScore === bestNameScore)
+
+  // Then choose the best matching path.
+  const sharePath = bestName.filter(({ pathScore }) => pathScore >= 0)
+  const mostResembles = (sharePath.length
+    ? sharePath.reduce((a, b) => a.pathScore < b.pathScore ? a : b)
+    : reasons[0])
 
   return mostResembles.item
 }
@@ -947,6 +988,9 @@ if (require.main === module) {
   console.log(getPathScore(['A', 'B', 'C'], ['A', 'B', 'C', 'D']))
   console.log(getPathScore(['A', 'B', 'C', 'E'], ['A', 'B', 'C']))
   console.log(getPathScore(['W', 'X'], ['Y', 'Z']))
+  console.log(getNameScore('C418 - Vlem', 'Vlem'))
+  console.log(getNameScore('glimmer', 'glimmer'))
+  console.log(getNameScore('C418 - Vlem', 'covet - glimmer'))
   console.log(findItemObject(
     // {name: 'T', downloaderArg: 'foo', path: ['A', 'B', 'C']},
     {name: 'B'},