github.com/dominikszabo/hugo-ds-clean@v0.47.1/resource/smartcrop.go (about)

     1  // Copyright 2017-present The Hugo Authors. All rights reserved.
     2  //
     3  // Licensed under the Apache License, Version 2.0 (the "License");
     4  // you may not use this file except in compliance with the License.
     5  // You may obtain a copy of the License at
     6  // http://www.apache.org/licenses/LICENSE-2.0
     7  //
     8  // Unless required by applicable law or agreed to in writing, software
     9  // distributed under the License is distributed on an "AS IS" BASIS,
    10  // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    11  // See the License for the specific language governing permissions and
    12  // limitations under the License.
    13  
    14  package resource
    15  
    16  import (
    17  	"image"
    18  
    19  	"github.com/disintegration/imaging"
    20  	"github.com/muesli/smartcrop"
    21  )
    22  
    23  const (
    24  	// Do not change.
    25  	smartCropIdentifier = "smart"
    26  
    27  	// This is just a increment, starting on 1. If Smart Crop improves its cropping, we
    28  	// need a way to trigger a re-generation of the crops in the wild, so increment this.
    29  	smartCropVersionNumber = 1
    30  )
    31  
    32  // Needed by smartcrop
    33  type imagingResizer struct {
    34  	filter imaging.ResampleFilter
    35  }
    36  
    37  func (r imagingResizer) Resize(img image.Image, width, height uint) image.Image {
    38  	return imaging.Resize(img, int(width), int(height), r.filter)
    39  }
    40  
    41  func newSmartCropAnalyzer(filter imaging.ResampleFilter) smartcrop.Analyzer {
    42  	return smartcrop.NewAnalyzer(imagingResizer{filter: filter})
    43  }
    44  
    45  func smartCrop(img image.Image, width, height int, anchor imaging.Anchor, filter imaging.ResampleFilter) (*image.NRGBA, error) {
    46  
    47  	if width <= 0 || height <= 0 {
    48  		return &image.NRGBA{}, nil
    49  	}
    50  
    51  	srcBounds := img.Bounds()
    52  	srcW := srcBounds.Dx()
    53  	srcH := srcBounds.Dy()
    54  
    55  	if srcW <= 0 || srcH <= 0 {
    56  		return &image.NRGBA{}, nil
    57  	}
    58  
    59  	if srcW == width && srcH == height {
    60  		return imaging.Clone(img), nil
    61  	}
    62  
    63  	smart := newSmartCropAnalyzer(filter)
    64  
    65  	rect, err := smart.FindBestCrop(img, width, height)
    66  
    67  	if err != nil {
    68  		return nil, err
    69  	}
    70  
    71  	b := img.Bounds().Intersect(rect)
    72  
    73  	cropped, err := imaging.Crop(img, b), nil
    74  	if err != nil {
    75  		return nil, err
    76  	}
    77  
    78  	return imaging.Resize(cropped, width, height, filter), nil
    79  
    80  }