66 lines
1.6 KiB
Go
66 lines
1.6 KiB
Go
package crop
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"image"
|
|
"io/ioutil"
|
|
"path/filepath"
|
|
|
|
"github.com/disintegration/imaging"
|
|
"github.com/photoprism/photoprism/internal/thumb"
|
|
"github.com/photoprism/photoprism/pkg/fs"
|
|
)
|
|
|
|
// FromRequest returns the crop file name for an image hash, and creates it if needed.
|
|
func FromRequest(hash, area string, size Size, thumbPath string) (fileName string, err error) {
|
|
if fileName, err = FromCache(hash, area, size, thumbPath); err == nil {
|
|
return fileName, err
|
|
}
|
|
|
|
a := AreaFromString(area)
|
|
|
|
thumbName, err := ThumbFileName(hash, a, size, thumbPath)
|
|
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
// Compose cached crop image file name.
|
|
cropBase := fmt.Sprintf("%s_%dx%d_crop_%s%s", hash, size.Width, size.Height, area, fs.JpegExt)
|
|
cropName := filepath.Join(filepath.Dir(thumbName), cropBase)
|
|
|
|
imageBuffer, err := ioutil.ReadFile(thumbName)
|
|
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
img, err := imaging.Decode(bytes.NewReader(imageBuffer))
|
|
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
// Get absolute crop coordinates and dimension.
|
|
min, max, dim := a.Bounds(img)
|
|
|
|
if dim < size.Width {
|
|
log.Debugf("crop: %s is too small, upscaling %dpx to %dpx", filepath.Base(thumbName), dim, size.Width)
|
|
}
|
|
|
|
// Crop area from image.
|
|
img = imaging.Crop(img, image.Rect(min.X, min.Y, max.X, max.Y))
|
|
|
|
// Resample crop area.
|
|
img = thumb.Resample(img, size.Width, size.Height, size.Options...)
|
|
|
|
// Save crop image.
|
|
if err := imaging.Save(img, cropName); err != nil {
|
|
log.Errorf("failed caching %s", filepath.Base(cropName))
|
|
} else {
|
|
log.Debugf("saved %s", filepath.Base(cropName))
|
|
}
|
|
|
|
return cropName, nil
|
|
}
|