summaryrefslogtreecommitdiff
path: root/pngmeta/metareader.go
diff options
context:
space:
mode:
authorGrail Finder <wohilas@gmail.com>2024-12-02 19:58:03 +0300
committerGrail Finder <wohilas@gmail.com>2024-12-02 19:58:03 +0300
commita5ab816c9434022159548ac0d1872160b3c8908d (patch)
treea4e68c334f2a514f02afd65a1bb70eff415fb8cd /pngmeta/metareader.go
parent8d3997baff25f28c23441699be7692f853cf1f0f (diff)
Feat: add character card support
Diffstat (limited to 'pngmeta/metareader.go')
-rw-r--r--pngmeta/metareader.go107
1 files changed, 107 insertions, 0 deletions
diff --git a/pngmeta/metareader.go b/pngmeta/metareader.go
new file mode 100644
index 0000000..ea726c9
--- /dev/null
+++ b/pngmeta/metareader.go
@@ -0,0 +1,107 @@
+package pngmeta
+
+import (
+ "bytes"
+ "elefant/models"
+ "encoding/base64"
+ "encoding/json"
+ "errors"
+ "io"
+ "os"
+ "path"
+ "strings"
+)
+
+const (
+ embType = "tEXt"
+)
+
+type PngEmbed struct {
+ Key string
+ Value string
+}
+
+func (c PngEmbed) GetDecodedValue() (*models.CharCardSpec, error) {
+ data, err := base64.StdEncoding.DecodeString(c.Value)
+ if err != nil {
+ return nil, err
+ }
+ card := &models.CharCardSpec{}
+ if err := json.Unmarshal(data, &card); err != nil {
+ return nil, err
+ }
+ return card, nil
+}
+
+func extractChar(fname string) (*PngEmbed, error) {
+ data, err := os.ReadFile(fname)
+ if err != nil {
+ return nil, err
+ }
+ reader := bytes.NewReader(data)
+ pr, err := NewPNGStepReader(reader)
+ if err != nil {
+ return nil, err
+ }
+ for {
+ step, err := pr.Next()
+ if err != nil {
+ if errors.Is(err, io.EOF) {
+ break
+ }
+ }
+ if step.Type() != embType {
+ if _, err := io.Copy(io.Discard, step); err != nil {
+ return nil, err
+ }
+ } else {
+ buf, err := io.ReadAll(step)
+ if err != nil {
+ return nil, err
+ }
+ dataInstep := string(buf)
+ values := strings.Split(dataInstep, "\x00")
+ if len(values) == 2 {
+ return &PngEmbed{Key: values[0], Value: values[1]}, nil
+ }
+ }
+ if err := step.Close(); err != nil {
+ return nil, err
+ }
+ }
+ return nil, errors.New("failed to find embedded char in png: " + fname)
+}
+
+func ReadCard(fname, uname string) (*models.CharCard, error) {
+ pe, err := extractChar(fname)
+ if err != nil {
+ return nil, err
+ }
+ charSpec, err := pe.GetDecodedValue()
+ if err != nil {
+ return nil, err
+ }
+ return charSpec.Simplify(uname, fname), nil
+}
+
+func ReadDirCards(dirname, uname string) ([]*models.CharCard, error) {
+ files, err := os.ReadDir(dirname)
+ if err != nil {
+ return nil, err
+ }
+ resp := []*models.CharCard{}
+ for _, f := range files {
+ if !strings.HasSuffix(f.Name(), ".png") {
+ continue
+ }
+ fpath := path.Join(dirname, f.Name())
+ cc, err := ReadCard(fpath, uname)
+ if err != nil {
+ // log err
+ return nil, err
+ // continue
+ }
+ resp = append(resp, cc)
+ }
+ return resp, nil
+}