Files
react-native/packages/react-native-codegen/src/generators/GeneratePropsCpp.js
Rick Hanlon 0827184c60 Add PointPrimitive
Summary: Adds support for the native type Point to the rn codegen

Reviewed By: TheSavior

Differential Revision: D14462164

fbshipit-source-id: 942b5697d616c6aa6289d01bb56382fd7adac203
2019-03-19 06:36:33 -07:00

171 lines
4.3 KiB
JavaScript

/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*
* @flow strict
* @format
*/
'use strict';
import type {ComponentShape, SchemaType} from '../CodegenSchema';
// File path -> contents
type FilesOutput = Map<string, string>;
const template = `
/**
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
#include <react/components/::_LIBRARY_::/Props.h>
::_IMPORTS_::
namespace facebook {
namespace react {
::_COMPONENT_CLASSES_::
} // namespace react
} // namespace facebook
`;
const componentTemplate = `
::_CLASSNAME_::::::_CLASSNAME_::(
const ::_CLASSNAME_:: &sourceProps,
const RawProps &rawProps):::_EXTEND_CLASSES_::
::_PROPS_::
{}
`.trim();
function generatePropsString(component: ComponentShape) {
return component.props
.map(prop => {
return `${prop.name}(convertRawProp(rawProps, "${
prop.name
}", sourceProps.${prop.name}, ${prop.name}))`;
})
.join(',\n');
}
function getClassExtendString(component): string {
const extendString =
' ' +
component.extendsProps
.map(extendProps => {
switch (extendProps.type) {
case 'ReactNativeBuiltInType':
switch (extendProps.knownTypeName) {
case 'ReactNativeCoreViewProps':
return 'ViewProps(sourceProps, rawProps)';
default:
(extendProps.knownTypeName: empty);
throw new Error('Invalid knownTypeName');
}
default:
(extendProps.type: empty);
throw new Error('Invalid extended type');
}
})
.join(', ') +
',';
return extendString;
}
function getImports(component): Set<string> {
const imports: Set<string> = new Set();
function addImportsForNativeName(name) {
switch (name) {
case 'ColorPrimitive':
return;
case 'PointPrimitive':
return;
case 'ImageSourcePrimitive':
imports.add('#include <react/components/image/conversions.h>');
return;
default:
(name: empty);
throw new Error(`Invalid name, got ${name}`);
}
}
component.props.forEach(prop => {
const typeAnnotation = prop.typeAnnotation;
if (typeAnnotation.type === 'NativePrimitiveTypeAnnotation') {
addImportsForNativeName(typeAnnotation.name);
}
if (
typeAnnotation.type === 'ArrayTypeAnnotation' &&
typeAnnotation.elementType.type === 'NativePrimitiveTypeAnnotation'
) {
addImportsForNativeName(typeAnnotation.elementType.name);
}
});
return imports;
}
module.exports = {
generate(libraryName: string, schema: SchemaType): FilesOutput {
const fileName = 'Props.cpp';
const allImports: Set<string> = new Set([
'#include <react/core/propsConversions.h>',
]);
const componentProps = Object.keys(schema.modules)
.map(moduleName => {
const components = schema.modules[moduleName].components;
// No components in this module
if (components == null) {
return null;
}
return Object.keys(components)
.map(componentName => {
const component = components[componentName];
const newName = `${componentName}Props`;
const propsString = generatePropsString(component);
const extendString = getClassExtendString(component);
const imports = getImports(component);
imports.forEach(allImports.add, allImports);
const replacedTemplate = componentTemplate
.replace(/::_CLASSNAME_::/g, newName)
.replace('::_EXTEND_CLASSES_::', extendString)
.replace('::_PROPS_::', propsString);
return replacedTemplate;
})
.join('\n');
})
.filter(Boolean)
.join('\n');
const replacedTemplate = template
.replace(/::_COMPONENT_CLASSES_::/g, componentProps)
.replace('::_LIBRARY_::', libraryName)
.replace(
'::_IMPORTS_::',
Array.from(allImports)
.sort()
.join('\n')
.trim(),
);
return new Map([[fileName, replacedTemplate]]);
},
};