mirror of
https://github.com/lovell/sharp.git
synced 2025-12-19 07:15:08 +01:00
Add autoOrient operation and constructor option #4144
This commit is contained in:
committed by
Lovell Fuller
parent
b7ff2645c4
commit
14c83e1f4c
@@ -166,6 +166,8 @@ namespace sharp {
|
||||
descriptor->access = AttrAsBool(input, "sequentialRead") ? VIPS_ACCESS_SEQUENTIAL : VIPS_ACCESS_RANDOM;
|
||||
// Remove safety features and allow unlimited input
|
||||
descriptor->unlimited = AttrAsBool(input, "unlimited");
|
||||
// Use the EXIF orientation to auto orient the image
|
||||
descriptor->autoOrient = AttrAsBool(input, "autoOrient");
|
||||
return descriptor;
|
||||
}
|
||||
|
||||
|
||||
@@ -33,6 +33,7 @@ namespace sharp {
|
||||
struct InputDescriptor { // NOLINT(runtime/indentation_namespace)
|
||||
std::string name;
|
||||
std::string file;
|
||||
bool autoOrient;
|
||||
char *buffer;
|
||||
VipsFailOn failOn;
|
||||
uint64_t limitInputPixels;
|
||||
@@ -73,6 +74,7 @@ namespace sharp {
|
||||
std::vector<double> pdfBackground;
|
||||
|
||||
InputDescriptor():
|
||||
autoOrient(false),
|
||||
buffer(nullptr),
|
||||
failOn(VIPS_FAIL_ON_WARNING),
|
||||
limitInputPixels(0x3FFF * 0x3FFF),
|
||||
|
||||
@@ -242,6 +242,15 @@ class MetadataWorker : public Napi::AsyncWorker {
|
||||
if (baton->orientation > 0) {
|
||||
info.Set("orientation", baton->orientation);
|
||||
}
|
||||
Napi::Object autoOrient = Napi::Object::New(env);
|
||||
info.Set("autoOrient", autoOrient);
|
||||
if (baton->orientation >= 5) {
|
||||
autoOrient.Set("width", baton->height);
|
||||
autoOrient.Set("height", baton->width);
|
||||
} else {
|
||||
autoOrient.Set("width", baton->width);
|
||||
autoOrient.Set("height", baton->height);
|
||||
}
|
||||
if (baton->exifLength > 0) {
|
||||
info.Set("exif", Napi::Buffer<char>::NewOrCopy(env, baton->exif, baton->exifLength, sharp::FreeCallback));
|
||||
}
|
||||
|
||||
@@ -63,14 +63,14 @@ class PipelineWorker : public Napi::AsyncWorker {
|
||||
bool autoFlip = false;
|
||||
bool autoFlop = false;
|
||||
|
||||
if (baton->useExifOrientation) {
|
||||
if (baton->input->autoOrient) {
|
||||
// Rotate and flip image according to Exif orientation
|
||||
std::tie(autoRotation, autoFlip, autoFlop) = CalculateExifRotationAndFlip(sharp::ExifOrientation(image));
|
||||
image = sharp::RemoveExifOrientation(image);
|
||||
} else {
|
||||
rotation = CalculateAngleRotation(baton->angle);
|
||||
}
|
||||
|
||||
rotation = CalculateAngleRotation(baton->angle);
|
||||
|
||||
// Rotate pre-extract
|
||||
bool const shouldRotateBefore = baton->rotateBeforePreExtract &&
|
||||
(rotation != VIPS_ANGLE_D0 || autoRotation != VIPS_ANGLE_D0 ||
|
||||
@@ -92,18 +92,14 @@ class PipelineWorker : public Napi::AsyncWorker {
|
||||
image = image.rot(autoRotation);
|
||||
autoRotation = VIPS_ANGLE_D0;
|
||||
}
|
||||
if (autoFlip) {
|
||||
if (autoFlip != baton->flip) {
|
||||
image = image.flip(VIPS_DIRECTION_VERTICAL);
|
||||
autoFlip = false;
|
||||
} else if (baton->flip) {
|
||||
image = image.flip(VIPS_DIRECTION_VERTICAL);
|
||||
baton->flip = false;
|
||||
}
|
||||
if (autoFlop) {
|
||||
if (autoFlop != baton->flop) {
|
||||
image = image.flip(VIPS_DIRECTION_HORIZONTAL);
|
||||
autoFlop = false;
|
||||
} else if (baton->flop) {
|
||||
image = image.flip(VIPS_DIRECTION_HORIZONTAL);
|
||||
baton->flop = false;
|
||||
}
|
||||
if (rotation != VIPS_ANGLE_D0) {
|
||||
@@ -396,11 +392,11 @@ class PipelineWorker : public Napi::AsyncWorker {
|
||||
image = image.rot(autoRotation);
|
||||
}
|
||||
// Mirror vertically (up-down) about the x-axis
|
||||
if (baton->flip || autoFlip) {
|
||||
if (baton->flip != autoFlip) {
|
||||
image = image.flip(VIPS_DIRECTION_VERTICAL);
|
||||
}
|
||||
// Mirror horizontally (left-right) about the y-axis
|
||||
if (baton->flop || autoFlop) {
|
||||
if (baton->flop != autoFlop) {
|
||||
image = image.flip(VIPS_DIRECTION_HORIZONTAL);
|
||||
}
|
||||
// Rotate post-extract 90-angle
|
||||
@@ -631,6 +627,30 @@ class PipelineWorker : public Napi::AsyncWorker {
|
||||
composite->input->access = access;
|
||||
std::tie(compositeImage, compositeImageType) = sharp::OpenInput(composite->input);
|
||||
compositeImage = sharp::EnsureColourspace(compositeImage, baton->colourspacePipeline);
|
||||
|
||||
if (composite->input->autoOrient) {
|
||||
// Respect EXIF Orientation
|
||||
VipsAngle compositeAutoRotation = VIPS_ANGLE_D0;
|
||||
bool compositeAutoFlip = false;
|
||||
bool compositeAutoFlop = false;
|
||||
std::tie(compositeAutoRotation, compositeAutoFlip, compositeAutoFlop) =
|
||||
CalculateExifRotationAndFlip(sharp::ExifOrientation(compositeImage));
|
||||
|
||||
compositeImage = sharp::RemoveExifOrientation(compositeImage);
|
||||
compositeImage = sharp::StaySequential(compositeImage,
|
||||
compositeAutoRotation != VIPS_ANGLE_D0 || compositeAutoFlip);
|
||||
|
||||
if (compositeAutoRotation != VIPS_ANGLE_D0) {
|
||||
compositeImage = compositeImage.rot(compositeAutoRotation);
|
||||
}
|
||||
if (compositeAutoFlip) {
|
||||
compositeImage = compositeImage.flip(VIPS_DIRECTION_VERTICAL);
|
||||
}
|
||||
if (compositeAutoFlop) {
|
||||
compositeImage = compositeImage.flip(VIPS_DIRECTION_HORIZONTAL);
|
||||
}
|
||||
}
|
||||
|
||||
// Verify within current dimensions
|
||||
if (compositeImage.width() > image.width() || compositeImage.height() > image.height()) {
|
||||
throw vips::VError("Image to composite must have same dimensions or smaller");
|
||||
@@ -1567,7 +1587,6 @@ Napi::Value pipeline(const Napi::CallbackInfo& info) {
|
||||
baton->claheWidth = sharp::AttrAsUint32(options, "claheWidth");
|
||||
baton->claheHeight = sharp::AttrAsUint32(options, "claheHeight");
|
||||
baton->claheMaxSlope = sharp::AttrAsUint32(options, "claheMaxSlope");
|
||||
baton->useExifOrientation = sharp::AttrAsBool(options, "useExifOrientation");
|
||||
baton->angle = sharp::AttrAsInt32(options, "angle");
|
||||
baton->rotationAngle = sharp::AttrAsDouble(options, "rotationAngle");
|
||||
baton->rotationBackground = sharp::AttrAsVectorOfDouble(options, "rotationBackground");
|
||||
|
||||
@@ -109,7 +109,6 @@ struct PipelineBaton {
|
||||
int claheWidth;
|
||||
int claheHeight;
|
||||
int claheMaxSlope;
|
||||
bool useExifOrientation;
|
||||
int angle;
|
||||
double rotationAngle;
|
||||
std::vector<double> rotationBackground;
|
||||
@@ -282,7 +281,6 @@ struct PipelineBaton {
|
||||
claheWidth(0),
|
||||
claheHeight(0),
|
||||
claheMaxSlope(3),
|
||||
useExifOrientation(false),
|
||||
angle(0),
|
||||
rotationAngle(0.0),
|
||||
rotationBackground{ 0.0, 0.0, 0.0, 255.0 },
|
||||
|
||||
Reference in New Issue
Block a user