Skip to content

Commit 0e5980f

Browse files
committed
[added] 'elementType' custom prop type validator
The code has been borrowed from the https://github.com/jquense/react-widgets project
1 parent 4f0ed4f commit 0e5980f

File tree

2 files changed

+64
-0
lines changed

2 files changed

+64
-0
lines changed

src/utils/CustomPropTypes.js

+36
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,5 @@
1+
import React from 'react';
2+
13
const ANONYMOUS = '<<anonymous>>';
24

35
const CustomPropTypes = {
@@ -15,6 +17,20 @@ const CustomPropTypes = {
1517
*/
1618
mountable: createMountableChecker(),
1719

20+
/**
21+
* Checks whether a prop provides a type of element.
22+
*
23+
* The type of element can be provided in two forms:
24+
* - tag name (string)
25+
* - a return value of React.createClass(...)
26+
*
27+
* @param props
28+
* @param propName
29+
* @param componentName
30+
* @returns {Error|undefined}
31+
*/
32+
elementType: createElementTypeChecker(),
33+
1834
/**
1935
* Checks whether a prop matches a key of an associated object
2036
*
@@ -135,4 +151,24 @@ function all(propTypes) {
135151
};
136152
}
137153

154+
function createElementTypeChecker() {
155+
function validate(props, propName, componentName) {
156+
let errMsg = `Invalid prop '${propName}' specified in '${componentName}'.` +
157+
' Expected an Element `type`';
158+
159+
if (typeof props[propName] !== 'function') {
160+
if (React.isValidElement(props[propName])) {
161+
return new Error(errMsg + ', not an actual Element');
162+
}
163+
164+
if (typeof props[propName] !== 'string') {
165+
return new Error(errMsg +
166+
' such as a tag name or return value of React.createClass(...)');
167+
}
168+
}
169+
}
170+
171+
return createChainableTypeChecker(validate);
172+
}
173+
138174
export default CustomPropTypes;

test/utils/CustomPropTypesSpec.js

+28
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,34 @@ describe('CustomPropTypes', function() {
2626
});
2727
});
2828

29+
describe('elementType', function () {
30+
function validate(prop) {
31+
return CustomPropTypes.elementType({p: prop}, 'p', 'TestComponent');
32+
}
33+
34+
it('Should validate OK with undifined or null values', function() {
35+
assert.isUndefined(validate());
36+
assert.isUndefined(validate(null));
37+
});
38+
39+
it('Should validate OK with elementType values', function() {
40+
assert.isUndefined(validate('span'));
41+
assert.isUndefined(validate(function(){}));
42+
});
43+
44+
it('Should return error with not a string or function values', function() {
45+
let err = validate({});
46+
assert.instanceOf(err, Error);
47+
assert.include(err.message, 'Expected an Element `type` such as a tag name or return value of React.createClass(...)');
48+
});
49+
50+
it('Should return error with react element', function() {
51+
let err = validate(React.createElement('span'));
52+
assert.instanceOf(err, Error);
53+
assert.include(err.message, 'Expected an Element `type`, not an actual Element');
54+
});
55+
});
56+
2957
describe('keyOf', function () {
3058
let obj = {'foo': 1};
3159
function validate(prop) {

0 commit comments

Comments
 (0)