Skip to content

Commit 2b938fb

Browse files
committed
Release 0.3.0
1 parent f901ef3 commit 2b938fb

File tree

6 files changed

+116
-86
lines changed

6 files changed

+116
-86
lines changed

CHANGELOG.md

+23
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,29 @@ The format is based on [Keep a Changelog](http://keepachangelog.com/) and this p
66

77
---
88

9+
## [0.3.0](https://github.com/FortAwesome/react-native-fontawesome/releases/tag/0.3.0) - 2022-06-07
10+
11+
**This release has a couple of breaking changes.**
12+
13+
1. Minimum supported version of React Native is 0.67
14+
1. Minimum supported react-native-svg is 11.x
15+
1. Using `width` or `height` props are not allowed (they were deprecated in 0.2.x)
16+
1. The `secondaryOpacity` will default to 40% (0.4) instead of 100% to match other Font Awesome implementations
17+
18+
### Added
19+
20+
- Support for specifying icons as strings like `icon="fa-solid fa-mug-empty"`
21+
- Optional testId to TypeScript .d.ts file
22+
- Prop `maskId` allows Jest snapshot testing to have consistent results when using masks
23+
24+
### Fixed
25+
26+
- Full support for version 6 of Font Awesome
27+
- Using icons with masks should now be fully functional
28+
- Duotone icons have also been fixed
29+
30+
---
31+
932
## [0.2.7](https://github.com/FortAwesome/react-native-fontawesome/releases/tag/0.2.7) - 2021-07-22
1033

1134
### Changed

DEVELOPMENT.md

+7-7
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,7 @@ to get your environment set up in your OS. For Mac OS X, that would look like:
2323

2424
## Launch the Example App
2525

26-
In the `examples/Hello` subdirectory, the following script commands are available:
26+
In the `examples/react-native-expo` subdirectory, the following script commands are available:
2727

2828
| Command | Purpose |
2929
| ------- | ------------------------------------- |
@@ -34,15 +34,15 @@ In the `examples/Hello` subdirectory, the following script commands are availabl
3434
| clean | clean out build cache. Useful when the build isn't working and you're pretty sure it should be |
3535

3636
In one terminal tab:
37-
1. `cd examples/Hello`
37+
1. `cd examples/react-native-expo`
3838
1. `npm install`
3939
1. `npm run start`
4040

4141
This will get the JavaScript bundler running and listening for connections from a device or iOS Simulator.
4242

4343
In another terminal tab:
44-
1 `cd examples/Hello`
45-
1. `react-native link react-native-svg` # to link the native components in the ios project
44+
1 `cd examples/react-native-expo`
45+
1. `react-native link react-native-svg` # to link the native components in the ios project
4646
1. `npm run ios`
4747

4848
This will build the project via XCode, launch the iOS Simulator, and when the project builds successfully,
@@ -56,7 +56,7 @@ Seems like things don't always go smoothly and you have to [use some hackery](ht
5656
Try this:
5757
1. shutdown any instances of the bundler you have running.
5858
1. `npm run clean`
59-
1. `npm run start-with-cache-reset` # from the examples/Hello directory
59+
1. `npm run start-with-cache-reset` # from the examples/react-native-expo directory
6060

6161
From another terminal tab:
6262
1. `npm run ios`
@@ -67,13 +67,13 @@ For now, we're just using the GitHub repo as our source for development versions
6767
test changes to the component using the example app, you'll have to push the component changes to a development branch
6868
and then update the `package.json` of the example app to pull the component from that branch.
6969

70-
Here's the step-by-step:
70+
Here's the step-by-step:
7171

7272
1. make changes to this component
7373
1. `npm run dist` # to transpile via babel into `dist/`
7474
1. `git commit` # whatever changes you're trying to commit
7575
1. `git push origin my-dev` # to whatever topic branch you're working on, say "my-dev"
76-
1. Modify `examples/Hello/package.json` and find the line that looks like this:
76+
1. Modify `examples/react-native-expo/package.json` and find the line that looks like this:
7777
`"@fortawesome/react-native-fontawesome": "^0.0.1"`
7878
And change it to something like this:
7979
`"@fortawesome/react-native-fontawesome": "https://github.com/FortAwesome/react-native-fontawesome#my-dev"`

README.md

+2-2
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@
66

77
[![npm](https://img.shields.io/npm/v/@fortawesome/react-native-fontawesome.svg?style=flat-square)](https://www.npmjs.com/package/@fortawesome/react-native-fontawesome)
88

9-
> Font Awesome 5 React Native component using SVG with JS
9+
> Font Awesome React Native component using SVG with JS
1010
1111
<!-- toc -->
1212

@@ -126,7 +126,7 @@ You can use Font Awesome icons in your React Native components as simply as this
126126
That simple usage is made possible when you add the `"mug-saucer"` icon, to the
127127
_library_.
128128

129-
This is one of the two ways you can use Font Awesome 5 with React Native. We'll
129+
This is one of the two ways you can use Font Awesome with React Native. We'll
130130
summarize both ways briefly and then get into the details of each below.
131131

132132
1. **Explicit Import**

dist/components/FontAwesomeIcon.js

+65-48
Original file line numberDiff line numberDiff line change
@@ -3,8 +3,8 @@
33
Object.defineProperty(exports, "__esModule", {
44
value: true
55
});
6+
exports.DEFAULT_SIZE = exports.DEFAULT_SECONDARY_OPACITY = exports.DEFAULT_COLOR = void 0;
67
exports["default"] = FontAwesomeIcon;
7-
exports.DEFAULT_SECONDARY_OPACITY = exports.DEFAULT_COLOR = exports.DEFAULT_SIZE = void 0;
88

99
var _react = _interopRequireDefault(require("react"));
1010

@@ -18,45 +18,44 @@ var _fontawesomeSvgCore = require("@fortawesome/fontawesome-svg-core");
1818

1919
var _logger = _interopRequireDefault(require("../logger"));
2020

21+
var _excluded = ["color"];
22+
2123
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { "default": obj }; }
2224

2325
function _objectWithoutProperties(source, excluded) { if (source == null) return {}; var target = _objectWithoutPropertiesLoose(source, excluded); var key, i; if (Object.getOwnPropertySymbols) { var sourceSymbolKeys = Object.getOwnPropertySymbols(source); for (i = 0; i < sourceSymbolKeys.length; i++) { key = sourceSymbolKeys[i]; if (excluded.indexOf(key) >= 0) continue; if (!Object.prototype.propertyIsEnumerable.call(source, key)) continue; target[key] = source[key]; } } return target; }
2426

2527
function _objectWithoutPropertiesLoose(source, excluded) { if (source == null) return {}; var target = {}; var sourceKeys = Object.keys(source); var key, i; for (i = 0; i < sourceKeys.length; i++) { key = sourceKeys[i]; if (excluded.indexOf(key) >= 0) continue; target[key] = source[key]; } return target; }
2628

27-
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); if (enumerableOnly) symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; }); keys.push.apply(keys, symbols); } return keys; }
29+
function ownKeys(object, enumerableOnly) { var keys = Object.keys(object); if (Object.getOwnPropertySymbols) { var symbols = Object.getOwnPropertySymbols(object); enumerableOnly && (symbols = symbols.filter(function (sym) { return Object.getOwnPropertyDescriptor(object, sym).enumerable; })), keys.push.apply(keys, symbols); } return keys; }
2830

29-
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i] != null ? arguments[i] : {}; if (i % 2) { ownKeys(Object(source), true).forEach(function (key) { _defineProperty(target, key, source[key]); }); } else if (Object.getOwnPropertyDescriptors) { Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)); } else { ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } } return target; }
31+
function _objectSpread(target) { for (var i = 1; i < arguments.length; i++) { var source = null != arguments[i] ? arguments[i] : {}; i % 2 ? ownKeys(Object(source), !0).forEach(function (key) { _defineProperty(target, key, source[key]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(target, Object.getOwnPropertyDescriptors(source)) : ownKeys(Object(source)).forEach(function (key) { Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key)); }); } return target; }
3032

31-
function _typeof(obj) { "@babel/helpers - typeof"; if (typeof Symbol === "function" && typeof Symbol.iterator === "symbol") { _typeof = function _typeof(obj) { return typeof obj; }; } else { _typeof = function _typeof(obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }; } return _typeof(obj); }
33+
function _typeof(obj) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (obj) { return typeof obj; } : function (obj) { return obj && "function" == typeof Symbol && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; }, _typeof(obj); }
3234

3335
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
3436

35-
var _Dimensions$get = _reactNative.Dimensions.get('window'),
36-
windowWidth = _Dimensions$get.width,
37-
windowHeight = _Dimensions$get.height;
38-
3937
var DEFAULT_SIZE = 16;
4038
exports.DEFAULT_SIZE = DEFAULT_SIZE;
4139
var DEFAULT_COLOR = '#000';
4240
exports.DEFAULT_COLOR = DEFAULT_COLOR;
43-
var DEFAULT_SECONDARY_OPACITY = 0.4; // Deprecated height and width defaults
44-
41+
var DEFAULT_SECONDARY_OPACITY = 0.4;
4542
exports.DEFAULT_SECONDARY_OPACITY = DEFAULT_SECONDARY_OPACITY;
46-
var DEFAULT_HEIGHT = windowHeight * 0.1;
47-
var DEFAULT_WIDTH = windowWidth * 0.1;
4843

4944
function objectWithKey(key, value) {
5045
return Array.isArray(value) && value.length > 0 || !Array.isArray(value) && value ? _defineProperty({}, key, value) : {};
5146
}
5247

5348
function normalizeIconArgs(icon) {
54-
if (icon === null) {
55-
return null;
49+
if (icon && _typeof(icon) === 'object' && icon.prefix && icon.iconName && icon.icon) {
50+
return icon;
5651
}
5752

58-
if (_typeof(icon) === 'object' && icon.prefix && icon.iconName) {
59-
return icon;
53+
if (_fontawesomeSvgCore.parse.icon) {
54+
return _fontawesomeSvgCore.parse.icon(icon);
55+
}
56+
57+
if (icon === null) {
58+
return null;
6059
}
6160

6261
if (Array.isArray(icon) && icon.length === 2) {
@@ -77,6 +76,7 @@ function normalizeIconArgs(icon) {
7776
function FontAwesomeIcon(props) {
7877
var iconArgs = props.icon,
7978
maskArgs = props.mask,
79+
maskId = props.maskId,
8080
height = props.height,
8181
width = props.width,
8282
size = props.size;
@@ -86,10 +86,12 @@ function FontAwesomeIcon(props) {
8686
var iconLookup = normalizeIconArgs(iconArgs);
8787
var transform = objectWithKey('transform', typeof props.transform === 'string' ? _fontawesomeSvgCore.parse.transform(props.transform) : props.transform);
8888
var mask = objectWithKey('mask', normalizeIconArgs(maskArgs));
89-
var renderedIcon = (0, _fontawesomeSvgCore.icon)(iconLookup, _objectSpread(_objectSpread({}, transform), mask));
89+
var renderedIcon = (0, _fontawesomeSvgCore.icon)(iconLookup, _objectSpread(_objectSpread(_objectSpread({}, transform), mask), {}, {
90+
maskId: maskId
91+
}));
9092

9193
if (!renderedIcon) {
92-
(0, _logger["default"])("ERROR: icon not found for icon = ", iconArgs);
94+
(0, _logger["default"])('ERROR: icon not found for icon = ', iconArgs);
9395
return null;
9496
}
9597

@@ -98,44 +100,29 @@ function FontAwesomeIcon(props) {
98100
var color = props.color || style.color || DEFAULT_COLOR; // This is the color that will be passed to the "fill" prop of the secondary Path element child (in Duotone Icons)
99101
// `null` value will result in using the primary color, at 40% opacity
100102

101-
var secondaryColor = props.secondaryColor || null; // Secondary layer opacity should default to 0.4, unless a specific opacity value or a specific secondary color was given
103+
var secondaryColor = props.secondaryColor || color; // Secondary layer opacity should default to 0.4, unless a specific opacity value or a specific secondary color was given
102104

103-
var secondaryOpacity = props.secondaryOpacity || (secondaryColor ? 1 : DEFAULT_SECONDARY_OPACITY); // To avoid confusion down the line, we'll remove properties from the StyleSheet, like color, that are being overridden
105+
var secondaryOpacity = props.secondaryOpacity || DEFAULT_SECONDARY_OPACITY; // To avoid confusion down the line, we'll remove properties from the StyleSheet, like color, that are being overridden
104106
// or resolved in other ways, to avoid ambiguity as to which inputs cause which outputs in the underlying rendering process.
105107
// In other words, we don't want color (for example) to be specified via two different inputs.
106108

107109
var styleColor = style.color,
108-
modifiedStyle = _objectWithoutProperties(style, ["color"]);
110+
modifiedStyle = _objectWithoutProperties(style, _excluded);
109111

110112
var resolvedHeight, resolvedWidth;
111113

112114
if (height || width) {
113-
if (size) {
114-
console.warn("DEPRECATION: height and width props on ".concat(FontAwesomeIcon.displayName, " have been deprecated. ") + "Since you've also provided a size prop, we'll use it to override the height and width props given. " + "You should probably go ahead and remove the height and width props to avoid confusion and resolve this warning.");
115-
resolvedHeight = resolvedWidth = size;
116-
} else {
117-
console.warn("DEPRECATION: height and width props on ".concat(FontAwesomeIcon.displayName, " have been deprecated. ") + "Use the size prop instead.");
118-
resolvedHeight = height || DEFAULT_HEIGHT;
119-
resolvedWidth = width || DEFAULT_WIDTH;
120-
}
115+
throw new Error("Prop height and width for component ".concat(FontAwesomeIcon.displayName, " have been deprecated. ") + "Use the size prop instead like <".concat(FontAwesomeIcon.displayName, " size={").concat(width, "} />."));
121116
} else {
122117
resolvedHeight = resolvedWidth = size || DEFAULT_SIZE;
123118
}
124119

125-
var extraProps = {
126-
height: resolvedHeight,
127-
width: resolvedWidth,
128-
fill: color,
129-
secondaryFill: secondaryColor,
130-
secondaryOpacity: secondaryOpacity,
131-
style: modifiedStyle
132-
};
133-
Object.keys(props).forEach(function (key) {
134-
if (!FontAwesomeIcon.defaultProps.hasOwnProperty(key)) {
135-
extraProps[key] = props[key];
136-
}
137-
});
138-
return convertCurry(_abstract[0], extraProps);
120+
var rootAttributes = _abstract[0].attributes;
121+
rootAttributes.height = resolvedHeight;
122+
rootAttributes.width = resolvedWidth;
123+
rootAttributes.style = modifiedStyle;
124+
replaceCurrentColor(_abstract[0], color, secondaryColor, secondaryOpacity);
125+
return convertCurry(_abstract[0]);
139126
}
140127

141128
FontAwesomeIcon.displayName = 'FontAwesomeIcon';
@@ -151,20 +138,50 @@ FontAwesomeIcon.propTypes = {
151138
}), _propTypes["default"].array]),
152139
icon: _propTypes["default"].oneOfType([_propTypes["default"].object, _propTypes["default"].array, _propTypes["default"].string]),
153140
mask: _propTypes["default"].oneOfType([_propTypes["default"].object, _propTypes["default"].array, _propTypes["default"].string]),
141+
maskId: _propTypes["default"].string,
154142
transform: _propTypes["default"].oneOfType([_propTypes["default"].string, _propTypes["default"].object])
155143
};
156144
FontAwesomeIcon.defaultProps = {
157145
icon: null,
158146
mask: null,
147+
maskId: null,
159148
transform: null,
160149
style: {},
161150
color: null,
162151
secondaryColor: null,
163152
secondaryOpacity: null,
164-
height: undefined,
165-
width: undefined // Once the deprecation of height and width props is complete, let's put the real default prop value for size here.
166-
// For now, adding it breaks the default/override logic for height/width/size.
167-
153+
size: DEFAULT_SIZE
168154
};
169155

170-
var convertCurry = _converter["default"].bind(null, _react["default"].createElement);
156+
var convertCurry = _converter["default"].bind(null, _react["default"].createElement);
157+
158+
function replaceCurrentColor(obj, primaryColor, secondaryColor, secondaryOpacity) {
159+
obj.children.forEach(function (child, childIndex) {
160+
replaceFill(child, primaryColor, secondaryColor, secondaryOpacity);
161+
162+
if (Object.prototype.hasOwnProperty.call(child, 'attributes')) {
163+
replaceFill(child.attributes, primaryColor, secondaryColor, secondaryOpacity);
164+
}
165+
166+
if (Array.isArray(child.children) && child.children.length > 0) {
167+
replaceCurrentColor(child, primaryColor, secondaryColor, secondaryOpacity);
168+
}
169+
});
170+
}
171+
172+
function replaceFill(obj, primaryColor, secondaryColor, secondaryOpacity) {
173+
if (hasPropertySetToValue(obj, 'fill', 'currentColor')) {
174+
if (hasPropertySetToValue(obj, 'class', 'fa-primary')) {
175+
obj.fill = primaryColor;
176+
} else if (hasPropertySetToValue(obj, 'class', 'fa-secondary')) {
177+
obj.fill = secondaryColor;
178+
obj.fillOpacity = secondaryOpacity;
179+
} else {
180+
obj.fill = primaryColor;
181+
}
182+
}
183+
}
184+
185+
function hasPropertySetToValue(obj, property, value) {
186+
return Object.prototype.hasOwnProperty.call(obj, property) && obj[property] === value;
187+
}

0 commit comments

Comments
 (0)