@react-native-windows/codegen 0.75.0 → 0.75.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +13 -5
- package/lib-commonjs/Cli.js +11 -0
- package/lib-commonjs/Cli.js.map +1 -1
- package/lib-commonjs/generators/AliasManaging.d.ts +3 -3
- package/lib-commonjs/generators/AliasManaging.js.map +1 -1
- package/lib-commonjs/generators/GenerateComponentWindows.d.ts +13 -0
- package/lib-commonjs/generators/GenerateComponentWindows.js +337 -0
- package/lib-commonjs/generators/GenerateComponentWindows.js.map +1 -0
- package/lib-commonjs/generators/PropObjectTypes.d.ts +18 -0
- package/lib-commonjs/generators/PropObjectTypes.js +217 -0
- package/lib-commonjs/generators/PropObjectTypes.js.map +1 -0
- package/lib-commonjs/index.d.ts +3 -1
- package/lib-commonjs/index.js +16 -13
- package/lib-commonjs/index.js.map +1 -1
- package/package.json +1 -1
- package/src/Cli.ts +11 -0
- package/src/generators/AliasManaging.ts +12 -12
- package/src/generators/GenerateComponentWindows.ts +393 -0
- package/src/generators/PropObjectTypes.ts +233 -0
- package/src/index.ts +25 -2
|
@@ -0,0 +1,393 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) Microsoft Corporation.
|
|
3
|
+
* Licensed under the MIT License.
|
|
4
|
+
* @format
|
|
5
|
+
*/
|
|
6
|
+
|
|
7
|
+
'use strict';
|
|
8
|
+
|
|
9
|
+
import type {SchemaType, EventTypeAnnotation, PropTypeAnnotation, ObjectTypeAnnotation, CommandParamTypeAnnotation} from '@react-native/codegen/lib/CodegenSchema';
|
|
10
|
+
import {getAliasCppName, setPreferredModuleName} from './AliasManaging';
|
|
11
|
+
import {translateComponentPropsFieldType, translateComponentEventType, translateCommandParamType} from './PropObjectTypes';
|
|
12
|
+
import type {CppStringTypes} from './ObjectTypes';
|
|
13
|
+
import type {AliasMap} from './AliasManaging';
|
|
14
|
+
|
|
15
|
+
export type {CppStringTypes} from './ObjectTypes';
|
|
16
|
+
|
|
17
|
+
type FilesOutput = Map<string, string>;
|
|
18
|
+
|
|
19
|
+
const headerTemplate = `/*
|
|
20
|
+
* This file is auto-generated from ::_COMPONENT_NAME_::NativeComponent spec file in flow / TypeScript.
|
|
21
|
+
*/
|
|
22
|
+
#pragma once
|
|
23
|
+
|
|
24
|
+
#include <JSValueComposition.h>
|
|
25
|
+
#include <NativeModules.h>
|
|
26
|
+
#include <winrt/Microsoft.ReactNative.Composition.h>
|
|
27
|
+
#include <winrt/Microsoft.UI.Composition.h>`
|
|
28
|
+
|
|
29
|
+
|
|
30
|
+
const propsTemplate = `REACT_STRUCT(::_PROPS_NAME_::)
|
|
31
|
+
struct ::_PROPS_NAME_:: : winrt::implements<::_PROPS_NAME_::, winrt::Microsoft::ReactNative::IComponentProps> {
|
|
32
|
+
::_PROPS_NAME_::(winrt::Microsoft::ReactNative::ViewProps props) : ViewProps(props) {}
|
|
33
|
+
|
|
34
|
+
void SetProp(uint32_t hash, winrt::hstring propName, winrt::Microsoft::ReactNative::IJSValueReader value) noexcept {
|
|
35
|
+
winrt::Microsoft::ReactNative::ReadProp(hash, propName, value, *this);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
::_PROPS_FIELDS_::
|
|
39
|
+
const winrt::Microsoft::ReactNative::ViewProps ViewProps;
|
|
40
|
+
};`
|
|
41
|
+
|
|
42
|
+
const propsObjectTemplate = `REACT_STRUCT(::_OBJECT_NAME_::)
|
|
43
|
+
struct ::_OBJECT_NAME_:: {
|
|
44
|
+
::_OBJECT_FIELDS_::};
|
|
45
|
+
`
|
|
46
|
+
const eventsObjectTemplate = `REACT_STRUCT(::_OBJECT_NAME_::)
|
|
47
|
+
struct ::_OBJECT_NAME_:: {
|
|
48
|
+
::_OBJECT_FIELDS_::};
|
|
49
|
+
`
|
|
50
|
+
|
|
51
|
+
const eventEmitterMethodTemplate = ` void ::_EVENT_NAME_::(::_EVENT_OBJECT_TYPE_:: &value) const {
|
|
52
|
+
m_eventEmitter.DispatchEvent(L"::_EVENT_NAME_NO_ON_::", [value](const winrt::Microsoft::ReactNative::IJSValueWriter writer) {
|
|
53
|
+
winrt::Microsoft::ReactNative::WriteValue(writer, value);
|
|
54
|
+
});
|
|
55
|
+
}`;
|
|
56
|
+
|
|
57
|
+
|
|
58
|
+
const eventEmitterTemplate = `::_COMPONENT_EVENT_OBJECT_TYPES_::
|
|
59
|
+
|
|
60
|
+
struct ::_EVENT_EMITTER_NAME_:: {
|
|
61
|
+
::_EVENT_EMITTER_NAME_::(const winrt::Microsoft::ReactNative::EventEmitter &eventEmitter)
|
|
62
|
+
: m_eventEmitter(eventEmitter) {}
|
|
63
|
+
|
|
64
|
+
::_EVENT_EMITTER_USINGS_::
|
|
65
|
+
|
|
66
|
+
::_EVENT_EMITTER_METHODS_::
|
|
67
|
+
|
|
68
|
+
private:
|
|
69
|
+
winrt::Microsoft::ReactNative::EventEmitter m_eventEmitter{nullptr};
|
|
70
|
+
};`
|
|
71
|
+
|
|
72
|
+
const baseStructTemplate = `
|
|
73
|
+
template<typename TUserData>
|
|
74
|
+
struct Base::_COMPONENT_NAME_:: {
|
|
75
|
+
|
|
76
|
+
virtual void UpdateProps(
|
|
77
|
+
const winrt::Microsoft::ReactNative::ComponentView &/*view*/,
|
|
78
|
+
const winrt::com_ptr<::_COMPONENT_NAME_::Props> &newProps,
|
|
79
|
+
const winrt::com_ptr<::_COMPONENT_NAME_::Props> &/*oldProps*/) noexcept {
|
|
80
|
+
m_props = newProps;
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// UpdateState will only be called if this method is overridden
|
|
84
|
+
virtual void UpdateState(
|
|
85
|
+
const winrt::Microsoft::ReactNative::ComponentView &/*view*/,
|
|
86
|
+
const winrt::Microsoft::ReactNative::IComponentState &/*newState*/) noexcept {
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
virtual void UpdateEventEmitter(const std::shared_ptr<::_COMPONENT_NAME_::EventEmitter> &eventEmitter) noexcept {
|
|
90
|
+
m_eventEmitter = eventEmitter;
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// MountChildComponentView will only be called if this method is overridden
|
|
94
|
+
virtual void MountChildComponentView(const winrt::Microsoft::ReactNative::ComponentView &/*view*/,
|
|
95
|
+
const winrt::Microsoft::ReactNative::MountChildComponentViewArgs &/*args*/) noexcept {
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
// UnmountChildComponentView will only be called if this method is overridden
|
|
99
|
+
virtual void UnmountChildComponentView(const winrt::Microsoft::ReactNative::ComponentView &/*view*/,
|
|
100
|
+
const winrt::Microsoft::ReactNative::UnmountChildComponentViewArgs &/*args*/) noexcept {
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
// Initialize will only be called if this method is overridden
|
|
104
|
+
virtual void Initialize(const winrt::Microsoft::ReactNative::ComponentView &/*view*/) noexcept {
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// CreateVisual will only be called if this method is overridden
|
|
108
|
+
virtual winrt::Microsoft::UI::Composition::Visual CreateVisual(const winrt::Microsoft::ReactNative::ComponentView &view) noexcept {
|
|
109
|
+
return view.as<winrt::Microsoft::ReactNative::Composition::ComponentView>().Compositor().CreateSpriteVisual();
|
|
110
|
+
}
|
|
111
|
+
|
|
112
|
+
// FinalizeUpdate will only be called if this method is overridden
|
|
113
|
+
virtual void FinalizeUpdate(const winrt::Microsoft::ReactNative::ComponentView &/*view*/,
|
|
114
|
+
winrt::Microsoft::ReactNative::ComponentViewUpdateMask /*mask*/) noexcept {
|
|
115
|
+
}
|
|
116
|
+
|
|
117
|
+
::_COMPONENT_VIEW_COMMAND_HANDLERS_::
|
|
118
|
+
|
|
119
|
+
::_COMPONENT_VIEW_COMMAND_HANDLER_::
|
|
120
|
+
|
|
121
|
+
const std::shared_ptr<::_COMPONENT_NAME_::EventEmitter>& EventEmitter() const { return m_eventEmitter; }
|
|
122
|
+
const winrt::com_ptr<::_COMPONENT_NAME_::Props>& Props() const { return m_props; }
|
|
123
|
+
|
|
124
|
+
private:
|
|
125
|
+
winrt::com_ptr<::_COMPONENT_NAME_::Props> m_props;
|
|
126
|
+
std::shared_ptr<::_COMPONENT_NAME_::EventEmitter> m_eventEmitter;
|
|
127
|
+
};
|
|
128
|
+
`;
|
|
129
|
+
|
|
130
|
+
const registerTemplate = `
|
|
131
|
+
template <typename TUserData>
|
|
132
|
+
void Register::_COMPONENT_NAME_::NativeComponent(
|
|
133
|
+
winrt::Microsoft::ReactNative::IReactPackageBuilder const &packageBuilder,
|
|
134
|
+
std::function<void(const winrt::Microsoft::ReactNative::Composition::IReactCompositionViewComponentBuilder&)> builderCallback) noexcept {
|
|
135
|
+
packageBuilder.as<winrt::Microsoft::ReactNative::IReactPackageBuilderFabric>().AddViewComponent(
|
|
136
|
+
L"::_COMPONENT_NAME_::", [builderCallback](winrt::Microsoft::ReactNative::IReactViewComponentBuilder const &builder) noexcept {
|
|
137
|
+
auto compBuilder = builder.as<winrt::Microsoft::ReactNative::Composition::IReactCompositionViewComponentBuilder>();
|
|
138
|
+
|
|
139
|
+
builder.SetCreateProps(
|
|
140
|
+
[](winrt::Microsoft::ReactNative::ViewProps props) noexcept { return winrt::make<::_COMPONENT_NAME_::Props>(props); });
|
|
141
|
+
|
|
142
|
+
builder.SetUpdatePropsHandler([](const winrt::Microsoft::ReactNative::ComponentView &view,
|
|
143
|
+
const winrt::Microsoft::ReactNative::IComponentProps &newProps,
|
|
144
|
+
const winrt::Microsoft::ReactNative::IComponentProps &oldProps) noexcept {
|
|
145
|
+
auto userData = view.UserData().as<TUserData>();
|
|
146
|
+
userData->UpdateProps(view, newProps ? newProps.as<::_COMPONENT_NAME_::Props>() : nullptr, oldProps ? oldProps.as<::_COMPONENT_NAME_::Props>() : nullptr);
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
builder.SetUpdateEventEmitterHandler([](const winrt::Microsoft::ReactNative::ComponentView &view,
|
|
150
|
+
const winrt::Microsoft::ReactNative::EventEmitter &eventEmitter) noexcept {
|
|
151
|
+
auto userData = view.UserData().as<TUserData>();
|
|
152
|
+
userData->UpdateEventEmitter(std::make_shared<::_COMPONENT_NAME_::EventEmitter>(eventEmitter));
|
|
153
|
+
});
|
|
154
|
+
|
|
155
|
+
if constexpr (&TUserData::FinalizeUpdate != &Base::_COMPONENT_NAME_::<TUserData>::FinalizeUpdate) {
|
|
156
|
+
builder.SetFinalizeUpdateHandler([](const winrt::Microsoft::ReactNative::ComponentView &view,
|
|
157
|
+
winrt::Microsoft::ReactNative::ComponentViewUpdateMask mask) noexcept {
|
|
158
|
+
auto userData = view.UserData().as<TUserData>();
|
|
159
|
+
userData->FinalizeUpdate(view, mask);
|
|
160
|
+
});
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
if constexpr (&TUserData::UpdateState != &Base::_COMPONENT_NAME_::<TUserData>::UpdateState) {
|
|
164
|
+
builder.SetUpdateStateHandler([](const winrt::Microsoft::ReactNative::ComponentView &view,
|
|
165
|
+
const winrt::Microsoft::ReactNative::IComponentState &newState) noexcept {
|
|
166
|
+
auto userData = view.UserData().as<TUserData>();
|
|
167
|
+
userData->member(view, newState);
|
|
168
|
+
});
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
::_REGISTER_CUSTOM_COMMAND_HANDLER_::
|
|
172
|
+
|
|
173
|
+
if constexpr (&TUserData::MountChildComponentView != &Base::_COMPONENT_NAME_::<TUserData>::MountChildComponentView) {
|
|
174
|
+
builder.SetMountChildComponentViewHandler([](const winrt::Microsoft::ReactNative::ComponentView &view,
|
|
175
|
+
const winrt::Microsoft::ReactNative::MountChildComponentViewArgs &args) noexcept {
|
|
176
|
+
auto userData = view.UserData().as<TUserData>();
|
|
177
|
+
return userData->MountChildComponentView(view, args);
|
|
178
|
+
});
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
if constexpr (&TUserData::UnmountChildComponentView != &Base::_COMPONENT_NAME_::<TUserData>::UnmountChildComponentView) {
|
|
182
|
+
builder.SetUnmountChildComponentViewHandler([](const winrt::Microsoft::ReactNative::ComponentView &view,
|
|
183
|
+
const winrt::Microsoft::ReactNative::UnmountChildComponentViewArgs &args) noexcept {
|
|
184
|
+
auto userData = view.UserData().as<TUserData>();
|
|
185
|
+
return userData->UnmountChildComponentView(view, args);
|
|
186
|
+
});
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
compBuilder.SetViewComponentViewInitializer([](const winrt::Microsoft::ReactNative::ComponentView &view) noexcept {
|
|
190
|
+
auto userData = winrt::make_self<TUserData>();
|
|
191
|
+
if constexpr (&TUserData::Initialize != &Base::_COMPONENT_NAME_::<TUserData>::Initialize) {
|
|
192
|
+
userData->Initialize(view);
|
|
193
|
+
}
|
|
194
|
+
view.UserData(*userData);
|
|
195
|
+
});
|
|
196
|
+
|
|
197
|
+
if constexpr (&TUserData::CreateVisual != &Base::_COMPONENT_NAME_::<TUserData>::CreateVisual) {
|
|
198
|
+
compBuilder.SetCreateVisualHandler([](const winrt::Microsoft::ReactNative::ComponentView &view) noexcept {
|
|
199
|
+
auto userData = view.UserData().as<TUserData>();
|
|
200
|
+
return userData->CreateVisual(view);
|
|
201
|
+
});
|
|
202
|
+
}
|
|
203
|
+
|
|
204
|
+
// Allow app to further customize the builder
|
|
205
|
+
if (builderCallback) {
|
|
206
|
+
builderCallback(compBuilder);
|
|
207
|
+
}
|
|
208
|
+
});
|
|
209
|
+
}
|
|
210
|
+
`;
|
|
211
|
+
|
|
212
|
+
const fileTemplate = `
|
|
213
|
+
${headerTemplate}
|
|
214
|
+
|
|
215
|
+
namespace ::_NAMESPACE_:: {
|
|
216
|
+
|
|
217
|
+
::_COMPONENT_PROP_OBJECT_TYPES_::
|
|
218
|
+
::_COMPONENT_PROP_TYPES_::
|
|
219
|
+
|
|
220
|
+
::_COMPONENT_EVENT_EMITTER_::
|
|
221
|
+
|
|
222
|
+
::_BASE_COMPONENT_STRUCT_::
|
|
223
|
+
|
|
224
|
+
::_COMPONENT_REGISTRATION_::
|
|
225
|
+
} // namespace ::_NAMESPACE_::
|
|
226
|
+
`;
|
|
227
|
+
|
|
228
|
+
function capitalizeFirstLetter(s: string) {
|
|
229
|
+
return s.charAt(0).toUpperCase() + s.slice(1);
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
export function createComponentGenerator({
|
|
233
|
+
namespace,
|
|
234
|
+
cppStringType,
|
|
235
|
+
}: {
|
|
236
|
+
namespace: string;
|
|
237
|
+
cppStringType: CppStringTypes;
|
|
238
|
+
}) {
|
|
239
|
+
return (
|
|
240
|
+
_libraryName: string,
|
|
241
|
+
schema: SchemaType,
|
|
242
|
+
_moduleSpecName: string,
|
|
243
|
+
): FilesOutput => {
|
|
244
|
+
const files = new Map<string, string>();
|
|
245
|
+
|
|
246
|
+
const cppCodegenOptions = {cppStringType};
|
|
247
|
+
|
|
248
|
+
for (const componentName of Object.keys(schema.modules)) {
|
|
249
|
+
const component = schema.modules[componentName];
|
|
250
|
+
setPreferredModuleName(componentName);
|
|
251
|
+
|
|
252
|
+
if (component.type === 'Component') {
|
|
253
|
+
console.log(`Generating ${componentName}.g.h`);
|
|
254
|
+
|
|
255
|
+
const componentShape = component.components[componentName];
|
|
256
|
+
|
|
257
|
+
componentShape.extendsProps.forEach(propsBaseType => {
|
|
258
|
+
// eslint-disable-next-line @typescript-eslint/no-unnecessary-condition
|
|
259
|
+
if (propsBaseType.type !== 'ReactNativeBuiltInType' || propsBaseType.knownTypeName !== 'ReactNativeCoreViewProps') {
|
|
260
|
+
throw new Error('Currently only supports props extending from ViewProps');
|
|
261
|
+
}
|
|
262
|
+
});
|
|
263
|
+
|
|
264
|
+
// Props
|
|
265
|
+
const propObjectAliases: AliasMap<ObjectTypeAnnotation<PropTypeAnnotation>> = {types:{}, jobs: []};
|
|
266
|
+
const propsName = `${componentName}Props`;
|
|
267
|
+
const propsFields = componentShape.props.map(prop => {
|
|
268
|
+
const propType = translateComponentPropsFieldType(prop.typeAnnotation, propObjectAliases, `${propsName}_${prop.name}`, cppCodegenOptions);
|
|
269
|
+
return ` REACT_FIELD(${prop.name})\n ${(prop.optional && !propType.alreadySupportsOptionalOrHasDefault) ? `std::optional<${propType.type}>` : propType.type} ${prop.name}${propType.initializer};\n`;
|
|
270
|
+
}).join('\n');
|
|
271
|
+
|
|
272
|
+
const propObjectTypes = propObjectAliases.jobs.map(propObjectTypeName => {
|
|
273
|
+
const propObjectType = propObjectAliases.types[propObjectTypeName]!;
|
|
274
|
+
const propsObjectFields = propObjectType.properties.map(property => {
|
|
275
|
+
const propType = translateComponentPropsFieldType(property.typeAnnotation, propObjectAliases, `${propsName}_${property.name}`, cppCodegenOptions);
|
|
276
|
+
return ` REACT_FIELD(${property.name})\n ${(property.optional && !propType.alreadySupportsOptionalOrHasDefault) ? `std::optional<${propType.type}>` : propType.type} ${property.name}${propType.initializer};\n`;
|
|
277
|
+
}).join('\n');
|
|
278
|
+
|
|
279
|
+
return propsObjectTemplate.replace(/::_OBJECT_NAME_::/g, getAliasCppName(propObjectTypeName)).replace(/::_OBJECT_FIELDS_::/g, propsObjectFields);
|
|
280
|
+
}).join('\n');
|
|
281
|
+
|
|
282
|
+
|
|
283
|
+
// Events
|
|
284
|
+
const eventObjectAliases: AliasMap<ObjectTypeAnnotation<EventTypeAnnotation>> = {types:{}, jobs: []};
|
|
285
|
+
const eventEmitterName = `${componentName}EventEmitter`;
|
|
286
|
+
const eventEmitterMethods = componentShape.events.filter(event => event.typeAnnotation.argument).map(event => {
|
|
287
|
+
if (event.typeAnnotation.argument?.baseTypes) {
|
|
288
|
+
throw new Error('Events with base type arguments not currently supported');
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
// Called to collect the eventObjectAliases
|
|
292
|
+
translateComponentEventType(event.typeAnnotation.argument!, eventObjectAliases, `${event.name}`, cppCodegenOptions);
|
|
293
|
+
|
|
294
|
+
// onSomething -> something
|
|
295
|
+
let eventNameLower = event.name.replace('on', '');
|
|
296
|
+
eventNameLower = eventNameLower[0].toLowerCase() + eventNameLower.slice(1);
|
|
297
|
+
|
|
298
|
+
return eventEmitterMethodTemplate
|
|
299
|
+
.replace(/::_EVENT_NAME_::/g,event.name)
|
|
300
|
+
.replace(/::_EVENT_NAME_NO_ON_::/g,eventNameLower)
|
|
301
|
+
.replace(/::_EVENT_OBJECT_TYPE_::/g, event.name.replace('on', 'On'));
|
|
302
|
+
}).join('\n\n');
|
|
303
|
+
|
|
304
|
+
const eventObjects = eventObjectAliases.jobs.map(eventObjectTypeName => {
|
|
305
|
+
const eventObjectType = eventObjectAliases.types[eventObjectTypeName]!;
|
|
306
|
+
const eventObjectFields = eventObjectType.properties.map(property => {
|
|
307
|
+
const eventPropType = translateComponentEventType(property.typeAnnotation, eventObjectAliases, eventObjectTypeName, cppCodegenOptions);
|
|
308
|
+
return ` REACT_FIELD(${property.name})\n ${(property.optional && !eventPropType.alreadySupportsOptionalOrHasDefault) ? `std::optional<${eventPropType.type}>` : eventPropType.type} ${property.name}${eventPropType.initializer};\n`;
|
|
309
|
+
}).join('\n');
|
|
310
|
+
return eventsObjectTemplate.replace(/::_OBJECT_NAME_::/g, `${componentName}_${eventObjectTypeName.replace('on', 'On')}`).replace(/::_OBJECT_FIELDS_::/g, eventObjectFields);
|
|
311
|
+
}).join('\n');
|
|
312
|
+
|
|
313
|
+
const eventObjectUsings = eventObjectAliases.jobs.map(eventObjectTypeName => {
|
|
314
|
+
return ` using ${eventObjectTypeName.replace('on', 'On')} = ${componentName}_${eventObjectTypeName.replace('on', 'On')};`
|
|
315
|
+
}).join('\n');
|
|
316
|
+
|
|
317
|
+
const eventEmitter = eventEmitterTemplate
|
|
318
|
+
.replace(/::_COMPONENT_EVENT_OBJECT_TYPES_::/g, eventObjects)
|
|
319
|
+
.replace(/::_EVENT_EMITTER_METHODS_::/g, eventEmitterMethods)
|
|
320
|
+
.replace(/::_EVENT_EMITTER_USINGS_::/g, eventObjectUsings);
|
|
321
|
+
|
|
322
|
+
|
|
323
|
+
// Commands
|
|
324
|
+
const commandAliases: AliasMap<ObjectTypeAnnotation<CommandParamTypeAnnotation>> = {types:{}, jobs: []};
|
|
325
|
+
const hasAnyCommands = (componentShape.commands.length !== 0);
|
|
326
|
+
const commandHandlers = hasAnyCommands ? componentShape.commands.map(command => {
|
|
327
|
+
const commandArgs = command.typeAnnotation.params.map(param => {
|
|
328
|
+
const commandArgType = translateCommandParamType(param.typeAnnotation, commandAliases, `${componentName}_${command.name}`, cppCodegenOptions);
|
|
329
|
+
return `${(param.optional && !commandArgType.alreadySupportsOptionalOrHasDefault) ? `std::optional<${commandArgType.type}>` : commandArgType.type} ${param.name}`;
|
|
330
|
+
}).join(', ');
|
|
331
|
+
|
|
332
|
+
return ` // You must provide an implementation of this method to handle the "${command.name}" command
|
|
333
|
+
virtual void Handle${capitalizeFirstLetter(command.name)}Command(${commandArgs}) noexcept = 0;`;
|
|
334
|
+
}).join('\n\n') : '';
|
|
335
|
+
|
|
336
|
+
|
|
337
|
+
const commandHandler = hasAnyCommands ? `void HandleCommand(const winrt::Microsoft::ReactNative::ComponentView &view, winrt::hstring commandName, const winrt::Microsoft::ReactNative::IJSValueReader &args) noexcept {
|
|
338
|
+
args;
|
|
339
|
+
auto userData = view.UserData().as<TUserData>();
|
|
340
|
+
${componentShape.commands.map(command => {
|
|
341
|
+
const commaSeparatedCommandArgs = command.typeAnnotation.params.map(param => param.name).join(', ');
|
|
342
|
+
return ` if (commandName == L"${command.name}") {
|
|
343
|
+
${command.typeAnnotation.params.length !== 0 ? ` ${command.typeAnnotation.params.map(param => {
|
|
344
|
+
const commandArgType = translateCommandParamType(param.typeAnnotation, commandAliases, `${componentName}_${command.name}`, cppCodegenOptions);
|
|
345
|
+
return `${(param.optional && !commandArgType.alreadySupportsOptionalOrHasDefault) ? `std::optional<${commandArgType.type}>` : commandArgType.type} ${param.name};`;
|
|
346
|
+
}).join('\n')}
|
|
347
|
+
winrt::Microsoft::ReactNative::ReadArgs(args, ${commaSeparatedCommandArgs});` : ''}
|
|
348
|
+
userData->Handle${capitalizeFirstLetter(command.name)}Command(${commaSeparatedCommandArgs});
|
|
349
|
+
return;
|
|
350
|
+
}`
|
|
351
|
+
}).join('\n\n')}
|
|
352
|
+
}` : '';
|
|
353
|
+
|
|
354
|
+
const registerCommandHandler = hasAnyCommands ? ` builder.SetCustomCommandHandler([](const winrt::Microsoft::ReactNative::ComponentView &view,
|
|
355
|
+
winrt::hstring commandName,
|
|
356
|
+
const winrt::Microsoft::ReactNative::IJSValueReader &args) noexcept {
|
|
357
|
+
auto userData = view.UserData().as<TUserData>();
|
|
358
|
+
userData->HandleCommand(view, commandName, args);
|
|
359
|
+
});` : '';
|
|
360
|
+
|
|
361
|
+
const baseType = baseStructTemplate
|
|
362
|
+
.replace(/::_COMPONENT_VIEW_COMMAND_HANDLERS_::/g, commandHandlers)
|
|
363
|
+
.replace(/::_COMPONENT_VIEW_COMMAND_HANDLER_::/g, commandHandler);
|
|
364
|
+
|
|
365
|
+
// Registration
|
|
366
|
+
const componentRegistration = registerTemplate.replace(/::_REGISTER_CUSTOM_COMMAND_HANDLER_::/g, registerCommandHandler);
|
|
367
|
+
|
|
368
|
+
// Final output
|
|
369
|
+
const replaceContent = function (template: string): string {
|
|
370
|
+
return template
|
|
371
|
+
.replace(/::_COMPONENT_PROP_OBJECT_TYPES_::/g, propObjectTypes)
|
|
372
|
+
.replace(/::_COMPONENT_PROP_TYPES_::/g, propsTemplate)
|
|
373
|
+
.replace(/::_COMPONENT_EVENT_EMITTER_::/g, eventEmitter)
|
|
374
|
+
.replace(/::_BASE_COMPONENT_STRUCT_::/g, baseType)
|
|
375
|
+
.replace(/::_COMPONENT_REGISTRATION_::/g, componentRegistration)
|
|
376
|
+
.replace(/::_EVENT_EMITTER_NAME_::/g, eventEmitterName)
|
|
377
|
+
.replace(/::_PROPS_NAME_::/g, propsName)
|
|
378
|
+
.replace(/::_COMPONENT_NAME_::/g, componentName)
|
|
379
|
+
.replace(/::_PROPS_FIELDS_::/g, propsFields)
|
|
380
|
+
.replace(/::_NAMESPACE_::/g, namespace)
|
|
381
|
+
.replace(/\n\n\n+/g, '\n\n');
|
|
382
|
+
};
|
|
383
|
+
|
|
384
|
+
files.set(
|
|
385
|
+
`${componentName}.g.h`,
|
|
386
|
+
replaceContent(fileTemplate),
|
|
387
|
+
);
|
|
388
|
+
}
|
|
389
|
+
}
|
|
390
|
+
|
|
391
|
+
return files;
|
|
392
|
+
};
|
|
393
|
+
}
|
|
@@ -0,0 +1,233 @@
|
|
|
1
|
+
import type { PropTypeAnnotation, ObjectTypeAnnotation, EventTypeAnnotation, CommandParamTypeAnnotation } from '@react-native/codegen/lib/CodegenSchema';
|
|
2
|
+
import type { CppCodegenOptions } from './ObjectTypes';
|
|
3
|
+
import {
|
|
4
|
+
AliasMap,
|
|
5
|
+
getAnonymousAliasCppName,
|
|
6
|
+
} from './AliasManaging';
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
// eslint-disable-next-line complexity
|
|
10
|
+
export function translateComponentPropsFieldType(type: PropTypeAnnotation,
|
|
11
|
+
aliases: AliasMap<ObjectTypeAnnotation<PropTypeAnnotation>>,
|
|
12
|
+
baseAliasName: string,
|
|
13
|
+
options: CppCodegenOptions): { type: string, initializer: string, alreadySupportsOptionalOrHasDefault?: boolean } {
|
|
14
|
+
switch (type.type) {
|
|
15
|
+
case 'StringTypeAnnotation':
|
|
16
|
+
return { type: options.cppStringType, initializer: type.default ? `{${type.default}}` : '', alreadySupportsOptionalOrHasDefault: !!type.default };
|
|
17
|
+
case 'FloatTypeAnnotation':
|
|
18
|
+
return { type: 'float', initializer: type.default ? `{${type.default}}` : '{}', alreadySupportsOptionalOrHasDefault: !!type.default };
|
|
19
|
+
case 'DoubleTypeAnnotation':
|
|
20
|
+
return { type: 'double', initializer: type.default ? `{${type.default}}` : '{}', alreadySupportsOptionalOrHasDefault: !!type.default };
|
|
21
|
+
case 'Int32TypeAnnotation':
|
|
22
|
+
return { type: 'int32_t', initializer: type.default ? `{${type.default}}` : '{}', alreadySupportsOptionalOrHasDefault: !!type.default };
|
|
23
|
+
case 'BooleanTypeAnnotation':
|
|
24
|
+
return { type: 'bool', initializer: type.default ? `{${type.default}}` : '{}', alreadySupportsOptionalOrHasDefault: !!type.default };
|
|
25
|
+
case 'ArrayTypeAnnotation':
|
|
26
|
+
|
|
27
|
+
let arrayTemplateArg = '';
|
|
28
|
+
switch (type.elementType.type) {
|
|
29
|
+
case 'BooleanTypeAnnotation':
|
|
30
|
+
arrayTemplateArg = 'bool';
|
|
31
|
+
break;
|
|
32
|
+
case 'DoubleTypeAnnotation':
|
|
33
|
+
arrayTemplateArg = 'double';
|
|
34
|
+
break;
|
|
35
|
+
case 'FloatTypeAnnotation':
|
|
36
|
+
arrayTemplateArg = 'float';
|
|
37
|
+
break;
|
|
38
|
+
case 'Int32TypeAnnotation':
|
|
39
|
+
arrayTemplateArg = 'int32_t';
|
|
40
|
+
break;
|
|
41
|
+
case 'StringTypeAnnotation':
|
|
42
|
+
arrayTemplateArg = options.cppStringType;
|
|
43
|
+
break;
|
|
44
|
+
case 'ArrayTypeAnnotation':
|
|
45
|
+
const innerType = translateComponentPropsFieldType(type.elementType, aliases, baseAliasName, options);
|
|
46
|
+
arrayTemplateArg = `std::vector<${innerType.type}>`;
|
|
47
|
+
break;
|
|
48
|
+
case 'ObjectTypeAnnotation':
|
|
49
|
+
arrayTemplateArg = translateComponentPropsFieldType(type.elementType, aliases, baseAliasName, options).type;
|
|
50
|
+
break;
|
|
51
|
+
case 'ReservedPropTypeAnnotation':
|
|
52
|
+
switch (type.elementType.name) {
|
|
53
|
+
case 'ColorPrimitive':
|
|
54
|
+
arrayTemplateArg = 'winrt::Microsoft::ReactNative::Color';
|
|
55
|
+
break;
|
|
56
|
+
case 'DimensionPrimitive':
|
|
57
|
+
case 'EdgeInsetsPrimitive':
|
|
58
|
+
case 'ImageRequestPrimitive':
|
|
59
|
+
case 'ImageSourcePrimitive':
|
|
60
|
+
case 'PointPrimitive':
|
|
61
|
+
arrayTemplateArg = 'winrt::Microsoft::ReactNative::JSValue'; // TODO - better handling for these types than JSValue
|
|
62
|
+
break;
|
|
63
|
+
default:
|
|
64
|
+
throw new Error(`Unhandled ReservedPropTypeAnnotation type: ${type.elementType.name}`);
|
|
65
|
+
}
|
|
66
|
+
break;
|
|
67
|
+
case 'StringEnumTypeAnnotation':
|
|
68
|
+
arrayTemplateArg = options.cppStringType; // TODO - better enum type handling than just passing a string
|
|
69
|
+
break;
|
|
70
|
+
default:
|
|
71
|
+
throw new Error(`Unhandled type: ${type.type}`);
|
|
72
|
+
}
|
|
73
|
+
|
|
74
|
+
return { type: `std::vector<${arrayTemplateArg}>`, initializer: '' };
|
|
75
|
+
case 'ReservedPropTypeAnnotation':
|
|
76
|
+
switch (type.name) {
|
|
77
|
+
case 'ColorPrimitive':
|
|
78
|
+
return { type: 'winrt::Microsoft::ReactNative::Color', initializer: '{nullptr}', alreadySupportsOptionalOrHasDefault: true };
|
|
79
|
+
case 'DimensionPrimitive':
|
|
80
|
+
case 'EdgeInsetsPrimitive':
|
|
81
|
+
case 'ImageRequestPrimitive':
|
|
82
|
+
case 'ImageSourcePrimitive':
|
|
83
|
+
case 'PointPrimitive':
|
|
84
|
+
return { type: 'winrt::Microsoft::ReactNative::JSValue', initializer: '{nullptr}', alreadySupportsOptionalOrHasDefault: true }; // TODO - better handling for these types than JSValue
|
|
85
|
+
default:
|
|
86
|
+
throw new Error(`Unhandled ReservedPropTypeAnnotation type: ${type.name}`);
|
|
87
|
+
}
|
|
88
|
+
case 'ObjectTypeAnnotation': {
|
|
89
|
+
return { type: getAnonymousAliasCppName<ObjectTypeAnnotation<PropTypeAnnotation>>(aliases, baseAliasName, type), initializer: '' };
|
|
90
|
+
}
|
|
91
|
+
case 'MixedTypeAnnotation':
|
|
92
|
+
return { type: 'winrt::Microsoft::ReactNative::JSValue', initializer: '{nullptr}', alreadySupportsOptionalOrHasDefault: true };
|
|
93
|
+
case 'Int32EnumTypeAnnotation':
|
|
94
|
+
return { type: 'int32_t', initializer: '' }; // TODO - better enum type handling than just passing a string
|
|
95
|
+
case 'StringEnumTypeAnnotation':
|
|
96
|
+
return { type: options.cppStringType, initializer: '' }; // TODO - better enum type handling than just passing an int
|
|
97
|
+
default:
|
|
98
|
+
throw new Error(`Unhandled type: ${(type as any).type}`);
|
|
99
|
+
}
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
export function translateComponentEventType(type: EventTypeAnnotation,
|
|
103
|
+
aliases: AliasMap<ObjectTypeAnnotation<EventTypeAnnotation>>,
|
|
104
|
+
baseAliasName: string,
|
|
105
|
+
options: CppCodegenOptions): { type: string, initializer: string, alreadySupportsOptionalOrHasDefault?: boolean } {
|
|
106
|
+
switch (type.type) {
|
|
107
|
+
case 'StringTypeAnnotation':
|
|
108
|
+
return { type: options.cppStringType, initializer: '' };
|
|
109
|
+
case 'FloatTypeAnnotation':
|
|
110
|
+
return { type: 'float', initializer: '{}' };
|
|
111
|
+
case 'DoubleTypeAnnotation':
|
|
112
|
+
return { type: 'double', initializer: '{}' };
|
|
113
|
+
case 'Int32TypeAnnotation':
|
|
114
|
+
return { type: 'int32_t', initializer: '{}' };
|
|
115
|
+
case 'BooleanTypeAnnotation':
|
|
116
|
+
return { type: 'bool', initializer: '{}' };
|
|
117
|
+
case 'ArrayTypeAnnotation':
|
|
118
|
+
{
|
|
119
|
+
let arrayTemplateArg = '';
|
|
120
|
+
switch (type.elementType.type) {
|
|
121
|
+
case 'BooleanTypeAnnotation':
|
|
122
|
+
arrayTemplateArg = 'bool';
|
|
123
|
+
break;
|
|
124
|
+
case 'DoubleTypeAnnotation':
|
|
125
|
+
arrayTemplateArg = 'double';
|
|
126
|
+
break;
|
|
127
|
+
case 'FloatTypeAnnotation':
|
|
128
|
+
arrayTemplateArg = 'float';
|
|
129
|
+
break;
|
|
130
|
+
case 'Int32TypeAnnotation':
|
|
131
|
+
arrayTemplateArg = 'int32_t';
|
|
132
|
+
break;
|
|
133
|
+
case 'StringTypeAnnotation':
|
|
134
|
+
arrayTemplateArg = options.cppStringType;
|
|
135
|
+
break;
|
|
136
|
+
case 'ArrayTypeAnnotation':
|
|
137
|
+
const innerType = translateComponentEventType(type.elementType, aliases, baseAliasName, options);
|
|
138
|
+
arrayTemplateArg = `std::vector<${innerType.type}>`;
|
|
139
|
+
break;
|
|
140
|
+
case 'MixedTypeAnnotation':
|
|
141
|
+
arrayTemplateArg = 'winrt::Microsoft::ReactNative::JSValue';
|
|
142
|
+
break;
|
|
143
|
+
case 'ObjectTypeAnnotation':
|
|
144
|
+
arrayTemplateArg = translateComponentEventType(type.elementType, aliases, baseAliasName, options).type;
|
|
145
|
+
break;
|
|
146
|
+
case 'StringEnumTypeAnnotation':
|
|
147
|
+
arrayTemplateArg = options.cppStringType; // TODO - better enum type handling than just passing a string
|
|
148
|
+
break;
|
|
149
|
+
default:
|
|
150
|
+
throw new Error(`Unhandled type: ${type.type}`);
|
|
151
|
+
}
|
|
152
|
+
|
|
153
|
+
return { type: `std::vector<${arrayTemplateArg}>`, initializer: '{}' };
|
|
154
|
+
}
|
|
155
|
+
case 'ObjectTypeAnnotation': {
|
|
156
|
+
return { type: getAnonymousAliasCppName<ObjectTypeAnnotation<EventTypeAnnotation>>(aliases, baseAliasName, type), initializer: '' };
|
|
157
|
+
}
|
|
158
|
+
case 'MixedTypeAnnotation': {
|
|
159
|
+
return { type: 'winrt::Microsoft::ReactNative::JSValue', initializer: '{nullptr}', alreadySupportsOptionalOrHasDefault: true };
|
|
160
|
+
}
|
|
161
|
+
case 'StringEnumTypeAnnotation':
|
|
162
|
+
return { type: options.cppStringType, initializer: '' }; // TODO - better enum type handling than just passing a string
|
|
163
|
+
default:
|
|
164
|
+
throw new Error(`Unhandled type: ${(type as any).type}`);
|
|
165
|
+
}
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
|
|
169
|
+
export function translateCommandParamType(type: CommandParamTypeAnnotation,
|
|
170
|
+
aliases: AliasMap<ObjectTypeAnnotation<CommandParamTypeAnnotation>>,
|
|
171
|
+
baseAliasName: string,
|
|
172
|
+
options: CppCodegenOptions): { type: string, initializer: string, alreadySupportsOptionalOrHasDefault?: boolean } {
|
|
173
|
+
switch (type.type) {
|
|
174
|
+
case 'StringTypeAnnotation':
|
|
175
|
+
return { type: options.cppStringType, initializer: '' };
|
|
176
|
+
case 'FloatTypeAnnotation':
|
|
177
|
+
return { type: 'float', initializer: '{}' };
|
|
178
|
+
case 'DoubleTypeAnnotation':
|
|
179
|
+
return { type: 'double', initializer: '{}' };
|
|
180
|
+
case 'Int32TypeAnnotation':
|
|
181
|
+
return { type: 'int32_t', initializer: '{}' };
|
|
182
|
+
case 'BooleanTypeAnnotation':
|
|
183
|
+
return { type: 'bool', initializer: '{}' };
|
|
184
|
+
case 'ArrayTypeAnnotation':
|
|
185
|
+
{
|
|
186
|
+
let arrayTemplateArg = '';
|
|
187
|
+
switch (type.elementType.type) {
|
|
188
|
+
case 'BooleanTypeAnnotation':
|
|
189
|
+
arrayTemplateArg = 'bool';
|
|
190
|
+
break;
|
|
191
|
+
case 'DoubleTypeAnnotation':
|
|
192
|
+
arrayTemplateArg = 'double';
|
|
193
|
+
break;
|
|
194
|
+
case 'FloatTypeAnnotation':
|
|
195
|
+
arrayTemplateArg = 'float';
|
|
196
|
+
break;
|
|
197
|
+
case 'Int32TypeAnnotation':
|
|
198
|
+
arrayTemplateArg = 'int32_t';
|
|
199
|
+
break;
|
|
200
|
+
case 'StringTypeAnnotation':
|
|
201
|
+
arrayTemplateArg = options.cppStringType;
|
|
202
|
+
break;
|
|
203
|
+
case 'ArrayTypeAnnotation':
|
|
204
|
+
const innerType = translateCommandParamType(type.elementType, aliases, baseAliasName, options);
|
|
205
|
+
arrayTemplateArg = `std::vector<${innerType.type}>`;
|
|
206
|
+
break;
|
|
207
|
+
case 'ReservedPropTypeAnnotation':
|
|
208
|
+
arrayTemplateArg = 'winrt::Microsoft::ReactNative::JSValue';
|
|
209
|
+
break;
|
|
210
|
+
case 'ObjectTypeAnnotation':
|
|
211
|
+
arrayTemplateArg = 'winrt::Microsoft::ReactNative::JSValueObject'; // TODO - better typing
|
|
212
|
+
break;
|
|
213
|
+
case 'StringEnumTypeAnnotation':
|
|
214
|
+
arrayTemplateArg = options.cppStringType; // TODO - better enum type handling than just passing a string
|
|
215
|
+
break;
|
|
216
|
+
case 'GenericTypeAnnotation' as any: // TODO verify schema - Getting this type when running codegen on all the built in types
|
|
217
|
+
arrayTemplateArg = 'winrt::Microsoft::ReactNative::JSValue';
|
|
218
|
+
break;
|
|
219
|
+
default:
|
|
220
|
+
throw new Error(`Unhandled type: ${(type.elementType as any).type} - ${JSON.stringify(type.elementType, null, 2)}`);
|
|
221
|
+
}
|
|
222
|
+
|
|
223
|
+
return { type: `std::vector<${arrayTemplateArg}>`, initializer: '{}' };
|
|
224
|
+
}
|
|
225
|
+
case 'ReservedTypeAnnotation':
|
|
226
|
+
if ((type.name as any) !== 'RootTag') {
|
|
227
|
+
throw new Error(`Unhandled ReservedTypeAnnotation: ${type.name}`)
|
|
228
|
+
}
|
|
229
|
+
return { type: 'bool', initializer: '{-1}' };
|
|
230
|
+
default:
|
|
231
|
+
throw new Error(`Unhandled type: ${(type as any).type}`);
|
|
232
|
+
}
|
|
233
|
+
}
|