mirror of
https://github.com/coalaura/ffwebp.git
synced 2025-09-08 13:59:54 +00:00
avif
This commit is contained in:
1
.gitignore
vendored
1
.gitignore
vendored
@@ -2,3 +2,4 @@ bin
|
|||||||
example.*
|
example.*
|
||||||
test.*
|
test.*
|
||||||
*.exe
|
*.exe
|
||||||
|
ffwebp
|
@@ -80,7 +80,7 @@ func main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if err := app.Run(context.Background(), os.Args); err != nil {
|
if err := app.Run(context.Background(), os.Args); err != nil {
|
||||||
logx.Errorf("fatal: %v", err)
|
logx.Errorf("fatal: %v\n", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -112,6 +112,13 @@ func run(_ context.Context, cmd *cli.Command) error {
|
|||||||
logx.Printf("reading input from <stdin>\n")
|
logx.Printf("reading input from <stdin>\n")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sniffed, reader, err := codec.Sniff(reader)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
logx.Printf("sniffed codec: %s (%q)\n", sniffed.Codec, sniffed)
|
||||||
|
|
||||||
if output = cmd.String("output"); output != "-" {
|
if output = cmd.String("output"); output != "-" {
|
||||||
logx.Printf("opening output file %q\n", filepath.ToSlash(output))
|
logx.Printf("opening output file %q\n", filepath.ToSlash(output))
|
||||||
|
|
||||||
@@ -132,13 +139,6 @@ func run(_ context.Context, cmd *cli.Command) error {
|
|||||||
|
|
||||||
common.FillDefaults()
|
common.FillDefaults()
|
||||||
|
|
||||||
sniffed, reader, err := codec.Sniff(reader)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
logx.Printf("sniffed codec: %s (%q)\n", sniffed.Codec, sniffed)
|
|
||||||
|
|
||||||
oCodec, err := codec.Detect(output, cmd.String("codec"))
|
oCodec, err := codec.Detect(output, cmd.String("codec"))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
|
1
go.mod
1
go.mod
@@ -5,6 +5,7 @@ go 1.24.2
|
|||||||
require github.com/urfave/cli/v3 v3.3.8
|
require github.com/urfave/cli/v3 v3.3.8
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
github.com/gen2brain/avif v0.4.4
|
||||||
github.com/gen2brain/webp v0.5.5
|
github.com/gen2brain/webp v0.5.5
|
||||||
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646
|
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646
|
||||||
golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8
|
golang.org/x/image v0.0.0-20191009234506-e7c1f5e7dbb8
|
||||||
|
2
go.sum
2
go.sum
@@ -2,6 +2,8 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c
|
|||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/ebitengine/purego v0.8.3 h1:K+0AjQp63JEZTEMZiwsI9g0+hAMNohwUOtY0RPGexmc=
|
github.com/ebitengine/purego v0.8.3 h1:K+0AjQp63JEZTEMZiwsI9g0+hAMNohwUOtY0RPGexmc=
|
||||||
github.com/ebitengine/purego v0.8.3/go.mod h1:iIjxzd6CiRiOG0UyXP+V1+jWqUXVjPKLAI0mRfJZTmQ=
|
github.com/ebitengine/purego v0.8.3/go.mod h1:iIjxzd6CiRiOG0UyXP+V1+jWqUXVjPKLAI0mRfJZTmQ=
|
||||||
|
github.com/gen2brain/avif v0.4.4 h1:Ga/ss7qcWWQm2bxFpnjYjhJsNfZrWs5RsyklgFjKRSE=
|
||||||
|
github.com/gen2brain/avif v0.4.4/go.mod h1:/XCaJcjZraQwKVhpu9aEd9aLOssYOawLvhMBtmHVGqk=
|
||||||
github.com/gen2brain/webp v0.5.5 h1:MvQR75yIPU/9nSqYT5h13k4URaJK3gf9tgz/ksRbyEg=
|
github.com/gen2brain/webp v0.5.5 h1:MvQR75yIPU/9nSqYT5h13k4URaJK3gf9tgz/ksRbyEg=
|
||||||
github.com/gen2brain/webp v0.5.5/go.mod h1:xOSMzp4aROt2KFW++9qcK/RBTOVC2S9tJG66ip/9Oc0=
|
github.com/gen2brain/webp v0.5.5/go.mod h1:xOSMzp4aROt2KFW++9qcK/RBTOVC2S9tJG66ip/9Oc0=
|
||||||
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 h1:zYyBkD/k9seD2A7fsi6Oo2LfFZAehjjQMERAvZLEDnQ=
|
github.com/nfnt/resize v0.0.0-20180221191011-83c6a9932646 h1:zYyBkD/k9seD2A7fsi6Oo2LfFZAehjjQMERAvZLEDnQ=
|
||||||
|
8
internal/builtins/avif.go
Normal file
8
internal/builtins/avif.go
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
//go:build avif || core || full
|
||||||
|
// +build avif core full
|
||||||
|
|
||||||
|
package builtins
|
||||||
|
|
||||||
|
import (
|
||||||
|
_ "github.com/coalaura/ffwebp/internal/codec/avif"
|
||||||
|
)
|
119
internal/codec/avif/avif.go
Normal file
119
internal/codec/avif/avif.go
Normal file
@@ -0,0 +1,119 @@
|
|||||||
|
package avif
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"fmt"
|
||||||
|
"image"
|
||||||
|
"io"
|
||||||
|
|
||||||
|
"github.com/gen2brain/avif"
|
||||||
|
|
||||||
|
"github.com/coalaura/ffwebp/internal/codec"
|
||||||
|
"github.com/coalaura/ffwebp/internal/logx"
|
||||||
|
"github.com/coalaura/ffwebp/internal/opts"
|
||||||
|
"github.com/urfave/cli/v3"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
quality int
|
||||||
|
qualityA int
|
||||||
|
speed int
|
||||||
|
chroma int
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
codec.Register(impl{})
|
||||||
|
}
|
||||||
|
|
||||||
|
type impl struct{}
|
||||||
|
|
||||||
|
func (impl) String() string {
|
||||||
|
return "avif"
|
||||||
|
}
|
||||||
|
|
||||||
|
func (impl) Extensions() []string {
|
||||||
|
return []string{"avif"}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (impl) Flags(flags []cli.Flag) []cli.Flag {
|
||||||
|
return append(flags,
|
||||||
|
&cli.IntFlag{
|
||||||
|
Name: "avif.quality-alpha",
|
||||||
|
Usage: "AVIF: alpha channel quality in range [0-100]",
|
||||||
|
Value: 60,
|
||||||
|
Destination: &qualityA,
|
||||||
|
Validator: func(v int) error {
|
||||||
|
if v < 0 || v > 100 {
|
||||||
|
return fmt.Errorf("invalid avif.quality-alpha: %d", v)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
},
|
||||||
|
&cli.IntFlag{
|
||||||
|
Name: "avif.speed",
|
||||||
|
Usage: "AVIF: encoding speed in range [0-10] (0=slowest/best)",
|
||||||
|
Value: 6,
|
||||||
|
Destination: &speed,
|
||||||
|
Validator: func(v int) error {
|
||||||
|
if v < 0 || v > 10 {
|
||||||
|
return fmt.Errorf("invalid avif.speed: %d", v)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
},
|
||||||
|
&cli.IntFlag{
|
||||||
|
Name: "avif.chroma",
|
||||||
|
Usage: "AVIF: chroma subsampling (444=best, 422, 420=smallest)",
|
||||||
|
Value: 444,
|
||||||
|
Destination: &chroma,
|
||||||
|
Validator: func(v int) error {
|
||||||
|
if v != 444 && v != 422 && v != 420 {
|
||||||
|
return fmt.Errorf("invalid avif.chroma: %d", v)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
},
|
||||||
|
},
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (impl) Sniff(reader io.ReaderAt) (int, []byte, error) {
|
||||||
|
buf := make([]byte, 12)
|
||||||
|
|
||||||
|
if _, err := reader.ReadAt(buf, 0); err != nil {
|
||||||
|
return 0, nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
if bytes.Equal(buf[4:12], []byte("ftypavif")) {
|
||||||
|
return 100, buf[:12], nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0, nil, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (impl) Decode(reader io.Reader) (image.Image, error) {
|
||||||
|
return avif.Decode(reader)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (impl) Encode(writer io.Writer, img image.Image, options opts.Common) error {
|
||||||
|
logx.Printf("avif: quality=%d, quality-alpha=%d, speed=%d, chroma=%d\n", options.Quality, qualityA, speed, chroma)
|
||||||
|
|
||||||
|
return avif.Encode(writer, img, avif.Options{
|
||||||
|
Quality: options.Quality,
|
||||||
|
QualityAlpha: qualityA,
|
||||||
|
Speed: speed,
|
||||||
|
ChromaSubsampling: chromaSubsampling(chroma),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func chromaSubsampling(c int) image.YCbCrSubsampleRatio {
|
||||||
|
switch c {
|
||||||
|
case 444:
|
||||||
|
return image.YCbCrSubsampleRatio444
|
||||||
|
case 422:
|
||||||
|
return image.YCbCrSubsampleRatio422
|
||||||
|
case 420:
|
||||||
|
return image.YCbCrSubsampleRatio420
|
||||||
|
default:
|
||||||
|
return image.YCbCrSubsampleRatio444
|
||||||
|
}
|
||||||
|
}
|
@@ -61,7 +61,7 @@ func Sniff(reader io.Reader) (*Sniffed, io.Reader, error) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if guess == nil {
|
if guess == nil {
|
||||||
return nil, nil, errors.New("unknown format")
|
return nil, nil, errors.New("unknown input format")
|
||||||
}
|
}
|
||||||
|
|
||||||
return &Sniffed{
|
return &Sniffed{
|
||||||
|
Reference in New Issue
Block a user