mirror of
https://github.com/charmbracelet/glow
synced 2024-12-14 06:02:27 +00:00
Normalize strings during filtering
This makes it easier to match characters with diacritics (i.e. you can type 'o' and match an 'ö').
This commit is contained in:
parent
6cf1152f87
commit
de09997fdb
3 changed files with 33 additions and 6 deletions
1
go.mod
1
go.mod
|
@ -24,4 +24,5 @@ require (
|
||||||
golang.org/x/crypto v0.0.0-20201016220609-9e8e0b390897
|
golang.org/x/crypto v0.0.0-20201016220609-9e8e0b390897
|
||||||
golang.org/x/net v0.0.0-20200813134508-3edf25e44fcc // indirect
|
golang.org/x/net v0.0.0-20200813134508-3edf25e44fcc // indirect
|
||||||
golang.org/x/sys v0.0.0-20201020230747-6e5568b54d1a
|
golang.org/x/sys v0.0.0-20201020230747-6e5568b54d1a
|
||||||
|
golang.org/x/text v0.3.2
|
||||||
)
|
)
|
||||||
|
|
28
ui/stash.go
28
ui/stash.go
|
@ -9,6 +9,7 @@ import (
|
||||||
"sort"
|
"sort"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
"unicode"
|
||||||
|
|
||||||
"github.com/charmbracelet/bubbles/paginator"
|
"github.com/charmbracelet/bubbles/paginator"
|
||||||
"github.com/charmbracelet/bubbles/spinner"
|
"github.com/charmbracelet/bubbles/spinner"
|
||||||
|
@ -21,6 +22,8 @@ import (
|
||||||
"github.com/muesli/reflow/ansi"
|
"github.com/muesli/reflow/ansi"
|
||||||
te "github.com/muesli/termenv"
|
te "github.com/muesli/termenv"
|
||||||
"github.com/sahilm/fuzzy"
|
"github.com/sahilm/fuzzy"
|
||||||
|
"golang.org/x/text/transform"
|
||||||
|
"golang.org/x/text/unicode/norm"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
|
@ -231,13 +234,16 @@ func (m *stashModel) getNotes() []*markdown {
|
||||||
targets := []string{}
|
targets := []string{}
|
||||||
|
|
||||||
for _, t := range m.markdowns {
|
for _, t := range m.markdowns {
|
||||||
note := ""
|
note, err := normalize(t.Note)
|
||||||
if t.markdownType == newsMarkdown {
|
if err != nil && debug {
|
||||||
note = "News: " + t.Note
|
log.Printf("error normalizing '%s': %v", t.Note, err)
|
||||||
} else {
|
|
||||||
note = t.Note
|
note = t.Note
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if t.markdownType == newsMarkdown {
|
||||||
|
note = "News: " + note
|
||||||
|
}
|
||||||
|
|
||||||
targets = append(targets, note)
|
targets = append(targets, note)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1081,6 +1087,20 @@ func deleteStashedItem(cc *charm.Client, id int) tea.Cmd {
|
||||||
|
|
||||||
// ETC
|
// ETC
|
||||||
|
|
||||||
|
// Normalize text to aid in the filtering process. in particular, we remove
|
||||||
|
// diacritics.
|
||||||
|
func normalize(in string) (string, error) {
|
||||||
|
t := transform.Chain(norm.NFD, transform.RemoveFunc(isMn), norm.NFC)
|
||||||
|
out, _, err := transform.String(t, in)
|
||||||
|
return out, err
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns whether a given rune is a nonspacing mark (Mn is the key for
|
||||||
|
// nonspacing marks)
|
||||||
|
func isMn(r rune) bool {
|
||||||
|
return unicode.Is(unicode.Mn, r)
|
||||||
|
}
|
||||||
|
|
||||||
// wrapMarkdowns wraps a *charm.Markdown with a *markdown in order to add some
|
// wrapMarkdowns wraps a *charm.Markdown with a *markdown in order to add some
|
||||||
// extra metadata.
|
// extra metadata.
|
||||||
func wrapMarkdowns(t markdownType, md []*charm.Markdown) (m []*markdown) {
|
func wrapMarkdowns(t markdownType, md []*charm.Markdown) (m []*markdown) {
|
||||||
|
|
|
@ -2,6 +2,7 @@ package ui
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"log"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/charmbracelet/charm/ui/common"
|
"github.com/charmbracelet/charm/ui/common"
|
||||||
|
@ -122,13 +123,18 @@ func stashItemView(b *strings.Builder, m stashModel, index int, md *markdown) {
|
||||||
func styleFilteredText(haystack, needles string, defaultStyle, matchedStyle termenv.Style) string {
|
func styleFilteredText(haystack, needles string, defaultStyle, matchedStyle termenv.Style) string {
|
||||||
b := strings.Builder{}
|
b := strings.Builder{}
|
||||||
|
|
||||||
matches := fuzzy.Find(needles, []string{haystack})
|
normalizedHay, err := normalize(haystack)
|
||||||
|
if err != nil && debug {
|
||||||
|
log.Printf("error normalizing '%s': %v", haystack, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
matches := fuzzy.Find(needles, []string{normalizedHay})
|
||||||
if len(matches) == 0 {
|
if len(matches) == 0 {
|
||||||
return defaultStyle.Styled(haystack)
|
return defaultStyle.Styled(haystack)
|
||||||
}
|
}
|
||||||
|
|
||||||
m := matches[0] // only one match exists
|
m := matches[0] // only one match exists
|
||||||
for i, rune := range haystack {
|
for i, rune := range []rune(haystack) {
|
||||||
styled := false
|
styled := false
|
||||||
for _, mi := range m.MatchedIndexes {
|
for _, mi := range m.MatchedIndexes {
|
||||||
if i == mi {
|
if i == mi {
|
||||||
|
|
Loading…
Reference in a new issue