74 const QList<MetaType> &types,
const QList<MetaType> &foreign,
75 const QAnyStringView &name,
const QList<QAnyStringView> &namespaces)
77 const auto tryFindType = [&](QAnyStringView qualifiedName) ->
FoundType {
79 for (
const QList<MetaType> &t : {types, foreign}) {
80 const auto [first, last] = std::equal_range(
81 t.begin(), t.end(), qualifiedName, Compare());
82 for (
auto it = first; it != last; ++it) {
83 Q_ASSERT(it->qualifiedClassName() == qualifiedName);
85 if (it->inputFile().isEmpty()) {
86 if (result.javaScript.isEmpty()) {
87 result.javaScript = *it;
88 result.javaScriptOrigin = (&t == &types)
90 : FoundType::ForeignTypes;
92 warning(result.javaScript)
93 <<
"Multiple JavaScript types called" << qualifiedName <<
"found!";
95 }
else if (result.native.isEmpty()) {
97 result.nativeOrigin = (&t == &types)
99 : FoundType::ForeignTypes;
101 warning(result.native)
102 <<
"Multiple C++ types called" << qualifiedName <<
"found!\n"
103 <<
"(other occurrence in :"
104 << it->inputFile() <<
":" << it->lineNumber() <<
")\n"
105 <<
"This violates the One Definition Rule!";
113 if (startsWith(name, QLatin1String(
"::")))
114 return tryFindType(name.mid(2));
117 for (
int i = 0, end = namespaces.length(); i != end; ++i) {
118 for (
int j = 0; j < end - i; ++j) {
119 namespaces[j].visit([&](
auto data) { qualified.append(data); });
120 qualified.append(QLatin1String(
"::"));
122 name.visit([&](
auto data) { qualified.append(data); });
123 if (
const FoundType found = tryFindType(qualified))
126 qualified.truncate(0);
129 return tryFindType(name);
215 const MetaType &classDef,
const QList<MetaType> &types,
216 const QList<MetaType> &foreign,
CollectMode mode, QTypeRevision defaultRevision)
218 if (file.isEmpty()) {
219 file = classDef.inputFile();
223 const QAnyStringView classDefName = classDef.className();
224 const QList<QAnyStringView> namespaces = MetaTypesJsonProcessor::namespaces(classDef);
226 QAnyStringView foreignTypeName;
227 bool foreignIsNamespace =
false;
228 bool isConstructible =
false;
229 for (
const ClassInfo &obj : classDef.classInfos()) {
230 const QAnyStringView name = obj.name;
231 const QAnyStringView value = obj.value;
233 if (name == S_DEFAULT_PROPERTY) {
234 if (mode != RelatedType && defaultProp.isEmpty())
239 if (name == S_PARENT_PROPERTY) {
240 if (mode != RelatedType && parentProp.isEmpty())
245 if (name == S_REGISTER_ENUM_CLASSES_UNSCOPED) {
246 if (mode != RelatedType)
247 handleRegisterEnumClassesUnscoped(classDef, value);
251 if (name == S_ADDED_IN_VERSION) {
252 const QTypeRevision revision = handleInMinorVersion(
253 QTypeRevision::fromEncodedVersion(toInt(value)),
254 defaultRevision.majorVersion());
255 revisions.append(revision);
256 if (mode == TopLevel)
257 addedInRevision = revision;
261 if (mode != TopLevel)
264 if (name == S_REMOVED_IN_VERSION) {
265 removedInRevision = handleInMinorVersion(
266 QTypeRevision::fromEncodedVersion(toInt(value)),
267 defaultRevision.majorVersion());
272 if (name == S_ELEMENT) {
274 elementNames.append(classDefName);
275 else if (value != S_ANONYMOUS)
276 elementNames.append(value);
277 }
else if (name == S_CREATABLE) {
278 isCreatable = (value != S_FALSE);
279 }
else if (name == S_CREATION_METHOD) {
280 isStructured = (value == S_STRUCTURED);
281 isConstructible = isStructured || (value == S_CONSTRUCT);
282 }
else if (name == S_ATTACHED) {
283 if (
const FoundType attached = collectRelated(
284 value, types, foreign, defaultRevision, namespaces)) {
285 attachedType = attached.select(classDef,
"Attached").qualifiedClassName();
287 }
else if (name == S_EXTENDED) {
288 if (
const FoundType extension = collectRelated(
289 value, types, foreign, defaultRevision, namespaces)) {
290 javaScriptExtensionType = extension.javaScript.qualifiedClassName();
291 nativeExtensionType = extension.native.qualifiedClassName();
293 }
else if (name == S_EXTENSION_IS_JAVA_SCRIPT) {
295 extensionIsJavaScript =
true;
296 }
else if (name == S_EXTENSION_IS_NAMESPACE) {
298 extensionIsNamespace =
true;
299 }
else if (name == S_SEQUENCE) {
300 if (
const FoundType element = collectRelated(
301 value, types, foreign, defaultRevision, namespaces)) {
302 sequenceValueType = element.select(classDef,
"Sequence value").qualifiedClassName();
305 sequenceValueType = value;
307 }
else if (name == S_SINGLETON) {
310 }
else if (name == S_FOREIGN) {
311 foreignTypeName = value;
312 }
else if (name == S_FOREIGN_IS_NAMESPACE) {
313 foreignIsNamespace = (value == S_TRUE);
314 }
else if (name == S_PRIMITIVE_ALIAS) {
315 primitiveAliases.append(value);
316 }
else if (name == S_ROOT) {
317 isRootClass = (value == S_TRUE);
318 }
else if (name == S_HAS_CUSTOM_PARSER) {
320 hasCustomParser =
true;
321 }
else if (name == S_DEFERRED_PROPERTY_NAMES) {
322 deferredNames = split(value, QLatin1StringView(
","));
323 }
else if (name == S_IMMEDIATE_PROPERTY_NAMES) {
324 immediateNames = split(value, QLatin1StringView(
","));
328 if (addedInRevision.isValid() && !elementNames.isEmpty())
329 revisions.append(addedInRevision);
333 const bool isNamespace = foreignIsNamespace || classDef.kind() == MetaType::Kind::Namespace;
336 if (!foreignTypeName.isEmpty()) {
338 if (
const FoundType found = findType(foreign, types, foreignTypeName, namespaces)) {
339 resolved = found.select(classDef,
"Foreign");
347 for (
const ClassInfo &obj : resolved.classInfos()) {
348 const QAnyStringView foreignName = obj.name;
349 const QAnyStringView foreignValue = obj.value;
350 if (defaultProp.isEmpty() && foreignName == S_DEFAULT_PROPERTY) {
351 defaultProp = foreignValue;
352 }
else if (parentProp.isEmpty() && foreignName == S_PARENT_PROPERTY) {
353 parentProp = foreignValue;
354 }
else if (foreignName == S_REGISTER_ENUM_CLASSES_UNSCOPED) {
355 handleRegisterEnumClassesUnscoped(resolved, foreignValue);
356 }
else if (foreignName == S_ATTACHED) {
357 if (
const FoundType attached = collectRelated(
358 foreignValue, types, foreign, defaultRevision, namespaces)) {
359 attachedType = attached.select(resolved,
"Attached").qualifiedClassName();
361 }
else if (foreignName == S_EXTENDED) {
362 if (
const FoundType extension = collectRelated(
363 foreignValue, types, foreign, defaultRevision, namespaces)) {
364 nativeExtensionType = extension.native.qualifiedClassName();
365 javaScriptExtensionType = extension.javaScript.qualifiedClassName();
367 }
else if (foreignName == S_EXTENSION_IS_JAVA_SCRIPT) {
368 if (foreignValue == S_TRUE)
369 extensionIsJavaScript =
true;
370 }
else if (foreignName == S_EXTENSION_IS_NAMESPACE) {
371 if (foreignValue == S_TRUE)
372 extensionIsNamespace =
true;
373 }
else if (foreignName == S_SEQUENCE) {
374 if (
const FoundType element = collectRelated(
375 foreignValue, types, foreign, defaultRevision, namespaces)) {
377 = element.select(resolved,
"Sequence value").qualifiedClassName();
382 className = foreignTypeName;
388 if (mode == RelatedType || !elementNames.isEmpty()) {
389 collectExtraVersions(resolved.properties(), revisions);
390 collectExtraVersions(resolved.methods(), revisions);
391 collectExtraVersions(resolved.sigs(), revisions);
394 collectSuperClasses(resolved, types, foreign, mode, defaultRevision);
401 collectInterfaces(resolved);
403 if (!addedInRevision.isValid()) {
404 addedInRevision = defaultRevision;
406 if (addedInRevision <= defaultRevision
407 && (!removedInRevision.isValid() || defaultRevision < removedInRevision)) {
408 revisions.append(defaultRevision);
411 std::sort(revisions.begin(), revisions.end());
412 const auto end = std::unique(revisions.begin(), revisions.end());
413 revisions.erase(QList<QTypeRevision>::const_iterator(end), revisions.constEnd());
416 if (className.isEmpty() && !resolved.isEmpty())
417 className = resolved.qualifiedClassName();
419 if (!sequenceValueType.isEmpty()) {
421 accessSemantics = DotQmltypes::S_SEQUENCE;
422 }
else if (isNamespace) {
424 accessSemantics = DotQmltypes::S_NONE;
425 }
else if (resolved.kind() == MetaType::Kind::Object) {
426 accessSemantics = DotQmltypes::S_REFERENCE;
431 if (elementNames.isEmpty()) {
434 accessSemantics = DotQmltypes::S_VALUE;
437 for (
auto elementName = elementNames.begin(); elementName != elementNames.end();) {
438 if (elementName->isEmpty() || elementName->front().isLower()) {
441 accessSemantics = DotQmltypes::S_VALUE;
450 <<
"Refusing to generate non-lowercase name"
451 << *elementName <<
"for unknown foreign type";
452 elementName = elementNames.erase(elementName);
454 if (elementNames.isEmpty()) {
457 accessSemantics = DotQmltypes::S_NONE;
461 }
else if (resolved.kind() == MetaType::Kind::Gadget) {
462 accessSemantics = DotQmltypes::S_VALUE;
464 accessSemantics = DotQmltypes::S_NONE;