convert.go 2.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. package images
  2. import (
  3. "bytes"
  4. "encoding/binary"
  5. "fmt"
  6. "image"
  7. "io"
  8. "os"
  9. "strings"
  10. "kitty/tools/cli"
  11. "kitty/tools/utils"
  12. )
  13. var _ = fmt.Print
  14. func encode_rgba(output io.Writer, img image.Image) (err error) {
  15. var final_img *image.NRGBA
  16. switch img.(type) {
  17. case *image.NRGBA:
  18. final_img = img.(*image.NRGBA)
  19. default:
  20. b := img.Bounds()
  21. final_img = image.NewNRGBA(image.Rect(0, 0, b.Dx(), b.Dy()))
  22. ctx := Context{}
  23. ctx.PasteCenter(final_img, img, nil)
  24. }
  25. b := final_img.Bounds()
  26. header := make([]byte, 8)
  27. var width = utils.Abs(b.Dx())
  28. var height = utils.Abs(b.Dy())
  29. binary.LittleEndian.PutUint32(header, uint32(width))
  30. binary.LittleEndian.PutUint32(header[4:], uint32(height))
  31. readers := []io.Reader{bytes.NewReader(header)}
  32. stride := 4 * width
  33. if final_img.Stride == stride {
  34. readers = append(readers, bytes.NewReader(final_img.Pix))
  35. } else {
  36. p := final_img.Pix
  37. for y := 0; y < b.Dy(); y++ {
  38. readers = append(readers, bytes.NewReader(p[:min(stride, len(p))]))
  39. p = p[final_img.Stride:]
  40. }
  41. }
  42. _, err = io.Copy(output, io.MultiReader(readers...))
  43. return
  44. }
  45. func convert_image(input io.ReadSeeker, output io.Writer, format string) (err error) {
  46. image_data, err := OpenNativeImageFromReader(input)
  47. if err != nil {
  48. return err
  49. }
  50. if len(image_data.Frames) == 0 {
  51. return fmt.Errorf("Image has no frames")
  52. }
  53. img := image_data.Frames[0].Img
  54. q := strings.ToLower(format)
  55. if q == "rgba" {
  56. return encode_rgba(output, img)
  57. }
  58. mt := utils.GuessMimeType("file." + q)
  59. if mt == "" {
  60. return fmt.Errorf("Unknown image output format: %s", format)
  61. }
  62. return Encode(output, img, mt)
  63. }
  64. func ConvertEntryPoint(root *cli.Command) {
  65. root.AddSubCommand(&cli.Command{
  66. Name: "__convert_image__",
  67. Hidden: true,
  68. OnlyArgsAllowed: true,
  69. Run: func(cmd *cli.Command, args []string) (rc int, err error) {
  70. if len(args) != 1 {
  71. return 1, fmt.Errorf("Usage: __convert_image__ OUTPUT_FORMAT")
  72. }
  73. format := args[0]
  74. buf := bytes.NewBuffer(make([]byte, 0, 1024*1024))
  75. if _, err = io.Copy(buf, os.Stdin); err != nil {
  76. return 1, err
  77. }
  78. if err = convert_image(bytes.NewReader(buf.Bytes()), os.Stdout, format); err != nil {
  79. rc = 1
  80. }
  81. return
  82. },
  83. })
  84. }