Skip to content

Commit

Permalink
Update ColorPropConverterto support color function values (facebook#4…
Browse files Browse the repository at this point in the history
…4237)

Summary:

This adds support for color function values to ColorPropConverter per the wide gamut color [RFC](react-native-community/discussions-and-proposals#738). It updates the color conversion code so that it returns a Color instance before ultimately being converted to an Integer in preparation for returning long values as needed.

## Changelog:

[ANDROID] [ADDED] - Update ColorPropConverter to support color function values


Test Plan:
Colors should work exactly the same as before.

Follow test steps from facebook#42831 to test support for color() function syntax.

While colors specified with color() function syntax will not yet render in DisplayP3 color space they will not be misrecognized as resource path colors but will instead fallback to their sRGB color space values.

---

After the failure with the tests, I reapplied the changes and test some Jest e2e tests that were failing yesterday:

{F1495277376}

Differential Revision: D56517579

Pulled By: cipolleschi
  • Loading branch information
ryanlntn authored and facebook-github-bot committed Apr 24, 2024
1 parent 849da21 commit 207af45
Show file tree
Hide file tree
Showing 2 changed files with 100 additions and 12 deletions.
1 change: 1 addition & 0 deletions packages/react-native/ReactAndroid/api/ReactAndroid.api
Original file line number Diff line number Diff line change
Expand Up @@ -649,6 +649,7 @@ public class com/facebook/react/bridge/ColorPropConverter {
public fun <init> ()V
public static fun getColor (Ljava/lang/Object;Landroid/content/Context;)Ljava/lang/Integer;
public static fun getColor (Ljava/lang/Object;Landroid/content/Context;I)Ljava/lang/Integer;
public static fun getColorInstance (Ljava/lang/Object;Landroid/content/Context;)Landroid/graphics/Color;
public static fun resolveResourcePath (Landroid/content/Context;Ljava/lang/String;)Ljava/lang/Integer;
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,13 +9,22 @@

import android.content.Context;
import android.content.res.Resources;
import android.graphics.Color;
import android.graphics.ColorSpace;
import android.os.Build;
import android.util.TypedValue;
import androidx.annotation.ColorLong;
import androidx.annotation.Nullable;
import androidx.core.content.res.ResourcesCompat;
import com.facebook.common.logging.FLog;
import com.facebook.react.common.ReactConstants;

public class ColorPropConverter {

private static Boolean supportWideGamut() {
return Build.VERSION.SDK_INT >= Build.VERSION_CODES.O;
}

private static final String JSON_KEY = "resource_paths";
private static final String PREFIX_RESOURCE = "@";
private static final String PREFIX_ATTR = "?";
Expand All @@ -24,7 +33,7 @@ public class ColorPropConverter {
private static final String ATTR = "attr";
private static final String ATTR_SEGMENT = "attr/";

public static Integer getColor(Object value, Context context) {
private static Integer getColorInteger(Object value, Context context) {
if (value == null) {
return null;
}
Expand All @@ -33,18 +42,13 @@ public static Integer getColor(Object value, Context context) {
return ((Double) value).intValue();
}

if (context == null) {
throw new RuntimeException("Context may not be null.");
}
throwIfNullContext(context);

if (value instanceof ReadableMap) {
ReadableMap map = (ReadableMap) value;
ReadableArray resourcePaths = map.getArray(JSON_KEY);

if (resourcePaths == null) {
throw new JSApplicationCausedNativeException(
"ColorValue: The `" + JSON_KEY + "` must be an array of color resource path strings.");
}
throwIfNullResourcePaths(resourcePaths);

for (int i = 0; i < resourcePaths.size(); i++) {
Integer result = resolveResourcePath(context, resourcePaths.getString(i));
Expand All @@ -53,16 +57,62 @@ public static Integer getColor(Object value, Context context) {
}
}

throw new JSApplicationCausedNativeException(
"ColorValue: None of the paths in the `"
+ JSON_KEY
+ "` array resolved to a color resource.");
throwColorResourceNotFound();
}

throw new JSApplicationCausedNativeException(
"ColorValue: the value must be a number or Object.");
}

public static Color getColorInstance(Object value, Context context) {
if (value == null) {
return null;
}

if (supportWideGamut() && value instanceof Double) {
return Color.valueOf(((Double) value).intValue());
}

throwIfNullContext(context);

if (value instanceof ReadableMap) {
ReadableMap map = (ReadableMap) value;

Color wideGamutColor = extractWideGamutColorIfPossible(map);
if (wideGamutColor != null) {
return wideGamutColor;
}

ReadableArray resourcePaths = map.getArray(JSON_KEY);
throwIfNullResourcePaths(resourcePaths);

for (int i = 0; i < resourcePaths.size(); i++) {
Integer result = resolveResourcePath(context, resourcePaths.getString(i));
if (supportWideGamut() && result != null) {
return Color.valueOf(result);
}
}

throwColorResourceNotFound();
}
throw new JSApplicationCausedNativeException(
"ColorValue: the value must be a number or Object.");
}

public static Integer getColor(Object value, Context context) {
try {
if (supportWideGamut()) {
Color color = getColorInstance(value, context);
if (color != null) {
return color.toArgb();
}
}
} catch (JSApplicationCausedNativeException ex) {
FLog.w(ReactConstants.TAG, ex, "Error extracting color from WideGamut");
}
return getColorInteger(value, context);
}

public static Integer getColor(Object value, Context context, int defaultInt) {
try {
return getColor(value, context);
Expand Down Expand Up @@ -139,4 +189,41 @@ private static int resolveThemeAttribute(Context context, String resourcePath) {

throw new Resources.NotFoundException();
}

private static void throwIfNullContext(Context context) {
if (context == null) {
throw new RuntimeException("Context may not be null.");
}
}

private static void throwIfNullResourcePaths(ReadableArray resourcePaths) {
if (resourcePaths == null) {
throw new JSApplicationCausedNativeException(
"ColorValue: The `" + JSON_KEY + "` must be an array of color resource path strings.");
}
}

private static void throwColorResourceNotFound() {
throw new JSApplicationCausedNativeException(
"ColorValue: None of the paths in the `"
+ JSON_KEY
+ "` array resolved to a color resource.");
}

private static Color extractWideGamutColorIfPossible(ReadableMap map) {
if (supportWideGamut() && map.hasKey("space")) {
String rawColorSpace = map.getString("space");
boolean isDisplayP3 = rawColorSpace.equals("display-p3");
ColorSpace space =
ColorSpace.get(isDisplayP3 ? ColorSpace.Named.DISPLAY_P3 : ColorSpace.Named.SRGB);
float r = (float) map.getDouble("r");
float g = (float) map.getDouble("g");
float b = (float) map.getDouble("b");
float a = (float) map.getDouble("a");

@ColorLong long color = Color.pack(r, g, b, a, space);
return Color.valueOf(color);
}
return null;
}
}

0 comments on commit 207af45

Please sign in to comment.