Skip to content

Commit

Permalink
fix: Unmultiply before brightening (#3407)
Browse files Browse the repository at this point in the history
The image data read back is in a pre-multiplied state. Convert it back
to straight alpha before brightening and then premultiply.

This just makes the colors go to white though:

![image](https://github.com/user-attachments/assets/1b38f1ed-85ec-4ac5-abc4-4ba7c5392173)

You could alternatively use the HSV color space and increase the Value
(brightness); this would be more subtle and keep the colors correct.


![image](https://github.com/user-attachments/assets/a87780ab-6698-46cc-aad6-ca37baad229c)
  • Loading branch information
jtmcdole authored Dec 16, 2024
1 parent bb734f4 commit 2d6c8c1
Showing 1 changed file with 67 additions and 22 deletions.
89 changes: 67 additions & 22 deletions packages/flame/lib/src/extensions/image.dart
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import 'dart:ui';

import 'package:flame/extensions.dart';
import 'package:flame/palette.dart';
import 'package:flutter/painting.dart';

export 'dart:ui' show Image;

Expand Down Expand Up @@ -53,17 +54,39 @@ extension ImageExtension on Image {
final newPixelData = Uint8List(pixelData.length);

for (var i = 0; i < pixelData.length; i += 4) {
final color = Color.fromARGB(
pixelData[i + 3],
pixelData[i + 0],
pixelData[i + 1],
pixelData[i + 2],
).darken(amount);

newPixelData[i] = (color.r * 255).round();
newPixelData[i + 1] = (color.g * 255).round();
newPixelData[i + 2] = (color.b * 255).round();
newPixelData[i + 3] = (color.a * 255).round();
final a = pixelData[i + 3] / 255;

// Lets avoid division by zero.
if (a == 0) {
newPixelData[i + 0] = pixelData[i + 0];
newPixelData[i + 1] = pixelData[i + 1];
newPixelData[i + 2] = pixelData[i + 2];
newPixelData[i + 3] = pixelData[i + 3];
continue;
}

// Reverse premultiplied alpha.
var r = (pixelData[i + 0] / 255) / a;
var g = (pixelData[i + 1] / 255) / a;
var b = (pixelData[i + 2] / 255) / a;

// Convert to HSL (Hue, Saturation, and Lightness).
var hsl =
HSLColor.fromColor(Color.from(alpha: a, red: r, green: g, blue: b));
hsl = hsl.withLightness(
clampDouble(hsl.lightness * amount, 0, 1.0),
);

final color = hsl.toColor();
r = color.r;
g = color.g;
b = color.b;

// Premultiply the new color.
newPixelData[i + 0] = (r * a * 255).round();
newPixelData[i + 1] = (g * a * 255).round();
newPixelData[i + 2] = (b * a * 255).round();
newPixelData[i + 3] = pixelData[i + 3];
}
return fromPixels(newPixelData, width, height);
}
Expand All @@ -78,17 +101,39 @@ extension ImageExtension on Image {
final newPixelData = Uint8List(pixelData.length);

for (var i = 0; i < pixelData.length; i += 4) {
final color = Color.fromARGB(
pixelData[i + 3],
pixelData[i + 0],
pixelData[i + 1],
pixelData[i + 2],
).brighten(amount);

newPixelData[i] = (color.r * 255).round();
newPixelData[i + 1] = (color.g * 255).round();
newPixelData[i + 2] = (color.b * 255).round();
newPixelData[i + 3] = (color.a * 255).round();
final a = pixelData[i + 3] / 255;

// Lets avoid division by zero.
if (a == 0) {
newPixelData[i + 0] = pixelData[i + 0];
newPixelData[i + 1] = pixelData[i + 1];
newPixelData[i + 2] = pixelData[i + 2];
newPixelData[i + 3] = pixelData[i + 3];
continue;
}

// Reverse premultiplied alpha.
var r = (pixelData[i + 0] / 255) / a;
var g = (pixelData[i + 1] / 255) / a;
var b = (pixelData[i + 2] / 255) / a;

// Convert to HSL (Hue, Saturation, and Lightness).
var hsl =
HSLColor.fromColor(Color.from(alpha: a, red: r, green: g, blue: b));
hsl = hsl.withLightness(
clampDouble(hsl.lightness + (1 - hsl.lightness) * amount, 0, 1.0),
);

final color = hsl.toColor();
r = color.r;
g = color.g;
b = color.b;

// Premultiply the new color.
newPixelData[i + 0] = (r * a * 255).round();
newPixelData[i + 1] = (g * a * 255).round();
newPixelData[i + 2] = (b * a * 255).round();
newPixelData[i + 3] = pixelData[i + 3];
}
return fromPixels(newPixelData, width, height);
}
Expand Down

0 comments on commit 2d6c8c1

Please sign in to comment.