This PR adds support for including/excluding specific items from team filtering. Users can use the filter modal to only show teams that include specific items, only show teams that _don't_ include specific items, or combine the two to create a very powerful filter.
80 lines
2.1 KiB
TypeScript
80 lines
2.1 KiB
TypeScript
import { escapeForRegEx, Range } from '@tiptap/core'
|
|
import { ResolvedPos } from '@tiptap/pm/model'
|
|
|
|
export interface Trigger {
|
|
char: string
|
|
allowSpaces: boolean
|
|
allowedPrefixes: string[] | null
|
|
startOfLine: boolean
|
|
$position: ResolvedPos
|
|
}
|
|
|
|
export type SuggestionMatch = {
|
|
range: Range
|
|
query: string
|
|
text: string
|
|
} | null
|
|
|
|
export function findSuggestionMatch(config: Trigger): SuggestionMatch {
|
|
const { char, allowSpaces, allowedPrefixes, startOfLine, $position } = config
|
|
|
|
const escapedChar = escapeForRegEx(char)
|
|
const suffix = new RegExp(`\\s${escapedChar}$`)
|
|
const prefix = startOfLine ? '^' : ''
|
|
// const regexp = allowSpaces
|
|
// ? new RegExp(`${prefix}${escapedChar}.*?(?=\\s${escapedChar}|$)`, 'gm')
|
|
// : new RegExp(`${prefix}(?:^)?${escapedChar}[^\\s${escapedChar}]*`, 'gm')
|
|
const regexp = new RegExp(`^(.*)$`, 'gm')
|
|
|
|
const text = $position.nodeBefore?.isText && $position.nodeBefore.text
|
|
|
|
if (!text) {
|
|
return null
|
|
}
|
|
|
|
const textFrom = $position.pos - text.length
|
|
const match = Array.from(text.matchAll(regexp)).pop()
|
|
|
|
if (!match || match.input === undefined || match.index === undefined) {
|
|
return null
|
|
}
|
|
|
|
// JavaScript doesn't have lookbehinds. This hacks a check that first character
|
|
// is a space or the start of the line
|
|
const matchPrefix = match.input.slice(
|
|
Math.max(0, match.index - 1),
|
|
match.index
|
|
)
|
|
const matchPrefixIsAllowed = new RegExp(
|
|
`^[${allowedPrefixes?.join('')}\0]?$`
|
|
).test(matchPrefix)
|
|
|
|
if (allowedPrefixes !== null && !matchPrefixIsAllowed) {
|
|
return null
|
|
}
|
|
|
|
// The absolute position of the match in the document
|
|
const from = textFrom + match.index
|
|
let to = from + match[0].length
|
|
|
|
// Edge case handling; if spaces are allowed and we're directly in between
|
|
// two triggers
|
|
if (allowSpaces && suffix.test(text.slice(to - 1, to + 1))) {
|
|
match[0] += ' '
|
|
to += 1
|
|
}
|
|
|
|
// If the $position is located within the matched substring, return that range
|
|
if (from < $position.pos && to >= $position.pos) {
|
|
return {
|
|
range: {
|
|
from,
|
|
to,
|
|
},
|
|
query: match[0],
|
|
text: match[0],
|
|
}
|
|
}
|
|
|
|
return null
|
|
}
|