generated from ElnuDev/go-project
Compare commits
7 commits
91fb5d9e3b
...
293941c011
Author | SHA1 | Date | |
---|---|---|---|
293941c011 | |||
395318b7f6 | |||
3965b59893 | |||
b20cdaf819 | |||
01e6ab46e7 | |||
d929d50a7c | |||
2b85099901 |
11 changed files with 314 additions and 35 deletions
1
dict/.gitignore
vendored
1
dict/.gitignore
vendored
|
@ -1,2 +1,3 @@
|
||||||
JMdict.xml
|
JMdict.xml
|
||||||
|
JmdictFurigana.txt
|
||||||
dict.bin
|
dict.bin
|
|
@ -9,6 +9,13 @@ Its primary goals are:
|
||||||
|
|
||||||
### Configuration
|
### Configuration
|
||||||
|
|
||||||
|
#### JMdict
|
||||||
|
|
||||||
1. [Download the latest version of JMdict](https://www.edrdg.org/wiki/index.php/JMdict-EDICT_Dictionary_Project#CURRENT_VERSION_&_DOWNLOAD). For development, download the file with only English glosses, which will substantially decrease parsing time.
|
1. [Download the latest version of JMdict](https://www.edrdg.org/wiki/index.php/JMdict-EDICT_Dictionary_Project#CURRENT_VERSION_&_DOWNLOAD). For development, download the file with only English glosses, which will substantially decrease parsing time.
|
||||||
2. Extract the archive
|
2. Extract the archive
|
||||||
3. Rename the file to JMDict.xml
|
3. Rename the file to JMDict.xml
|
||||||
|
|
||||||
|
#### JmdictFurigana
|
||||||
|
|
||||||
|
1. [Download the latest version of JmdictFurigana.txt](https://github.com/Doublevil/JmdictFurigana/releases)
|
||||||
|
2. Ensure it's named JmdictFurigana.txt.
|
186
dict/main.go
186
dict/main.go
|
@ -1,12 +1,15 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bufio"
|
||||||
"encoding/gob"
|
"encoding/gob"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"html/template"
|
||||||
"log"
|
"log"
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
"os"
|
||||||
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"foosoft.net/projects/jmdict"
|
"foosoft.net/projects/jmdict"
|
||||||
|
@ -14,9 +17,15 @@ import (
|
||||||
"github.com/gorilla/mux"
|
"github.com/gorilla/mux"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var words []string // since iterating over map isn't the same every time
|
||||||
var dict map[string]Entry
|
var dict map[string]Entry
|
||||||
|
|
||||||
func LoadDict() error {
|
func LoadDict() error {
|
||||||
|
type binaryData struct {
|
||||||
|
Words []string
|
||||||
|
Dict map[string]Entry
|
||||||
|
}
|
||||||
|
|
||||||
// Loading from binary
|
// Loading from binary
|
||||||
const binaryFile = "dict.bin"
|
const binaryFile = "dict.bin"
|
||||||
file, err := os.Open(binaryFile)
|
file, err := os.Open(binaryFile)
|
||||||
|
@ -27,7 +36,10 @@ func LoadDict() error {
|
||||||
} else {
|
} else {
|
||||||
defer file.Close()
|
defer file.Close()
|
||||||
decoder := gob.NewDecoder(file)
|
decoder := gob.NewDecoder(file)
|
||||||
err = decoder.Decode(&dict)
|
var data binaryData
|
||||||
|
err = decoder.Decode(&data)
|
||||||
|
words = data.Words
|
||||||
|
dict = data.Dict
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -41,10 +53,52 @@ func LoadDict() error {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const jmdictFuriganaFile = "JmdictFurigana.txt"
|
||||||
|
reader, err = os.Open(jmdictFuriganaFile)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
scanner := bufio.NewScanner(reader)
|
||||||
|
furiganaData := make(map[string]string)
|
||||||
|
for scanner.Scan() {
|
||||||
|
params := strings.Split(scanner.Text(), "|")
|
||||||
|
// We need to include the reading as well since some words have the same kanji
|
||||||
|
furiganaData[fmt.Sprintf("%s|%s", params[0], params[1])] = params[2]
|
||||||
|
}
|
||||||
|
|
||||||
|
words = make([]string, len(jmdict.Entries))
|
||||||
dict = make(map[string]Entry)
|
dict = make(map[string]Entry)
|
||||||
for _, jmdictEntry := range jmdict.Entries {
|
for i, jmdictEntry := range jmdict.Entries {
|
||||||
entry := ParseEntry(&jmdictEntry)
|
// お願い致します|おねがいいたします|1:ねが;3:いた
|
||||||
dict[entry.Kanji] = entry
|
var furiganaInfo *string
|
||||||
|
if len(jmdictEntry.Kanji) > 0 {
|
||||||
|
data := furiganaData[fmt.Sprintf("%s|%s", jmdictEntry.Kanji[0].Expression, jmdictEntry.Readings[0].Reading)]
|
||||||
|
furiganaInfo = &data
|
||||||
|
} else {
|
||||||
|
furiganaInfo = nil
|
||||||
|
}
|
||||||
|
entry := ParseEntry(&jmdictEntry, furiganaInfo)
|
||||||
|
offset := 0
|
||||||
|
getKey := func() string {
|
||||||
|
if offset == 0 {
|
||||||
|
// unique
|
||||||
|
return entry.Kanji
|
||||||
|
} else {
|
||||||
|
return fmt.Sprintf("%s-%d", entry.Kanji, offset)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for {
|
||||||
|
if _, ok := dict[getKey()]; ok {
|
||||||
|
offset++
|
||||||
|
} else {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
key := getKey()
|
||||||
|
entry.Key = key
|
||||||
|
words[i] = key
|
||||||
|
dict[key] = entry
|
||||||
}
|
}
|
||||||
|
|
||||||
// Encoding to binary
|
// Encoding to binary
|
||||||
|
@ -54,7 +108,11 @@ func LoadDict() error {
|
||||||
}
|
}
|
||||||
defer file.Close()
|
defer file.Close()
|
||||||
encoder := gob.NewEncoder(file)
|
encoder := gob.NewEncoder(file)
|
||||||
err = encoder.Encode(&dict)
|
data := binaryData{
|
||||||
|
Words: words,
|
||||||
|
Dict: dict,
|
||||||
|
}
|
||||||
|
err = encoder.Encode(&data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -62,8 +120,16 @@ func LoadDict() error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Furigana struct {
|
||||||
|
Kanji string
|
||||||
|
Furigana string
|
||||||
|
}
|
||||||
|
|
||||||
type Entry struct {
|
type Entry struct {
|
||||||
Kanji string
|
Key string
|
||||||
|
Kanji string
|
||||||
|
// Mapping of character index to furigana
|
||||||
|
Furigana []Furigana
|
||||||
Reading string
|
Reading string
|
||||||
Definitions []Definition
|
Definitions []Definition
|
||||||
}
|
}
|
||||||
|
@ -73,13 +139,15 @@ type Definition struct {
|
||||||
PartOfSpeech []string
|
PartOfSpeech []string
|
||||||
}
|
}
|
||||||
|
|
||||||
func ParseEntry(entry *jmdict.JmdictEntry) Entry {
|
func ParseEntry(entry *jmdict.JmdictEntry, furiganaInfo *string) Entry {
|
||||||
kanji := ""
|
kanji := ""
|
||||||
if len(entry.Kanji) > 0 {
|
if len(entry.Kanji) > 0 {
|
||||||
kanji = entry.Kanji[0].Expression
|
kanji = entry.Kanji[0].Expression
|
||||||
|
} else {
|
||||||
|
kanji = entry.Readings[0].Reading
|
||||||
}
|
}
|
||||||
reading := ""
|
reading := ""
|
||||||
if len(entry.Readings) > 0 {
|
if kanji != "" {
|
||||||
reading = entry.Readings[0].Reading
|
reading = entry.Readings[0].Reading
|
||||||
}
|
}
|
||||||
var definitions []Definition
|
var definitions []Definition
|
||||||
|
@ -96,29 +164,104 @@ func ParseEntry(entry *jmdict.JmdictEntry) Entry {
|
||||||
PartOfSpeech: sense.PartsOfSpeech,
|
PartOfSpeech: sense.PartsOfSpeech,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
// 1:ねが;3:いた
|
||||||
|
var furiganaList []Furigana
|
||||||
|
if reading == "" || furiganaInfo == nil || *furiganaInfo == "" {
|
||||||
|
furiganaList = []Furigana{{Kanji: reading, Furigana: ""}}
|
||||||
|
} else {
|
||||||
|
furiganaEntries := strings.Split(*furiganaInfo, ";")
|
||||||
|
// ["1:ねが", "3:いた"]
|
||||||
|
type rawFurigana struct {
|
||||||
|
from int
|
||||||
|
to int
|
||||||
|
furigana string
|
||||||
|
}
|
||||||
|
ruby := make([]rawFurigana, 0)
|
||||||
|
for _, entry := range furiganaEntries {
|
||||||
|
// 1:ねが
|
||||||
|
// multiple: 0-1:きょう
|
||||||
|
params := strings.Split(entry, ":")
|
||||||
|
// ["1", "ねが"]
|
||||||
|
// multiple: ["0-1", "きょう"]
|
||||||
|
indexRange := strings.Split(params[0], "-")
|
||||||
|
// [1]
|
||||||
|
// multiple: [0, 1]
|
||||||
|
var from, to int
|
||||||
|
if len(indexRange) == 1 {
|
||||||
|
index, _ := strconv.Atoi(indexRange[0])
|
||||||
|
from, to = index, index
|
||||||
|
} else {
|
||||||
|
from, _ = strconv.Atoi(indexRange[0])
|
||||||
|
to, _ = strconv.Atoi(indexRange[1])
|
||||||
|
}
|
||||||
|
ruby = append(ruby, rawFurigana{
|
||||||
|
from: from,
|
||||||
|
to: to,
|
||||||
|
furigana: params[1],
|
||||||
|
})
|
||||||
|
}
|
||||||
|
furiganaList = make([]Furigana, 0)
|
||||||
|
slice := func(from, to int) string {
|
||||||
|
return string([]rune(kanji)[from : to+1])
|
||||||
|
}
|
||||||
|
nextIndex := 0
|
||||||
|
for _, raw := range ruby {
|
||||||
|
if raw.from > nextIndex {
|
||||||
|
furiganaList = append(furiganaList, Furigana{
|
||||||
|
Kanji: slice(nextIndex, raw.from-1),
|
||||||
|
Furigana: "",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
furiganaList = append(furiganaList, Furigana{
|
||||||
|
Kanji: slice(raw.from, raw.to),
|
||||||
|
Furigana: raw.furigana,
|
||||||
|
})
|
||||||
|
nextIndex = raw.to + 1
|
||||||
|
}
|
||||||
|
length := len([]rune(kanji))
|
||||||
|
if nextIndex < length {
|
||||||
|
furiganaList = append(furiganaList, Furigana{
|
||||||
|
Kanji: slice(nextIndex, length-1),
|
||||||
|
Furigana: "",
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
return Entry{
|
return Entry{
|
||||||
Kanji: kanji,
|
Kanji: kanji,
|
||||||
|
Furigana: furiganaList,
|
||||||
Reading: reading,
|
Reading: reading,
|
||||||
Definitions: definitions,
|
Definitions: definitions,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func highlight(input, substring string) template.HTML {
|
||||||
|
// Replace all occurrences of substring with the highlighted version
|
||||||
|
replacement := fmt.Sprintf("<mark>%s</mark>", substring)
|
||||||
|
result := strings.ReplaceAll(input, substring, replacement)
|
||||||
|
return template.HTML(result)
|
||||||
|
}
|
||||||
|
|
||||||
func Search(query string) queryResult {
|
func Search(query string) queryResult {
|
||||||
query = strings.TrimSpace(query)
|
query = strings.TrimSpace(query)
|
||||||
exactResults := make([]Entry, 0)
|
exactResults := make([]Entry, 0)
|
||||||
otherResults := make([]Entry, 0)
|
otherResults := make([]Entry, 0)
|
||||||
truncated := false
|
truncated := false
|
||||||
count := 0
|
count := 0
|
||||||
for kanji := range dict {
|
for _, key := range words {
|
||||||
exactMatch := false
|
exactMatch := false
|
||||||
entry := dict[kanji]
|
entry := dict[key]
|
||||||
if kanji == query {
|
if entry.Kanji == query {
|
||||||
exactMatch = true
|
exactMatch = true
|
||||||
goto match
|
goto match
|
||||||
}
|
}
|
||||||
if strings.Contains(kanji, query) {
|
if strings.Contains(entry.Kanji, query) {
|
||||||
goto match
|
goto match
|
||||||
}
|
}
|
||||||
|
for _, definition := range entry.Definitions {
|
||||||
|
if strings.Contains(strings.ToLower(definition.Definition), strings.ToLower(strings.TrimSpace(query))) {
|
||||||
|
goto match
|
||||||
|
}
|
||||||
|
}
|
||||||
// TODO: Skip if query contains kanji
|
// TODO: Skip if query contains kanji
|
||||||
if strings.Contains(entry.Reading, query) {
|
if strings.Contains(entry.Reading, query) {
|
||||||
goto match
|
goto match
|
||||||
|
@ -171,6 +314,12 @@ func main() {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
fmt.Println("JMdict loaded!")
|
fmt.Println("JMdict loaded!")
|
||||||
|
httputils.DefaultTemplateFuncs = template.FuncMap{
|
||||||
|
"highlight": func(input string) string {
|
||||||
|
return input
|
||||||
|
},
|
||||||
|
}
|
||||||
|
httputils.TemplateFuncs = httputils.DefaultTemplateFuncs
|
||||||
r := mux.NewRouter()
|
r := mux.NewRouter()
|
||||||
r.HandleFunc("/", httputils.GenerateHandler(
|
r.HandleFunc("/", httputils.GenerateHandler(
|
||||||
func(w http.ResponseWriter, r *http.Request) bool { return true },
|
func(w http.ResponseWriter, r *http.Request) bool { return true },
|
||||||
|
@ -211,15 +360,20 @@ func main() {
|
||||||
},
|
},
|
||||||
httputils.NewTemplateSet("index.html", "search.html"),
|
httputils.NewTemplateSet("index.html", "search.html"),
|
||||||
// template data
|
// template data
|
||||||
func(w http.ResponseWriter, r *http.Request) (template string, data any) {
|
func(w http.ResponseWriter, r *http.Request) (templateName string, data any) {
|
||||||
if r.Header.Get("HX-Request") == "" {
|
if r.Header.Get("HX-Request") == "" {
|
||||||
template = "search.html"
|
templateName = "search.html"
|
||||||
} else {
|
} else {
|
||||||
template = "search"
|
templateName = "search"
|
||||||
}
|
}
|
||||||
// Only runs if handler returns true
|
// Only runs if handler returns true
|
||||||
query := mux.Vars(r)["query"]
|
query := mux.Vars(r)["query"]
|
||||||
data = Search(query)
|
data = Search(query)
|
||||||
|
httputils.TemplateFuncs = template.FuncMap{
|
||||||
|
"highlight": func(input string) template.HTML {
|
||||||
|
return highlight(input, strings.TrimSpace(query))
|
||||||
|
},
|
||||||
|
}
|
||||||
return
|
return
|
||||||
},
|
},
|
||||||
[]string{http.MethodGet},
|
[]string{http.MethodGet},
|
||||||
|
@ -250,6 +404,6 @@ func main() {
|
||||||
},
|
},
|
||||||
[]string{http.MethodGet},
|
[]string{http.MethodGet},
|
||||||
))
|
))
|
||||||
r.Handle("/", http.FileServer(http.Dir("static")))
|
r.PathPrefix("/").Handler(http.FileServer(http.Dir("static")))
|
||||||
log.Fatal(http.ListenAndServe(":3334", r))
|
log.Fatal(http.ListenAndServe(":3334", r))
|
||||||
}
|
}
|
||||||
|
|
90
dict/static/logo.svg
Normal file
90
dict/static/logo.svg
Normal file
|
@ -0,0 +1,90 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||||
|
<!-- Created with Inkscape (http://www.inkscape.org/) -->
|
||||||
|
|
||||||
|
<svg
|
||||||
|
width="306.39175mm"
|
||||||
|
height="69.833031mm"
|
||||||
|
viewBox="0 0 306.39175 69.833031"
|
||||||
|
version="1.1"
|
||||||
|
id="svg5"
|
||||||
|
sodipodi:docname="logo.svg"
|
||||||
|
inkscape:version="1.2.2 (b0a8486541, 2022-12-01)"
|
||||||
|
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||||
|
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
xmlns:svg="http://www.w3.org/2000/svg">
|
||||||
|
<sodipodi:namedview
|
||||||
|
id="namedview7"
|
||||||
|
pagecolor="#ffffff"
|
||||||
|
bordercolor="#000000"
|
||||||
|
borderopacity="0.25"
|
||||||
|
inkscape:showpageshadow="2"
|
||||||
|
inkscape:pageopacity="0.0"
|
||||||
|
inkscape:pagecheckerboard="0"
|
||||||
|
inkscape:deskcolor="#d1d1d1"
|
||||||
|
inkscape:document-units="mm"
|
||||||
|
showgrid="false"
|
||||||
|
showguides="false"
|
||||||
|
inkscape:zoom="1.1951475"
|
||||||
|
inkscape:cx="579.00803"
|
||||||
|
inkscape:cy="132.20126"
|
||||||
|
inkscape:window-width="1920"
|
||||||
|
inkscape:window-height="1048"
|
||||||
|
inkscape:window-x="1920"
|
||||||
|
inkscape:window-y="32"
|
||||||
|
inkscape:window-maximized="1"
|
||||||
|
inkscape:current-layer="layer1" />
|
||||||
|
<defs
|
||||||
|
id="defs2" />
|
||||||
|
<g
|
||||||
|
inkscape:label="Layer 1"
|
||||||
|
inkscape:groupmode="layer"
|
||||||
|
id="layer1"
|
||||||
|
transform="translate(-11.936241,34.186479)">
|
||||||
|
<g
|
||||||
|
aria-label="jichan.org"
|
||||||
|
id="text843"
|
||||||
|
style="font-weight:bold;font-size:67.7333px;font-family:Arimo;-inkscape-font-specification:'Arimo Bold';letter-spacing:-4.23333px;fill:#ffffff;fill-opacity:0.75;stroke-width:0.264583">
|
||||||
|
<path
|
||||||
|
d="m 25.6861,-23.010484 q -3.047998,0 -4.809064,-1.4224 -1.693332,-1.490132 -1.693332,-4.199464 0,-2.438399 1.693332,-3.996265 1.761066,-1.557866 4.809064,-1.557866 3.047999,0 4.741331,1.490133 1.693333,1.422399 1.693333,4.063998 0,2.506132 -1.761066,4.063998 -1.693332,1.557866 -4.673598,1.557866 z m -9.618128,57.302371 -4.131731,-8.398929 q 4.605864,0 6.36693,-1.896533 1.761066,-1.828799 1.761066,-6.705596 v -33.189317 h 10.837328 v 35.763182 q 0,7.450663 -3.860799,10.905061 -3.793064,3.522132 -10.972794,3.522132 z"
|
||||||
|
style="fill:#d94455;fill-opacity:1"
|
||||||
|
id="path1545" />
|
||||||
|
<path
|
||||||
|
d="m 42.517825,-23.010484 q -3.047998,0 -4.809064,-1.4224 -1.693333,-1.490132 -1.693333,-4.199464 0,-2.438399 1.693333,-3.996265 1.761066,-1.557866 4.809064,-1.557866 3.047999,0 4.741331,1.490133 1.693332,1.422399 1.693332,4.063998 0,2.506132 -1.761065,4.063998 -1.693333,1.557866 -4.673598,1.557866 z m -5.418664,43.078378 v -35.966382 h 10.837328 v 35.966382 z"
|
||||||
|
style="fill:#d94455;fill-opacity:1"
|
||||||
|
id="path1547" />
|
||||||
|
<path
|
||||||
|
d="m 69.645028,20.745227 q -5.215464,0 -9.347196,-2.438399 -4.131731,-2.438399 -6.57013,-6.637863 -2.370665,-4.267198 -2.370665,-9.6181286 0,-5.3509307 2.370665,-9.5503952 2.438399,-4.2671982 6.57013,-6.7055962 4.131732,-2.438399 9.347196,-2.438399 4.809064,0 8.805329,1.693332 4.063998,1.693333 6.434663,4.741331 l -5.892797,7.0442633 q -1.354666,-1.6933324 -3.589865,-2.8447985 -2.167465,-1.2191994 -4.741331,-1.2191994 -3.860798,0 -6.299197,2.6415987 -2.370665,2.5738653 -2.370665,6.6378633 0,3.9285313 2.438399,6.57013 2.438399,2.6415986 6.231463,2.6415986 2.573866,0 4.605865,-1.015999 2.099732,-1.0159999 3.725331,-2.9125323 l 5.825064,7.1119963 q -2.302932,2.912532 -6.434664,4.605864 -4.063998,1.693333 -8.737595,1.693333 z"
|
||||||
|
style="fill:#d94455;fill-opacity:1"
|
||||||
|
id="path1549" />
|
||||||
|
<path
|
||||||
|
d="m 86.544497,20.067894 v -50.122641 h 10.634128 v 18.355724 q 1.896532,-2.235199 4.741335,-3.589865 2.91253,-1.354666 6.09599,-1.354666 6.096,0 9.27946,3.725331 3.18347,3.7253317 3.18347,10.9050614 V 20.067894 H 109.57382 V -1.1326287 q 0,-6.57013 -5.55413,-6.3669301 -2.8448,0 -4.741333,1.7610658 -1.896532,1.6933324 -1.896532,4.2671978 V 20.067894 Z"
|
||||||
|
style="fill:#d94455;fill-opacity:1"
|
||||||
|
id="path1551" />
|
||||||
|
<path
|
||||||
|
d="m 138.59755,20.677494 q -4.53813,0 -8.12799,-2.370666 -3.52214,-2.438399 -5.62187,-6.637863 -2.09973,-4.267198 -2.09973,-9.6181286 0,-5.4863973 2.09973,-9.6858618 2.09973,-4.1994646 5.6896,-6.5701296 3.6576,-2.438399 8.39893,-2.438399 3.52213,0 6.23146,1.422399 2.70933,1.422399 4.4704,3.657598 v -4.334931 h 10.76959 v 35.966382 h -10.97279 v -4.334931 q -1.8288,2.235199 -4.6736,3.589865 -2.77706,1.354666 -6.16373,1.354666 z m 2.98027,-9.143996 q 3.72533,0 5.96053,-2.5738651 2.2352,-2.5738654 2.2352,-6.9087965 0,-4.3349312 -2.2352,-6.9087965 -2.2352,-2.6415987 -5.96053,-2.6415987 -3.6576,0 -5.8928,2.6415987 -2.16747,2.5738653 -2.16747,6.9087965 0,4.3349311 2.16747,6.9087965 2.2352,2.5738651 5.8928,2.5738651 z"
|
||||||
|
style="fill:#d94455;fill-opacity:1"
|
||||||
|
id="path1553" />
|
||||||
|
<path
|
||||||
|
d="m 165.72476,20.067894 v -35.966382 h 10.29547 l 0.2032,4.334931 q 1.89653,-2.235199 4.80906,-3.657598 2.91253,-1.422399 6.16373,-1.422399 6.096,0 9.27946,3.725331 3.18347,3.7253317 3.18347,10.9050614 V 20.067894 H 188.75409 V -1.1326287 q 0,-6.57013 -5.55413,-6.3669301 -2.8448,0 -4.74134,1.7610658 -1.89653,1.6933324 -1.89653,4.2671978 V 20.067894 Z"
|
||||||
|
style="fill:#d94455;fill-opacity:1"
|
||||||
|
id="path1555" />
|
||||||
|
<path
|
||||||
|
d="m 208.36291,20.81296 q -2.77707,0 -4.33494,-1.557866 -1.49013,-1.557865 -1.49013,-4.470397 0,-2.709332 1.6256,-4.402665 1.6256,-1.6933323 4.19947,-1.6933323 2.77706,0 4.33493,1.6255993 1.55786,1.557866 1.55786,4.470398 0,2.641598 -1.6256,4.334931 -1.6256,1.693332 -4.26719,1.693332 z"
|
||||||
|
style="fill:#708090;fill-opacity:1"
|
||||||
|
id="path1557" />
|
||||||
|
<path
|
||||||
|
d="m 234.60955,20.745227 q -5.62186,0 -10.02453,-2.370665 -4.33493,-2.438399 -6.90879,-6.637864 -2.50614,-4.2671977 -2.50614,-9.6858616 0,-5.418664 2.50614,-9.6181285 2.57386,-4.1994649 6.90879,-6.6378629 4.40267,-2.438399 10.02453,-2.438399 5.55413,0 9.88906,2.438399 4.40267,2.438398 6.9088,6.6378629 2.57386,4.1994645 2.57386,9.6181285 0,5.4186639 -2.57386,9.6858616 -2.50613,4.199465 -6.9088,6.637864 -4.33493,2.370665 -9.88906,2.370665 z m 0,-9.347195 q 3.6576,0 6.02826,-2.641599 2.37067,-2.6415987 2.30294,-6.7055966 0.0677,-4.1317313 -2.30294,-6.7733299 -2.37066,-2.6415987 -6.02826,-2.6415987 -3.6576,0 -6.096,2.709332 -2.37066,2.6415986 -2.30293,6.7055966 -0.0677,4.0639979 2.30293,6.7055966 2.4384,2.641599 6.096,2.641599 z"
|
||||||
|
style="fill:#708090;fill-opacity:1"
|
||||||
|
id="path1559" />
|
||||||
|
<path
|
||||||
|
d="m 256.85999,20.067894 v -35.966382 h 10.29546 l 0.2032,5.486397 q 1.8288,-2.777065 4.60586,-4.470397 2.8448,-1.761066 6.02827,-1.761066 1.2192,0 2.2352,0.2032 1.08373,0.2032 1.96426,0.474133 l -2.98026,11.9887937 q -0.74507,-0.474133 -2.032,-0.7450662 -1.2192,-0.3386665 -2.57387,-0.3386665 -2.98026,0 -4.94453,1.9642656 -1.89653,1.9642657 -1.89653,5.1477308 V 20.067894 Z"
|
||||||
|
style="fill:#708090;fill-opacity:1"
|
||||||
|
id="path1561" />
|
||||||
|
<path
|
||||||
|
d="m 298.88853,35.646553 q -4.40266,0 -8.80533,-1.286933 -4.33493,-1.286933 -7.04426,-3.318932 l 3.79306,-7.586129 q 2.16747,1.490133 4.8768,2.302932 2.77707,0.880533 5.75733,0.880533 5.08,0 7.5184,-2.302932 2.4384,-2.235199 2.4384,-6.97653 V 15.46203 q -1.76107,1.896532 -4.60587,3.115731 -2.77706,1.2192 -5.8928,1.2192 -4.94453,0 -8.80532,-2.235199 -3.79307,-2.302932 -6.02827,-6.299197 -2.16746,-4.0639979 -2.16746,-9.3471952 0,-5.418664 2.16746,-9.5503952 2.16747,-4.1994646 5.96053,-6.5701296 3.79307,-2.438399 8.5344,-2.438399 3.52213,0 6.5024,1.354666 2.98026,1.286932 4.74133,3.251198 l 0.2032,-3.860798 h 10.29546 v 33.05385 q 0,8.805329 -5.21547,13.614393 -5.14773,4.876798 -14.22399,4.876798 z m 0.27093,-24.248521 q 3.92854,0 6.23147,-2.5061324 2.37066,-2.5738654 2.37066,-6.9087965 0,-4.3349312 -2.37066,-6.9087965 -2.30293,-2.5738654 -6.23147,-2.5738654 -3.86079,0 -6.16373,2.6415987 -2.30293,2.5738653 -2.30293,6.8410632 0,4.2671978 2.30293,6.8410632 2.30294,2.5738657 6.16373,2.5738657 z"
|
||||||
|
style="fill:#708090;fill-opacity:1"
|
||||||
|
id="path1563" />
|
||||||
|
</g>
|
||||||
|
</g>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 7.8 KiB |
|
@ -22,8 +22,11 @@
|
||||||
<body hx-boost="true">
|
<body hx-boost="true">
|
||||||
<main>
|
<main>
|
||||||
<a href="/">
|
<a href="/">
|
||||||
<img src="https://jichan.org/logo.svg" style="height: 4em; display: block; margin: 1em auto 1em auto">
|
<img src="/logo.svg" style="height: 4em; display: block; margin: 1em auto 1em auto">
|
||||||
</a>
|
</a>
|
||||||
|
<div class="box warn">
|
||||||
|
<b>jichan.org</b> is in open alpha. The site may be buggy. More features are coming soon. If you have ideas or want to contribute, DM me on Discord <a href="https://discord.com/users/441283734214279178" target="_blank">@elnudev</a> or <a href="https://codeberg.org/ElnuDev/jichanorg/issues" target="_blank">open an issue on Codeberg</a>.
|
||||||
|
</div>
|
||||||
<form
|
<form
|
||||||
hx-get="/search"
|
hx-get="/search"
|
||||||
hx-replace-url="true"
|
hx-replace-url="true"
|
||||||
|
@ -32,7 +35,18 @@
|
||||||
<input type="text" name="q" value="{{ block "value" . }}{{ end }}" placeholder="辞書をサーチする" class="width:100%" autocomplete="false" required>
|
<input type="text" name="q" value="{{ block "value" . }}{{ end }}" placeholder="辞書をサーチする" class="width:100%" autocomplete="false" required>
|
||||||
</form>
|
</form>
|
||||||
<div id="results">
|
<div id="results">
|
||||||
{{ block "results" . }}{{ if .Query }}{{ template "search" . }}{{ end }}{{ end }}
|
{{ block "results" . }}
|
||||||
|
{{ if .Query }}
|
||||||
|
{{ template "search" . }}
|
||||||
|
{{ else }}
|
||||||
|
<h3>Thank you to</h3>
|
||||||
|
<ul>
|
||||||
|
<li><a href="https://www.edrdg.org/wiki/index.php/JMdict-EDICT_Dictionary_Project" target="_blank"><b>JMdict</b></a> for providing free and open dictionary data</li>
|
||||||
|
<li><a href="https://foosoft.net/" target="_blank">Alexei Yatskov</a> a.k.a. Foosoft Productions for providing the <a href="https:/foosoft.net/projects/jmdict">foosoft.net/projects/jmdict</a> Go package for parsing JMdict, and also for creating Yomichan, a great inspiration</li>
|
||||||
|
<li>Doublevil for providing <a href="https://github.com/Doublevil/JmdictFurigana">JmdictFurigana</a></li>
|
||||||
|
</ul>
|
||||||
|
{{ end }}
|
||||||
|
{{ end }}
|
||||||
</div>
|
</div>
|
||||||
<br>
|
<br>
|
||||||
</main>
|
</main>
|
||||||
|
|
|
@ -1,3 +1,3 @@
|
||||||
{{- define "definition" -}}
|
{{- define "definition" -}}
|
||||||
{{ if .PartOfSpeech }}<small><chip>{{ .PartOfSpeech }}</chip></small><br>{{ end }}{{ .Definition -}}
|
{{ if .PartOfSpeech }}<small><chip>{{ .PartOfSpeech }}</chip></small><br>{{ end }}{{ highlight .Definition -}}
|
||||||
{{ end }}
|
{{ end }}
|
|
@ -1,11 +1,15 @@
|
||||||
{{ define "entry" }}
|
{{ define "entry" }}
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<h3>
|
<h3 lang="ja">
|
||||||
{{- if .Kanji -}}
|
<a href="/word/{{ .Key }}">
|
||||||
<a href="/word/{{ .Kanji }}"><ruby>{{- .Kanji -}}<rp>(</rp><rt>{{- .Reading -}}</rt><rp>)</rp></ruby></a>
|
{{- range .Furigana -}}
|
||||||
{{- else -}}
|
{{- if .Furigana -}}
|
||||||
{{- .Reading -}}
|
<ruby>{{- .Kanji -}}<rp>(</rp><rt>{{- .Furigana -}}</rt><rp>)</rp></ruby>
|
||||||
|
{{- else -}}
|
||||||
|
{{- .Kanji -}}
|
||||||
|
{{- end -}}
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
|
</a>
|
||||||
</h3>
|
</h3>
|
||||||
{{- $count := len .Definitions -}}
|
{{- $count := len .Definitions -}}
|
||||||
{{ if eq $count 1 -}}
|
{{ if eq $count 1 -}}
|
||||||
|
|
|
@ -1,10 +1,12 @@
|
||||||
{{ define "entryfull" }}
|
{{ define "entryfull" }}
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<h3>
|
<h3 lang="ja">
|
||||||
{{- if .Kanji -}}
|
{{- range .Furigana -}}
|
||||||
<ruby>{{- .Kanji -}}<rp>(</rp><rt>{{- .Reading -}}</rt><rp>)</rp></ruby>
|
{{- if .Furigana -}}
|
||||||
{{- else -}}
|
<ruby>{{- .Kanji -}}<rp>(</rp><rt>{{- .Furigana -}}</rt><rp>)</rp></ruby>
|
||||||
{{- .Reading -}}
|
{{- else -}}
|
||||||
|
{{- .Kanji -}}
|
||||||
|
{{- end -}}
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
</h3>
|
</h3>
|
||||||
{{- $count := len .Definitions -}}
|
{{- $count := len .Definitions -}}
|
||||||
|
|
|
@ -1 +1 @@
|
||||||
{{ define "sitetitle" }}jidict{{ end }}
|
{{ define "sitetitle" }}jichan.org{{ end }}
|
|
@ -3,7 +3,7 @@
|
||||||
{{- define "value" }}{{ .Query }}{{- end -}}
|
{{- define "value" }}{{ .Query }}{{- end -}}
|
||||||
|
|
||||||
{{- define "results" -}}
|
{{- define "results" -}}
|
||||||
{{- template "entryfull" .Entry -}}
|
{{- template "search" . -}}
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
|
|
||||||
{{- template "index" . -}}
|
{{- template "index" . -}}
|
|
@ -29,9 +29,11 @@ func newTemplateSet(partials *TemplateSet, paths ...string) TemplateSet {
|
||||||
fileInfo, _ := os.Stat(path)
|
fileInfo, _ := os.Stat(path)
|
||||||
modTimes[path] = fileInfo.ModTime()
|
modTimes[path] = fileInfo.ModTime()
|
||||||
}
|
}
|
||||||
templates := template.Must(template.ParseFiles(allPaths...))
|
templates := template.Template{}
|
||||||
|
templates.Funcs(DefaultTemplateFuncs)
|
||||||
|
templates.ParseFiles(allPaths...)
|
||||||
return TemplateSet{
|
return TemplateSet{
|
||||||
templates: templates,
|
templates: &templates,
|
||||||
paths: allPaths,
|
paths: allPaths,
|
||||||
loadTimes: modTimes,
|
loadTimes: modTimes,
|
||||||
}
|
}
|
||||||
|
@ -46,7 +48,10 @@ func NewTemplateSet(paths ...string) TemplateSet {
|
||||||
|
|
||||||
func (templateSet *TemplateSet) ExecuteTemplate(wr io.Writer, name string, data any) error {
|
func (templateSet *TemplateSet) ExecuteTemplate(wr io.Writer, name string, data any) error {
|
||||||
templateSet.reloadTemplatesIfModified()
|
templateSet.reloadTemplatesIfModified()
|
||||||
return templateSet.templates.ExecuteTemplate(wr, name, data)
|
templateSet.templates.Funcs(TemplateFuncs)
|
||||||
|
err := templateSet.templates.ExecuteTemplate(wr, name, data)
|
||||||
|
TemplateFuncs = DefaultTemplateFuncs
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (templateSet *TemplateSet) reloadTemplatesIfModified() {
|
func (templateSet *TemplateSet) reloadTemplatesIfModified() {
|
||||||
|
@ -86,3 +91,5 @@ const partialsFolder = templateFolder + "/partials"
|
||||||
|
|
||||||
var paths, _ = getTemplatePathsInDirectory(partialsFolder)
|
var paths, _ = getTemplatePathsInDirectory(partialsFolder)
|
||||||
var partials = newTemplateSet(nil, paths...)
|
var partials = newTemplateSet(nil, paths...)
|
||||||
|
var DefaultTemplateFuncs template.FuncMap
|
||||||
|
var TemplateFuncs template.FuncMap
|
||||||
|
|
Reference in a new issue