Skip to content

Commit 05cd128

Browse files
committed
Add gravity to extend option
1 parent 7c7ac56 commit 05cd128

6 files changed

+88
-82
lines changed

CHANGELOG.md

+1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88

99
### Changed
1010
- Docker image base is changed to Debian 10 for better stability and performance.
11+
- `extend` option now supports gravity.
1112

1213
## [2.7.0] - 2019-11-13
1314
### Changed

docs/generating_the_url_advanced.md

+6-5
Original file line numberDiff line numberDiff line change
@@ -121,13 +121,14 @@ Default: false
121121
#### Extend
122122

123123
```
124-
extend:%extend
125-
ex:%extend
124+
extend:%extend:%gravity
125+
ex:%extend:%gravity
126126
```
127127

128-
When set to `1`, `t` or `true`, imgproxy will extend the image if it is smaller than the given size.
128+
* When `extend` is set to `1`, `t` or `true`, imgproxy will extend the image if it is smaller than the given size.
129+
* `gravity` _(optional)_ accepts the same values as [gravity](#gravity) option, except `sm`. When `gravity` is not set, imgproxy will use `ce` gravity without offsets.
129130

130-
Default: false
131+
Default: `false:ce:0:0`
131132

132133
#### Gravity
133134

@@ -167,7 +168,7 @@ c:%width:%height:%gravity
167168
Defines an area of the image to be processed (crop before resize).
168169

169170
* `width` and `height` define the size of the area. When `width` or `height` is set to `0`, imgproxy will use the full width/height of the source image.
170-
* `gravity` accepts the same values as [gravity](#gravity) option. When `gravity` is not set, imgproxy will use the value of the [gravity](#gravity) option.
171+
* `gravity` _(optional)_ accepts the same values as [gravity](#gravity) option. When `gravity` is not set, imgproxy will use the value of the [gravity](#gravity) option.
171172

172173
#### Quality
173174

process.go

+35-24
Original file line numberDiff line numberDiff line change
@@ -157,40 +157,48 @@ func calcJpegShink(scale float64, imgtype imageType) int {
157157
return 1
158158
}
159159

160-
func calcCrop(width, height, cropWidth, cropHeight int, gravity *gravityOptions) (left, top int) {
160+
func calcPosition(width, height, innerWidth, innerHeight int, gravity *gravityOptions, allowOverflow bool) (left, top int) {
161161
if gravity.Type == gravityFocusPoint {
162162
pointX := scaleInt(width, gravity.X)
163163
pointY := scaleInt(height, gravity.Y)
164164

165-
left = maxInt(0, minInt(pointX-cropWidth/2, width-cropWidth))
166-
top = maxInt(0, minInt(pointY-cropHeight/2, height-cropHeight))
165+
left = pointX - innerWidth/2
166+
top = pointY - innerHeight/2
167+
} else {
168+
offX, offY := int(gravity.X), int(gravity.Y)
167169

168-
return
169-
}
170+
left = (width-innerWidth+1)/2 + offX
171+
top = (height-innerHeight+1)/2 + offY
170172

171-
offX, offY := int(gravity.X), int(gravity.Y)
173+
if gravity.Type == gravityNorth || gravity.Type == gravityNorthEast || gravity.Type == gravityNorthWest {
174+
top = 0 + offY
175+
}
172176

173-
left = (width-cropWidth+1)/2 + offX
174-
top = (height-cropHeight+1)/2 + offY
177+
if gravity.Type == gravityEast || gravity.Type == gravityNorthEast || gravity.Type == gravitySouthEast {
178+
left = width - innerWidth - offX
179+
}
175180

176-
if gravity.Type == gravityNorth || gravity.Type == gravityNorthEast || gravity.Type == gravityNorthWest {
177-
top = 0 + offY
178-
}
181+
if gravity.Type == gravitySouth || gravity.Type == gravitySouthEast || gravity.Type == gravitySouthWest {
182+
top = height - innerHeight - offY
183+
}
179184

180-
if gravity.Type == gravityEast || gravity.Type == gravityNorthEast || gravity.Type == gravitySouthEast {
181-
left = width - cropWidth - offX
185+
if gravity.Type == gravityWest || gravity.Type == gravityNorthWest || gravity.Type == gravitySouthWest {
186+
left = 0 + offX
187+
}
182188
}
183189

184-
if gravity.Type == gravitySouth || gravity.Type == gravitySouthEast || gravity.Type == gravitySouthWest {
185-
top = height - cropHeight - offY
186-
}
190+
var minX, maxX, minY, maxY int
187191

188-
if gravity.Type == gravityWest || gravity.Type == gravityNorthWest || gravity.Type == gravitySouthWest {
189-
left = 0 + offX
192+
if allowOverflow {
193+
minX, maxX = -innerWidth+1, width-1
194+
minY, maxY = -innerHeight+1, height-1
195+
} else {
196+
minX, maxX = 0, width-innerWidth
197+
minY, maxY = 0, height-innerHeight
190198
}
191199

192-
left = maxInt(0, minInt(left, width-cropWidth))
193-
top = maxInt(0, minInt(top, height-cropHeight))
200+
left = maxInt(minX, minInt(left, maxX))
201+
top = maxInt(minY, minInt(top, maxY))
194202

195203
return
196204
}
@@ -221,7 +229,7 @@ func cropImage(img *vipsImage, cropWidth, cropHeight int, gravity *gravityOption
221229
return img.CopyMemory()
222230
}
223231

224-
left, top := calcCrop(imgWidth, imgHeight, cropWidth, cropHeight, gravity)
232+
left, top := calcPosition(imgWidth, imgHeight, cropWidth, cropHeight, gravity, false)
225233
return img.Crop(left, top, cropWidth, cropHeight)
226234
}
227235

@@ -253,7 +261,9 @@ func prepareWatermark(wm *vipsImage, wmData *imageData, opts *watermarkOptions,
253261
return wm.Replicate(imgWidth, imgHeight)
254262
}
255263

256-
return wm.Embed(opts.Gravity, imgWidth, imgHeight, opts.OffsetX, opts.OffsetY, rgbColor{0, 0, 0})
264+
left, top := calcPosition(imgWidth, imgWidth, wm.Width(), wm.Height(), &opts.Gravity, true)
265+
266+
return wm.Embed(imgWidth, imgHeight, left, top, rgbColor{0, 0, 0})
257267
}
258268

259269
func applyWatermark(img *vipsImage, wmData *imageData, opts *watermarkOptions, framesCount int) error {
@@ -440,8 +450,9 @@ func transformImage(ctx context.Context, img *vipsImage, data []byte, po *proces
440450
}
441451
}
442452

443-
if po.Extend && (po.Width > img.Width() || po.Height > img.Height()) {
444-
if err = img.Embed(gravityCenter, po.Width, po.Height, 0, 0, po.Background); err != nil {
453+
if po.Extend.Enabled && (po.Width > img.Width() || po.Height > img.Height()) {
454+
offX, offY := calcPosition(po.Width, po.Height, img.Width(), img.Height(), &po.Extend.Gravity, false)
455+
if err = img.Embed(po.Width, po.Height, offX, offY, po.Background); err != nil {
445456
return err
446457
}
447458
}

processing_options.go

+28-14
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,11 @@ type gravityOptions struct {
8989
X, Y float64
9090
}
9191

92+
type extendOptions struct {
93+
Enabled bool
94+
Gravity gravityOptions
95+
}
96+
9297
type cropOptions struct {
9398
Width int
9499
Height int
@@ -99,9 +104,7 @@ type watermarkOptions struct {
99104
Enabled bool
100105
Opacity float64
101106
Replicate bool
102-
Gravity gravityType
103-
OffsetX int
104-
OffsetY int
107+
Gravity gravityOptions
105108
Scale float64
106109
}
107110

@@ -112,7 +115,7 @@ type processingOptions struct {
112115
Dpr float64
113116
Gravity gravityOptions
114117
Enlarge bool
115-
Extend bool
118+
Extend extendOptions
116119
Crop cropOptions
117120
Format imageType
118121
Quality int
@@ -194,14 +197,15 @@ func newProcessingOptions() *processingOptions {
194197
Height: 0,
195198
Gravity: gravityOptions{Type: gravityCenter},
196199
Enlarge: false,
200+
Extend: extendOptions{Enabled: false, Gravity: gravityOptions{Type: gravityCenter}},
197201
Quality: conf.Quality,
198202
MaxBytes: 0,
199203
Format: imageTypeUnknown,
200204
Background: rgbColor{255, 255, 255},
201205
Blur: 0,
202206
Sharpen: 0,
203207
Dpr: 1,
204-
Watermark: watermarkOptions{Opacity: 1, Replicate: false, Gravity: gravityCenter},
208+
Watermark: watermarkOptions{Opacity: 1, Replicate: false, Gravity: gravityOptions{Type: gravityCenter}},
205209
}
206210
})
207211

@@ -416,17 +420,27 @@ func applyEnlargeOption(po *processingOptions, args []string) error {
416420
}
417421

418422
func applyExtendOption(po *processingOptions, args []string) error {
419-
if len(args) > 1 {
423+
if len(args) > 4 {
420424
return fmt.Errorf("Invalid extend arguments: %v", args)
421425
}
422426

423-
po.Extend = parseBoolOption(args[0])
427+
po.Extend.Enabled = parseBoolOption(args[0])
428+
429+
if len(args) > 1 {
430+
if err := parseGravity(&po.Extend.Gravity, args[1:]); err != nil {
431+
return err
432+
}
433+
434+
if po.Extend.Gravity.Type == gravitySmart {
435+
return errors.New("extend doesn't support smart gravity")
436+
}
437+
}
424438

425439
return nil
426440
}
427441

428442
func applySizeOption(po *processingOptions, args []string) (err error) {
429-
if len(args) > 4 {
443+
if len(args) > 7 {
430444
return fmt.Errorf("Invalid size arguments: %v", args)
431445
}
432446

@@ -448,8 +462,8 @@ func applySizeOption(po *processingOptions, args []string) (err error) {
448462
}
449463
}
450464

451-
if len(args) == 4 && len(args[3]) > 0 {
452-
if err = applyExtendOption(po, args[3:4]); err != nil {
465+
if len(args) >= 4 && len(args[3]) > 0 {
466+
if err = applyExtendOption(po, args[3:]); err != nil {
453467
return
454468
}
455469
}
@@ -472,7 +486,7 @@ func applyResizingTypeOption(po *processingOptions, args []string) error {
472486
}
473487

474488
func applyResizeOption(po *processingOptions, args []string) error {
475-
if len(args) > 5 {
489+
if len(args) > 8 {
476490
return fmt.Errorf("Invalid resize arguments: %v", args)
477491
}
478492

@@ -664,23 +678,23 @@ func applyWatermarkOption(po *processingOptions, args []string) error {
664678
if args[1] == "re" {
665679
po.Watermark.Replicate = true
666680
} else if g, ok := gravityTypes[args[1]]; ok && g != gravityFocusPoint && g != gravitySmart {
667-
po.Watermark.Gravity = g
681+
po.Watermark.Gravity.Type = g
668682
} else {
669683
return fmt.Errorf("Invalid watermark position: %s", args[1])
670684
}
671685
}
672686

673687
if len(args) > 2 && len(args[2]) > 0 {
674688
if x, err := strconv.Atoi(args[2]); err == nil {
675-
po.Watermark.OffsetX = x
689+
po.Watermark.Gravity.X = float64(x)
676690
} else {
677691
return fmt.Errorf("Invalid watermark X offset: %s", args[2])
678692
}
679693
}
680694

681695
if len(args) > 3 && len(args[3]) > 0 {
682696
if y, err := strconv.Atoi(args[3]); err == nil {
683-
po.Watermark.OffsetY = y
697+
po.Watermark.Gravity.Y = float64(y)
684698
} else {
685699
return fmt.Errorf("Invalid watermark Y offset: %s", args[3])
686700
}

processing_options_test.go

+16-3
Original file line numberDiff line numberDiff line change
@@ -196,6 +196,19 @@ func (s *ProcessingOptionsTestSuite) TestParsePathAdvancedEnlarge() {
196196
assert.True(s.T(), po.Enlarge)
197197
}
198198

199+
func (s *ProcessingOptionsTestSuite) TestParsePathAdvancedExtend() {
200+
req := s.getRequest("http://example.com/unsafe/extend:1:so:10:20/plain/http://images.dev/lorem/ipsum.jpg")
201+
ctx, err := parsePath(context.Background(), req)
202+
203+
require.Nil(s.T(), err)
204+
205+
po := getProcessingOptions(ctx)
206+
assert.Equal(s.T(), true, po.Extend.Enabled)
207+
assert.Equal(s.T(), gravitySouth, po.Extend.Gravity.Type)
208+
assert.Equal(s.T(), 10.0, po.Extend.Gravity.X)
209+
assert.Equal(s.T(), 20.0, po.Extend.Gravity.Y)
210+
}
211+
199212
func (s *ProcessingOptionsTestSuite) TestParsePathAdvancedGravity() {
200213
req := s.getRequest("http://example.com/unsafe/gravity:soea/plain/http://images.dev/lorem/ipsum.jpg")
201214
ctx, err := parsePath(context.Background(), req)
@@ -300,9 +313,9 @@ func (s *ProcessingOptionsTestSuite) TestParsePathAdvancedWatermark() {
300313

301314
po := getProcessingOptions(ctx)
302315
assert.True(s.T(), po.Watermark.Enabled)
303-
assert.Equal(s.T(), gravitySouthEast, po.Watermark.Gravity)
304-
assert.Equal(s.T(), 10, po.Watermark.OffsetX)
305-
assert.Equal(s.T(), 20, po.Watermark.OffsetY)
316+
assert.Equal(s.T(), gravitySouthEast, po.Watermark.Gravity.Type)
317+
assert.Equal(s.T(), 10.0, po.Watermark.Gravity.X)
318+
assert.Equal(s.T(), 20.0, po.Watermark.Gravity.Y)
306319
assert.Equal(s.T(), 0.6, po.Watermark.Scale)
307320
}
308321

vips.go

+2-36
Original file line numberDiff line numberDiff line change
@@ -489,41 +489,7 @@ func (img *vipsImage) Replicate(width, height int) error {
489489
return nil
490490
}
491491

492-
func (img *vipsImage) Embed(gravity gravityType, width, height int, offX, offY int, bg rgbColor) error {
493-
wmWidth := img.Width()
494-
wmHeight := img.Height()
495-
496-
left := (width-wmWidth+1)/2 + offX
497-
top := (height-wmHeight+1)/2 + offY
498-
499-
if gravity == gravityNorth || gravity == gravityNorthEast || gravity == gravityNorthWest {
500-
top = offY
501-
}
502-
503-
if gravity == gravityEast || gravity == gravityNorthEast || gravity == gravitySouthEast {
504-
left = width - wmWidth - offX
505-
}
506-
507-
if gravity == gravitySouth || gravity == gravitySouthEast || gravity == gravitySouthWest {
508-
top = height - wmHeight - offY
509-
}
510-
511-
if gravity == gravityWest || gravity == gravityNorthWest || gravity == gravitySouthWest {
512-
left = offX
513-
}
514-
515-
if left > width {
516-
left = width - wmWidth
517-
} else if left < -wmWidth {
518-
left = 0
519-
}
520-
521-
if top > height {
522-
top = height - wmHeight
523-
} else if top < -wmHeight {
524-
top = 0
525-
}
526-
492+
func (img *vipsImage) Embed(width, height int, offX, offY int, bg rgbColor) error {
527493
if err := img.RgbColourspace(); err != nil {
528494
return err
529495
}
@@ -536,7 +502,7 @@ func (img *vipsImage) Embed(gravity gravityType, width, height int, offX, offY i
536502
}
537503

538504
var tmp *C.VipsImage
539-
if C.vips_embed_go(img.VipsImage, &tmp, C.int(left), C.int(top), C.int(width), C.int(height), &bgc[0], C.int(len(bgc))) != 0 {
505+
if C.vips_embed_go(img.VipsImage, &tmp, C.int(offX), C.int(offY), C.int(width), C.int(height), &bgc[0], C.int(len(bgc))) != 0 {
540506
return vipsError()
541507
}
542508
C.swap_and_clear(&img.VipsImage, tmp)

0 commit comments

Comments
 (0)