OpenShot Library | libopenshot  0.7.0
Sharpen.cpp
Go to the documentation of this file.
1 
9 // Copyright (c) 2008-2025 OpenShot Studios, LLC
10 //
11 // SPDX-License-Identifier: LGPL-3.0-or-later
12 
13 
14 #include "Sharpen.h"
15 #include "Exceptions.h"
16 #include <algorithm>
17 #include <cmath>
18 #include <vector>
19 #include <omp.h>
20 
21 using namespace openshot;
22 
23 // Constructor with default keyframes
25  : amount(10.0)
26  , radius(3.0)
27  , threshold(0.0)
28  , mode(0)
29  , channel(1)
30  , mask_mode(SHARPEN_MASK_LIMIT_TO_AREA)
31 {
32  init_effect_details();
33 }
34 
35 // Constructor from keyframes
37  : amount(a)
38  , radius(r)
39  , threshold(t)
40  , mode(0)
41  , channel(1)
42  , mask_mode(SHARPEN_MASK_LIMIT_TO_AREA)
43 {
44  init_effect_details();
45 }
46 
47 // Initialize effect metadata
48 void Sharpen::init_effect_details()
49 {
51  info.class_name = "Sharpen";
52  info.name = "Sharpen";
53  info.description = "Boost edge contrast to make video details look crisper.";
54  info.has_audio = false;
55  info.has_video = true;
56 }
57 
58 bool Sharpen::UseCustomMaskBlend(int64_t frame_number) const
59 {
60  (void) frame_number;
62 }
63 
64 void Sharpen::ApplyCustomMaskBlend(std::shared_ptr<QImage> original_image, std::shared_ptr<QImage> effected_image,
65  std::shared_ptr<QImage> mask_image, int64_t frame_number) const
66 {
67  (void) frame_number;
68  if (!original_image || !effected_image || !mask_image)
69  return;
70  if (original_image->size() != effected_image->size() || effected_image->size() != mask_image->size())
71  return;
72 
73  unsigned char* original_pixels = reinterpret_cast<unsigned char*>(original_image->bits());
74  unsigned char* effected_pixels = reinterpret_cast<unsigned char*>(effected_image->bits());
75  unsigned char* mask_pixels = reinterpret_cast<unsigned char*>(mask_image->bits());
76  const int pixel_count = effected_image->width() * effected_image->height();
77 
78  #pragma omp parallel for schedule(static)
79  for (int i = 0; i < pixel_count; ++i) {
80  const int idx = i * 4;
81  float factor = static_cast<float>(qGray(mask_pixels[idx], mask_pixels[idx + 1], mask_pixels[idx + 2])) / 255.0f;
82  if (mask_invert)
83  factor = 1.0f - factor;
84  factor = factor * factor;
85  const float inverse = 1.0f - factor;
86 
87  effected_pixels[idx] = static_cast<unsigned char>(
88  (original_pixels[idx] * inverse) + (effected_pixels[idx] * factor));
89  effected_pixels[idx + 1] = static_cast<unsigned char>(
90  (original_pixels[idx + 1] * inverse) + (effected_pixels[idx + 1] * factor));
91  effected_pixels[idx + 2] = static_cast<unsigned char>(
92  (original_pixels[idx + 2] * inverse) + (effected_pixels[idx + 2] * factor));
93  effected_pixels[idx + 3] = original_pixels[idx + 3];
94  }
95 }
96 
97 // Compute three box sizes to approximate a Gaussian of sigma
98 static void boxes_for_gauss(double sigma, int b[3])
99 {
100  const int n = 3;
101  double wi = std::sqrt((12.0 * sigma * sigma / n) + 1.0);
102  int wl = int(std::floor(wi));
103  if (!(wl & 1)) --wl;
104  int wu = wl + 2;
105  double mi = (12.0 * sigma * sigma - n*wl*wl - 4.0*n*wl - 3.0*n)
106  / (-4.0*wl - 4.0);
107  int m = int(std::round(mi));
108  for (int i = 0; i < n; ++i)
109  b[i] = i < m ? wl : wu;
110 }
111 
112 // Blur one axis with an edge-replicate sliding window
113 static void blur_axis(const QImage& src, QImage& dst, int r, bool vertical)
114 {
115  if (r <= 0) {
116  dst = src.copy();
117  return;
118  }
119 
120  int W = src.width();
121  int H = src.height();
122  int bpl = src.bytesPerLine();
123  const uchar* in = src.bits();
124  uchar* out = dst.bits();
125  int window = 2*r + 1;
126  const float inv_w = 1.0f / window;
127 
128  if (!vertical) {
129  #pragma omp parallel for
130  for (int y = 0; y < H; ++y) {
131  const uchar* rowIn = in + y*bpl;
132  uchar* rowOut = out + y*bpl;
133  int sB = rowIn[0]*(r+1), sG = rowIn[1]*(r+1), sR = rowIn[2]*(r+1);
134  for (int x = 1; x <= r; ++x) {
135  const uchar* p = rowIn + std::min(x, W-1)*4;
136  sB += p[0]; sG += p[1]; sR += p[2];
137  }
138  for (int x = 0; x < W; ++x) {
139  uchar* o = rowOut + x*4;
140  o[0] = uchar(sB * inv_w + 0.5f);
141  o[1] = uchar(sG * inv_w + 0.5f);
142  o[2] = uchar(sR * inv_w + 0.5f);
143 
144  const uchar* addP = rowIn + std::min(x+r+1, W-1)*4;
145  const uchar* subP = rowIn + std::max(x-r, 0)*4;
146  sB += addP[0] - subP[0];
147  sG += addP[1] - subP[1];
148  sR += addP[2] - subP[2];
149  }
150  }
151  }
152  else {
153  #pragma omp parallel for
154  for (int x = 0; x < W; ++x) {
155  const uchar* p0 = in + x*4;
156  int sB = p0[0]*(r+1), sG = p0[1]*(r+1), sR = p0[2]*(r+1);
157  for (int y = 1; y <= r; ++y) {
158  const uchar* p = in + std::min(y, H-1)*bpl + x*4;
159  sB += p[0]; sG += p[1]; sR += p[2];
160  }
161  for (int y = 0; y < H; ++y) {
162  uchar* o = out + y*bpl + x*4;
163  o[0] = uchar(sB * inv_w + 0.5f);
164  o[1] = uchar(sG * inv_w + 0.5f);
165  o[2] = uchar(sR * inv_w + 0.5f);
166 
167  const uchar* addP = in + std::min(y+r+1, H-1)*bpl + x*4;
168  const uchar* subP = in + std::max(y-r, 0)*bpl + x*4;
169  sB += addP[0] - subP[0];
170  sG += addP[1] - subP[1];
171  sR += addP[2] - subP[2];
172  }
173  }
174  }
175 }
176 
177 // Wrapper to handle fractional radius by blending two integer passes
178 static void box_blur(const QImage& src, QImage& dst, double rf, bool vertical)
179 {
180  int r0 = int(std::floor(rf));
181  int r1 = r0 + 1;
182  double f = rf - r0;
183  if (f < 1e-4) {
184  blur_axis(src, dst, r0, vertical);
185  }
186  else {
187  QImage a(src.size(), QImage::Format_ARGB32);
188  QImage b(src.size(), QImage::Format_ARGB32);
189  blur_axis(src, a, r0, vertical);
190  blur_axis(src, b, r1, vertical);
191 
192  int pixels = src.width() * src.height();
193  const uchar* pa = a.bits();
194  const uchar* pb = b.bits();
195  uchar* pd = dst.bits();
196  const float ff = float(f);
197  const float ff1 = 1.0f - ff;
198  #pragma omp parallel for
199  for (int i = 0; i < pixels; ++i) {
200  for (int c = 0; c < 3; ++c) {
201  pd[i*4+c] = uchar(ff1 * pa[i*4+c] + ff * pb[i*4+c] + 0.5f);
202  }
203  }
204  }
205 }
206 
207 // Apply three sequential box blurs to approximate Gaussian
208 static void gauss_blur(const QImage& src, QImage& dst, double sigma)
209 {
210  int b[3];
211  boxes_for_gauss(sigma, b);
212  QImage t1(src.size(), QImage::Format_ARGB32);
213  QImage t2(src.size(), QImage::Format_ARGB32);
214 
215  double r = 0.5 * (b[0] - 1);
216  box_blur(src , t1, r, false);
217  box_blur(t1, t2, r, true);
218 
219  r = 0.5 * (b[1] - 1);
220  box_blur(t2, t1, r, false);
221  box_blur(t1, t2, r, true);
222 
223  r = 0.5 * (b[2] - 1);
224  box_blur(t2, t1, r, false);
225  box_blur(t1, dst, r, true);
226 }
227 
228 // Main frame processing
229 std::shared_ptr<Frame> Sharpen::GetFrame(
230  std::shared_ptr<Frame> frame, int64_t frame_number)
231 {
232  auto img = frame->GetImage();
233  if (!img || img->isNull())
234  return frame;
235  if (img->format() != QImage::Format_ARGB32)
236  *img = img->convertToFormat(QImage::Format_ARGB32);
237 
238  int W = img->width();
239  int H = img->height();
240  if (W <= 0 || H <= 0)
241  return frame;
242 
243  // Retrieve keyframe values
244  double amt = amount.GetValue(frame_number); // 0–40
245  double rpx = radius.GetValue(frame_number); // px
246  double thrUI = threshold.GetValue(frame_number); // 0–1
247 
248  if (amt == 0.0 || rpx <= 0.0)
249  return frame;
250 
251  // Sigma scaled against 720p reference
252  double sigma = std::max(0.1, rpx * H / 720.0);
253 
254  // Generate blurred image
255  QImage blur(W, H, QImage::Format_ARGB32);
256  gauss_blur(*img, blur, sigma);
257 
258  // Precompute maximum luma difference for adaptive threshold
259  int bplS = img->bytesPerLine();
260  int bplB = blur.bytesPerLine();
261  uchar* sBits = img->bits();
262  uchar* bBits = blur.bits();
263 
264  float maxDY = 0.0f;
265  if (thrUI > 0.0) {
266  #pragma omp parallel for reduction(max:maxDY)
267  for (int y = 0; y < H; ++y) {
268  uchar* sRow = sBits + y * bplS;
269  uchar* bRow = bBits + y * bplB;
270  for (int x = 0; x < W; ++x) {
271  float dB = float(sRow[x*4+0]) - float(bRow[x*4+0]);
272  float dG = float(sRow[x*4+1]) - float(bRow[x*4+1]);
273  float dR = float(sRow[x*4+2]) - float(bRow[x*4+2]);
274  float dY = std::abs(0.114f*dB + 0.587f*dG + 0.299f*dR);
275  maxDY = std::max(maxDY, dY);
276  }
277  }
278  }
279 
280  // Compute actual threshold in luma units
281  const float thr = float(thrUI) * maxDY;
282  const float famt = float(amt);
283  // Process pixels
284  #pragma omp parallel for
285  for (int y = 0; y < H; ++y) {
286  uchar* sRow = sBits + y * bplS;
287  uchar* bRow = bBits + y * bplB;
288  for (int x = 0; x < W; ++x) {
289  uchar* sp = sRow + x*4;
290  uchar* bp = bRow + x*4;
291 
292  // Detail per channel
293  float dB = float(sp[0]) - float(bp[0]);
294  float dG = float(sp[1]) - float(bp[1]);
295  float dR = float(sp[2]) - float(bp[2]);
296  float dY = 0.114f*dB + 0.587f*dG + 0.299f*dR;
297 
298  // Skip if below adaptive threshold
299  if (std::abs(dY) < thr)
300  continue;
301 
302  // Halo limiter
303  auto halo = [](float d) {
304  return (255.0f - std::abs(d)) * (1.0f / 255.0f);
305  };
306 
307  float outC[3];
308 
309  if (mode == 1) {
310  // HighPass: base = blurred image, detail = original – blurred, no halo limiter
311  const float wB = 0.114f, wG = 0.587f, wR = 0.299f;
312 
313  if (channel == 1) {
314  // Luma only: add back luma detail weighted per channel
315  float lumaInc = famt * dY;
316  outC[0] = bp[0] + lumaInc * wB;
317  outC[1] = bp[1] + lumaInc * wG;
318  outC[2] = bp[2] + lumaInc * wR;
319  }
320  else if (channel == 2) {
321  // Chroma only: subtract luma from detail, add chroma back
322  float chromaB = dB - dY * wB;
323  float chromaG = dG - dY * wG;
324  float chromaR = dR - dY * wR;
325  outC[0] = bp[0] + famt * chromaB;
326  outC[1] = bp[1] + famt * chromaG;
327  outC[2] = bp[2] + famt * chromaR;
328  }
329  else {
330  // All channels: add full per-channel detail
331  outC[0] = bp[0] + famt * dB;
332  outC[1] = bp[1] + famt * dG;
333  outC[2] = bp[2] + famt * dR;
334  }
335  }
336  else {
337  // Unsharp-Mask: base = original + amt * detail * halo(detail)
338  if (channel == 1) {
339  // Luma only
340  float inc = famt * dY * halo(dY);
341  for (int c = 0; c < 3; ++c)
342  outC[c] = sp[c] + inc;
343  }
344  else if (channel == 2) {
345  // Chroma only
346  float chroma[3] = { dB - dY, dG - dY, dR - dY };
347  for (int c = 0; c < 3; ++c)
348  outC[c] = sp[c] + famt * chroma[c] * halo(chroma[c]);
349  }
350  else {
351  // All channels
352  outC[0] = sp[0] + famt * dB * halo(dB);
353  outC[1] = sp[1] + famt * dG * halo(dG);
354  outC[2] = sp[2] + famt * dR * halo(dR);
355  }
356  }
357 
358  // Write back clamped
359  for (int c = 0; c < 3; ++c) {
360  sp[c] = uchar(std::clamp(outC[c], 0.0f, 255.0f) + 0.5f);
361  }
362  }
363  }
364 
365  return frame;
366 }
367 
368 // JSON serialization
369 std::string Sharpen::Json() const
370 {
371  return JsonValue().toStyledString();
372 }
373 
374 Json::Value Sharpen::JsonValue() const
375 {
376  Json::Value root = EffectBase::JsonValue();
377  root["type"] = info.class_name;
378  root["amount"] = amount.JsonValue();
379  root["radius"] = radius.JsonValue();
380  root["threshold"] = threshold.JsonValue();
381  root["mode"] = mode;
382  root["channel"] = channel;
383  root["mask_mode"] = mask_mode;
384  return root;
385 }
386 
387 // JSON deserialization
388 void Sharpen::SetJson(std::string value)
389 {
390  auto root = openshot::stringToJson(value);
391  SetJsonValue(root);
392 }
393 
394 void Sharpen::SetJsonValue(Json::Value root)
395 {
397  if (!root["amount"].isNull())
398  amount.SetJsonValue(root["amount"]);
399  if (!root["radius"].isNull())
400  radius.SetJsonValue(root["radius"]);
401  if (!root["threshold"].isNull())
402  threshold.SetJsonValue(root["threshold"]);
403  if (!root["mode"].isNull())
404  mode = root["mode"].asInt();
405  if (!root["channel"].isNull())
406  channel = root["channel"].asInt();
407  if (!root["mask_mode"].isNull())
408  mask_mode = root["mask_mode"].asInt();
409 }
410 
411 // UI property definitions
412 std::string Sharpen::PropertiesJSON(int64_t t) const
413 {
414  Json::Value root = BasePropertiesJSON(t);
415  root["amount"] = add_property_json(
416  "Amount", amount.GetValue(t), "float", "", &amount, 0, 40, false, t);
417  root["radius"] = add_property_json(
418  "Radius", radius.GetValue(t), "float", "pixels", &radius, 0, 10, false, t);
419  root["threshold"] = add_property_json(
420  "Threshold", threshold.GetValue(t), "float", "ratio", &threshold, 0, 1, false, t);
421  root["mode"] = add_property_json(
422  "Mode", mode, "int", "", nullptr, 0, 1, false, t);
423  root["mode"]["choices"].append(add_property_choice_json("UnsharpMask", 0, mode));
424  root["mode"]["choices"].append(add_property_choice_json("HighPassBlend", 1, mode));
425  root["channel"] = add_property_json(
426  "Channel", channel, "int", "", nullptr, 0, 2, false, t);
427  root["channel"]["choices"].append(add_property_choice_json("All", 0, channel));
428  root["channel"]["choices"].append(add_property_choice_json("Luma", 1, channel));
429  root["channel"]["choices"].append(add_property_choice_json("Chroma", 2, channel));
430  root["mask_mode"] = add_property_json(
431  "Mask Mode", mask_mode, "int", "", nullptr, 0, 1, false, t);
432  root["mask_mode"]["choices"].append(add_property_choice_json("Limit to Mask", SHARPEN_MASK_LIMIT_TO_AREA, mask_mode));
433  root["mask_mode"]["choices"].append(add_property_choice_json("Vary Strength", SHARPEN_MASK_VARY_STRENGTH, mask_mode));
434  return root.toStyledString();
435 }
openshot::ClipBase::add_property_json
Json::Value add_property_json(std::string name, float value, std::string type, std::string memo, const Keyframe *keyframe, float min_value, float max_value, bool readonly, int64_t requested_frame) const
Generate JSON for a property.
Definition: ClipBase.cpp:96
openshot::stringToJson
const Json::Value stringToJson(const std::string value)
Definition: Json.cpp:16
openshot::Sharpen::SetJson
void SetJson(const std::string value) override
Load JSON string into this object.
Definition: Sharpen.cpp:388
openshot::Sharpen::mode
int mode
Sharpening mode (0 = UnsharpMask, 1 = HighPassBlend)
Definition: Sharpen.h:57
openshot::SHARPEN_MASK_VARY_STRENGTH
@ SHARPEN_MASK_VARY_STRENGTH
Definition: Sharpen.h:27
openshot::EffectBase::info
EffectInfoStruct info
Information about the current effect.
Definition: EffectBase.h:110
openshot::EffectBase::mask_invert
bool mask_invert
Invert grayscale mask values before blending.
Definition: EffectBase.h:111
openshot
This namespace is the default namespace for all code in the openshot library.
Definition: AnimatedCurve.h:24
openshot::ClipBase::add_property_choice_json
Json::Value add_property_choice_json(std::string name, int value, int selected_value) const
Generate JSON choice for a property (dropdown properties)
Definition: ClipBase.cpp:132
openshot::EffectBase::JsonValue
virtual Json::Value JsonValue() const
Generate Json::Value for this object.
Definition: EffectBase.cpp:96
openshot::Sharpen::mask_mode
int mask_mode
Mask behavior mode for this effect.
Definition: Sharpen.h:63
openshot::Sharpen::PropertiesJSON
std::string PropertiesJSON(int64_t requested_frame) const override
Definition: Sharpen.cpp:412
openshot::Sharpen::Json
std::string Json() const override
Get and Set JSON methods.
Definition: Sharpen.cpp:369
openshot::Sharpen::radius
Keyframe radius
Radius of the blur used in sharpening (0 to 10 pixels for 1080p)
Definition: Sharpen.h:51
openshot::Sharpen::GetFrame
std::shared_ptr< Frame > GetFrame(std::shared_ptr< Frame > frame, int64_t frame_number) override
This method is required for all derived classes of EffectBase, and returns a modified openshot::Frame...
Definition: Sharpen.cpp:229
openshot::Keyframe::SetJsonValue
void SetJsonValue(const Json::Value root)
Load Json::Value into this object.
Definition: KeyFrame.cpp:372
openshot::Sharpen::channel
int channel
Channel to apply sharpening to (0 = All, 1 = Luma, 2 = Chroma)
Definition: Sharpen.h:60
openshot::Keyframe::JsonValue
Json::Value JsonValue() const
Generate Json::Value for this object.
Definition: KeyFrame.cpp:339
openshot::EffectBase::BasePropertiesJSON
Json::Value BasePropertiesJSON(int64_t requested_frame) const
Generate JSON object of base properties (recommended to be used by all effects)
Definition: EffectBase.cpp:236
Sharpen.h
Header file for Sharpen effect class.
openshot::Keyframe
A Keyframe is a collection of Point instances, which is used to vary a number or property over time.
Definition: KeyFrame.h:53
openshot::Sharpen::ApplyCustomMaskBlend
void ApplyCustomMaskBlend(std::shared_ptr< QImage > original_image, std::shared_ptr< QImage > effected_image, std::shared_ptr< QImage > mask_image, int64_t frame_number) const override
Optional override for effects with custom mask implementation.
Definition: Sharpen.cpp:64
openshot::EffectBase::InitEffectInfo
void InitEffectInfo()
Definition: EffectBase.cpp:37
openshot::EffectInfoStruct::has_audio
bool has_audio
Determines if this effect manipulates the audio of a frame.
Definition: EffectBase.h:44
openshot::SHARPEN_MASK_LIMIT_TO_AREA
@ SHARPEN_MASK_LIMIT_TO_AREA
Definition: Sharpen.h:26
openshot::EffectInfoStruct::class_name
std::string class_name
The class name of the effect.
Definition: EffectBase.h:39
openshot::EffectInfoStruct::description
std::string description
The description of this effect and what it does.
Definition: EffectBase.h:41
openshot::Sharpen::JsonValue
Json::Value JsonValue() const override
Generate Json::Value for this object.
Definition: Sharpen.cpp:374
openshot::EffectInfoStruct::has_video
bool has_video
Determines if this effect manipulates the image of a frame.
Definition: EffectBase.h:43
openshot::Sharpen::SetJsonValue
void SetJsonValue(const Json::Value root) override
Load Json::Value into this object.
Definition: Sharpen.cpp:394
openshot::EffectInfoStruct::name
std::string name
The name of the effect.
Definition: EffectBase.h:40
openshot::Sharpen::Sharpen
Sharpen()
Default constructor.
Definition: Sharpen.cpp:24
openshot::Sharpen::threshold
Keyframe threshold
Threshold for applying sharpening (0 to 1)
Definition: Sharpen.h:54
openshot::Sharpen::amount
Keyframe amount
Amount of sharpening to apply (0 to 2)
Definition: Sharpen.h:48
openshot::Sharpen::UseCustomMaskBlend
bool UseCustomMaskBlend(int64_t frame_number) const override
Optional override for effects that need custom mask behavior.
Definition: Sharpen.cpp:58
Exceptions.h
Header file for all Exception classes.
openshot::EffectBase::SetJsonValue
virtual void SetJsonValue(const Json::Value root)
Load Json::Value into this object.
Definition: EffectBase.cpp:139
openshot::Keyframe::GetValue
double GetValue(int64_t index) const
Get the value at a specific index.
Definition: KeyFrame.cpp:258