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
|
||||
JmdictFurigana.txt
|
||||
dict.bin
|
|
@ -9,6 +9,13 @@ Its primary goals are:
|
|||
|
||||
### 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.
|
||||
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
|
||||
|
||||
import (
|
||||
"bufio"
|
||||
"encoding/gob"
|
||||
"encoding/json"
|
||||
"fmt"
|
||||
"html/template"
|
||||
"log"
|
||||
"net/http"
|
||||
"os"
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
"foosoft.net/projects/jmdict"
|
||||
|
@ -14,9 +17,15 @@ import (
|
|||
"github.com/gorilla/mux"
|
||||
)
|
||||
|
||||
var words []string // since iterating over map isn't the same every time
|
||||
var dict map[string]Entry
|
||||
|
||||
func LoadDict() error {
|
||||
type binaryData struct {
|
||||
Words []string
|
||||
Dict map[string]Entry
|
||||
}
|
||||
|
||||
// Loading from binary
|
||||
const binaryFile = "dict.bin"
|
||||
file, err := os.Open(binaryFile)
|
||||
|
@ -27,7 +36,10 @@ func LoadDict() error {
|
|||
} else {
|
||||
defer file.Close()
|
||||
decoder := gob.NewDecoder(file)
|
||||
err = decoder.Decode(&dict)
|
||||
var data binaryData
|
||||
err = decoder.Decode(&data)
|
||||
words = data.Words
|
||||
dict = data.Dict
|
||||
return err
|
||||
}
|
||||
|
||||
|
@ -41,10 +53,52 @@ func LoadDict() error {
|
|||
if err != nil {
|
||||
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)
|
||||
for _, jmdictEntry := range jmdict.Entries {
|
||||
entry := ParseEntry(&jmdictEntry)
|
||||
dict[entry.Kanji] = entry
|
||||
for i, jmdictEntry := range jmdict.Entries {
|
||||
// お願い致します|おねがいいたします|1:ねが;3:いた
|
||||
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
|
||||
|
@ -54,7 +108,11 @@ func LoadDict() error {
|
|||
}
|
||||
defer file.Close()
|
||||
encoder := gob.NewEncoder(file)
|
||||
err = encoder.Encode(&dict)
|
||||
data := binaryData{
|
||||
Words: words,
|
||||
Dict: dict,
|
||||
}
|
||||
err = encoder.Encode(&data)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
@ -62,8 +120,16 @@ func LoadDict() error {
|
|||
return nil
|
||||
}
|
||||
|
||||
type Furigana struct {
|
||||
Kanji string
|
||||
Furigana string
|
||||
}
|
||||
|
||||
type Entry struct {
|
||||
Kanji string
|
||||
Key string
|
||||
Kanji string
|
||||
// Mapping of character index to furigana
|
||||
Furigana []Furigana
|
||||
Reading string
|
||||
Definitions []Definition
|
||||
}
|
||||
|
@ -73,13 +139,15 @@ type Definition struct {
|
|||
PartOfSpeech []string
|
||||
}
|
||||
|
||||
func ParseEntry(entry *jmdict.JmdictEntry) Entry {
|
||||
func ParseEntry(entry *jmdict.JmdictEntry, furiganaInfo *string) Entry {
|
||||
kanji := ""
|
||||
if len(entry.Kanji) > 0 {
|
||||
kanji = entry.Kanji[0].Expression
|
||||
} else {
|
||||
kanji = entry.Readings[0].Reading
|
||||
}
|
||||
reading := ""
|
||||
if len(entry.Readings) > 0 {
|
||||
if kanji != "" {
|
||||
reading = entry.Readings[0].Reading
|
||||
}
|
||||
var definitions []Definition
|
||||
|
@ -96,29 +164,104 @@ func ParseEntry(entry *jmdict.JmdictEntry) Entry {
|
|||
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{
|
||||
Kanji: kanji,
|
||||
Furigana: furiganaList,
|
||||
Reading: reading,
|
||||
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 {
|
||||
query = strings.TrimSpace(query)
|
||||
exactResults := make([]Entry, 0)
|
||||
otherResults := make([]Entry, 0)
|
||||
truncated := false
|
||||
count := 0
|
||||
for kanji := range dict {
|
||||
for _, key := range words {
|
||||
exactMatch := false
|
||||
entry := dict[kanji]
|
||||
if kanji == query {
|
||||
entry := dict[key]
|
||||
if entry.Kanji == query {
|
||||
exactMatch = true
|
||||
goto match
|
||||
}
|
||||
if strings.Contains(kanji, query) {
|
||||
if strings.Contains(entry.Kanji, query) {
|
||||
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
|
||||
if strings.Contains(entry.Reading, query) {
|
||||
goto match
|
||||
|
@ -171,6 +314,12 @@ func main() {
|
|||
return
|
||||
}
|
||||
fmt.Println("JMdict loaded!")
|
||||
httputils.DefaultTemplateFuncs = template.FuncMap{
|
||||
"highlight": func(input string) string {
|
||||
return input
|
||||
},
|
||||
}
|
||||
httputils.TemplateFuncs = httputils.DefaultTemplateFuncs
|
||||
r := mux.NewRouter()
|
||||
r.HandleFunc("/", httputils.GenerateHandler(
|
||||
func(w http.ResponseWriter, r *http.Request) bool { return true },
|
||||
|
@ -211,15 +360,20 @@ func main() {
|
|||
},
|
||||
httputils.NewTemplateSet("index.html", "search.html"),
|
||||
// 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") == "" {
|
||||
template = "search.html"
|
||||
templateName = "search.html"
|
||||
} else {
|
||||
template = "search"
|
||||
templateName = "search"
|
||||
}
|
||||
// Only runs if handler returns true
|
||||
query := mux.Vars(r)["query"]
|
||||
data = Search(query)
|
||||
httputils.TemplateFuncs = template.FuncMap{
|
||||
"highlight": func(input string) template.HTML {
|
||||
return highlight(input, strings.TrimSpace(query))
|
||||
},
|
||||
}
|
||||
return
|
||||
},
|
||||
[]string{http.MethodGet},
|
||||
|
@ -250,6 +404,6 @@ func main() {
|
|||
},
|
||||
[]string{http.MethodGet},
|
||||
))
|
||||
r.Handle("/", http.FileServer(http.Dir("static")))
|
||||
r.PathPrefix("/").Handler(http.FileServer(http.Dir("static")))
|
||||
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">
|
||||
<main>
|
||||
<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>
|
||||
<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
|
||||
hx-get="/search"
|
||||
hx-replace-url="true"
|
||||
|
@ -32,7 +35,18 @@
|
|||
<input type="text" name="q" value="{{ block "value" . }}{{ end }}" placeholder="辞書をサーチする" class="width:100%" autocomplete="false" required>
|
||||
</form>
|
||||
<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>
|
||||
<br>
|
||||
</main>
|
||||
|
|
|
@ -1,3 +1,3 @@
|
|||
{{- define "definition" -}}
|
||||
{{ if .PartOfSpeech }}<small><chip>{{ .PartOfSpeech }}</chip></small><br>{{ end }}{{ .Definition -}}
|
||||
{{ if .PartOfSpeech }}<small><chip>{{ .PartOfSpeech }}</chip></small><br>{{ end }}{{ highlight .Definition -}}
|
||||
{{ end }}
|
|
@ -1,11 +1,15 @@
|
|||
{{ define "entry" }}
|
||||
<div class="box">
|
||||
<h3>
|
||||
{{- if .Kanji -}}
|
||||
<a href="/word/{{ .Kanji }}"><ruby>{{- .Kanji -}}<rp>(</rp><rt>{{- .Reading -}}</rt><rp>)</rp></ruby></a>
|
||||
{{- else -}}
|
||||
{{- .Reading -}}
|
||||
<h3 lang="ja">
|
||||
<a href="/word/{{ .Key }}">
|
||||
{{- range .Furigana -}}
|
||||
{{- if .Furigana -}}
|
||||
<ruby>{{- .Kanji -}}<rp>(</rp><rt>{{- .Furigana -}}</rt><rp>)</rp></ruby>
|
||||
{{- else -}}
|
||||
{{- .Kanji -}}
|
||||
{{- end -}}
|
||||
{{- end -}}
|
||||
</a>
|
||||
</h3>
|
||||
{{- $count := len .Definitions -}}
|
||||
{{ if eq $count 1 -}}
|
||||
|
|
|
@ -1,10 +1,12 @@
|
|||
{{ define "entryfull" }}
|
||||
<div class="box">
|
||||
<h3>
|
||||
{{- if .Kanji -}}
|
||||
<ruby>{{- .Kanji -}}<rp>(</rp><rt>{{- .Reading -}}</rt><rp>)</rp></ruby>
|
||||
{{- else -}}
|
||||
{{- .Reading -}}
|
||||
<h3 lang="ja">
|
||||
{{- range .Furigana -}}
|
||||
{{- if .Furigana -}}
|
||||
<ruby>{{- .Kanji -}}<rp>(</rp><rt>{{- .Furigana -}}</rt><rp>)</rp></ruby>
|
||||
{{- else -}}
|
||||
{{- .Kanji -}}
|
||||
{{- end -}}
|
||||
{{- end -}}
|
||||
</h3>
|
||||
{{- $count := len .Definitions -}}
|
||||
|
|
|
@ -1 +1 @@
|
|||
{{ define "sitetitle" }}jidict{{ end }}
|
||||
{{ define "sitetitle" }}jichan.org{{ end }}
|
|
@ -3,7 +3,7 @@
|
|||
{{- define "value" }}{{ .Query }}{{- end -}}
|
||||
|
||||
{{- define "results" -}}
|
||||
{{- template "entryfull" .Entry -}}
|
||||
{{- template "search" . -}}
|
||||
{{- end -}}
|
||||
|
||||
{{- template "index" . -}}
|
|
@ -29,9 +29,11 @@ func newTemplateSet(partials *TemplateSet, paths ...string) TemplateSet {
|
|||
fileInfo, _ := os.Stat(path)
|
||||
modTimes[path] = fileInfo.ModTime()
|
||||
}
|
||||
templates := template.Must(template.ParseFiles(allPaths...))
|
||||
templates := template.Template{}
|
||||
templates.Funcs(DefaultTemplateFuncs)
|
||||
templates.ParseFiles(allPaths...)
|
||||
return TemplateSet{
|
||||
templates: templates,
|
||||
templates: &templates,
|
||||
paths: allPaths,
|
||||
loadTimes: modTimes,
|
||||
}
|
||||
|
@ -46,7 +48,10 @@ func NewTemplateSet(paths ...string) TemplateSet {
|
|||
|
||||
func (templateSet *TemplateSet) ExecuteTemplate(wr io.Writer, name string, data any) error {
|
||||
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() {
|
||||
|
@ -86,3 +91,5 @@ const partialsFolder = templateFolder + "/partials"
|
|||
|
||||
var paths, _ = getTemplatePathsInDirectory(partialsFolder)
|
||||
var partials = newTemplateSet(nil, paths...)
|
||||
var DefaultTemplateFuncs template.FuncMap
|
||||
var TemplateFuncs template.FuncMap
|
||||
|
|
Reference in a new issue