/* Copyright © 2024 Seednode */ package images import ( "errors" "fmt" "image" _ "image/gif" _ "image/jpeg" _ "image/png" "math/rand" "net/http" "os" "strings" _ "golang.org/x/image/bmp" _ "golang.org/x/image/webp" "seedno.de/seednode/roulette/types" ) type dimensions struct { width int height int } type Format struct { NoButtons bool Fun bool } func (t Format) CSP(w http.ResponseWriter) string { nonce := types.GetNonce() w.Header().Add("Content-Security-Policy", fmt.Sprintf("default-src 'self' 'nonce-%s';", nonce)) return nonce } func (t Format) CSS() string { var css strings.Builder css.WriteString(`html,body{margin:0;padding:0;height:100%;}`) if t.NoButtons { css.WriteString(`a{color:inherit;display:block;height:100%;width:100%;text-decoration:none;}`) } else { css.WriteString(`a{color:inherit;display:block;height:97%;width:100%;text-decoration:none;}`) } css.WriteString(`table{margin-left:auto;margin-right:auto;}`) css.WriteString(`img{margin:auto;display:block;max-width:97%;max-height:97%;`) css.WriteString(`object-fit:scale-down;position:absolute;top:50%;left:50%;transform:translate(-50%,-50%)`) if t.Fun { rotate := rand.Intn(360) css.WriteString(fmt.Sprintf(" rotate(%ddeg);", rotate)) css.WriteString(fmt.Sprintf("-ms-transform:rotate(%ddeg);", rotate)) css.WriteString(fmt.Sprintf("-webkit-transform:rotate(%ddeg);", rotate)) css.WriteString(fmt.Sprintf("-moz-transform:rotate(%ddeg);", rotate)) css.WriteString(fmt.Sprintf("-o-transform:rotate(%ddeg)", rotate)) } css.WriteString(`;}`) return css.String() } func (t Format) Title(rootUrl, fileUri, filePath, fileName, prefix, mime string) (string, error) { dimensions, err := ImageDimensions(filePath) if err != nil { return "", err } return fmt.Sprintf(`%s (%dx%d)`, fileName, dimensions.width, dimensions.height), nil } func (t Format) Body(rootUrl, fileUri, filePath, fileName, prefix, mime, nonce string) (string, error) { dimensions, err := ImageDimensions(filePath) if err != nil { return "", err } var w strings.Builder w.WriteString(fmt.Sprintf(`Roulette selected: %s`, rootUrl, nonce, fileUri, dimensions.width, dimensions.height, mime, fileName)) return w.String(), nil } func (t Format) Extensions() map[string]string { return map[string]string{ `.apng`: `image/apng`, `.avif`: `image/avif`, `.bmp`: `image/bmp`, `.gif`: `image/gif`, `.jpg`: `image/jpeg`, `.jpeg`: `image/jpeg`, `.jfif`: `image/jpeg`, `.pjp`: `image/jpeg`, `.pjpeg`: `image/jpeg`, `.png`: `image/png`, `.webp`: `image/webp`, } } func (t Format) Validate(filePath string) bool { return true } func (t Format) MediaType(extension string) string { extensions := t.Extensions() value, exists := extensions[extension] if exists { return value } return "" } func ImageDimensions(path string) (*dimensions, error) { file, err := os.Open(path) switch { case errors.Is(err, os.ErrNotExist): fmt.Printf("File %s does not exist\n", path) return &dimensions{}, nil case err != nil: fmt.Printf("File %s open returned error: %s\n", path, err) return &dimensions{}, err } defer file.Close() decodedConfig, _, err := image.DecodeConfig(file) switch { case errors.Is(err, image.ErrFormat): fmt.Printf("File %s has invalid image format\n", path) return &dimensions{width: 0, height: 0}, nil case err != nil: fmt.Printf("File %s decode returned error: %s\n", path, err) return &dimensions{}, err } return &dimensions{width: decodedConfig.Width, height: decodedConfig.Height}, nil } func (t Format) Type() string { return "embed" } func init() { types.SupportedFormats.Register(Format{}) }