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 if (value == classDefName)
294 isSelfExtension =
true;
295 }
else if (name == S_EXTENSION_IS_JAVA_SCRIPT) {
297 extensionIsJavaScript =
true;
298 }
else if (name == S_EXTENSION_IS_NAMESPACE) {
300 extensionIsNamespace =
true;
301 }
else if (name == S_SEQUENCE) {
302 if (
const FoundType element = collectRelated(
303 value, types, foreign, defaultRevision, namespaces)) {
304 sequenceValueType = element.select(classDef,
"Sequence value").qualifiedClassName();
307 sequenceValueType = value;
309 }
else if (name == S_SINGLETON) {
312 }
else if (name == S_FOREIGN) {
313 foreignTypeName = value;
314 }
else if (name == S_FOREIGN_IS_NAMESPACE) {
315 foreignIsNamespace = (value == S_TRUE);
316 }
else if (name == S_PRIMITIVE_ALIAS) {
317 primitiveAliases.append(value);
318 }
else if (name == S_ROOT) {
319 isRootClass = (value == S_TRUE);
320 }
else if (name == S_HAS_CUSTOM_PARSER) {
322 hasCustomParser =
true;
323 }
else if (name == S_DEFERRED_PROPERTY_NAMES) {
324 deferredNames = split(value, QLatin1StringView(
","));
325 }
else if (name == S_IMMEDIATE_PROPERTY_NAMES) {
326 immediateNames = split(value, QLatin1StringView(
","));
330 if (addedInRevision.isValid() && !elementNames.isEmpty())
331 revisions.append(addedInRevision);
335 const bool isNamespace = foreignIsNamespace || classDef.kind() == MetaType::Kind::Namespace;
338 if (!foreignTypeName.isEmpty()) {
340 if (
const FoundType found = findType(foreign, types, foreignTypeName, namespaces)) {
341 resolved = found.select(classDef,
"Foreign");
349 for (
const ClassInfo &obj : resolved.classInfos()) {
350 const QAnyStringView foreignName = obj.name;
351 const QAnyStringView foreignValue = obj.value;
352 if (defaultProp.isEmpty() && foreignName == S_DEFAULT_PROPERTY) {
353 defaultProp = foreignValue;
354 }
else if (parentProp.isEmpty() && foreignName == S_PARENT_PROPERTY) {
355 parentProp = foreignValue;
356 }
else if (foreignName == S_REGISTER_ENUM_CLASSES_UNSCOPED) {
357 handleRegisterEnumClassesUnscoped(resolved, foreignValue);
358 }
else if (foreignName == S_ATTACHED) {
359 if (
const FoundType attached = collectRelated(
360 foreignValue, types, foreign, defaultRevision, namespaces)) {
361 attachedType = attached.select(resolved,
"Attached").qualifiedClassName();
363 }
else if (foreignName == S_EXTENDED) {
364 if (
const FoundType extension = collectRelated(
365 foreignValue, types, foreign, defaultRevision, namespaces)) {
366 nativeExtensionType = extension.native.qualifiedClassName();
367 javaScriptExtensionType = extension.javaScript.qualifiedClassName();
369 }
else if (foreignName == S_EXTENSION_IS_JAVA_SCRIPT) {
370 if (foreignValue == S_TRUE)
371 extensionIsJavaScript =
true;
372 }
else if (foreignName == S_EXTENSION_IS_NAMESPACE) {
373 if (foreignValue == S_TRUE)
374 extensionIsNamespace =
true;
375 }
else if (foreignName == S_SEQUENCE) {
376 if (
const FoundType element = collectRelated(
377 foreignValue, types, foreign, defaultRevision, namespaces)) {
379 = element.select(resolved,
"Sequence value").qualifiedClassName();
384 className = foreignTypeName;
390 if (mode == RelatedType || !elementNames.isEmpty()) {
391 collectExtraVersions(resolved.properties(), revisions);
392 collectExtraVersions(resolved.methods(), revisions);
393 collectExtraVersions(resolved.sigs(), revisions);
396 collectSuperClasses(resolved, types, foreign, mode, defaultRevision);
403 collectInterfaces(resolved);
405 if (!addedInRevision.isValid()) {
406 addedInRevision = defaultRevision;
408 if (addedInRevision <= defaultRevision
409 && (!removedInRevision.isValid() || defaultRevision < removedInRevision)) {
410 revisions.append(defaultRevision);
413 std::sort(revisions.begin(), revisions.end());
414 const auto end = std::unique(revisions.begin(), revisions.end());
415 revisions.erase(QList<QTypeRevision>::const_iterator(end), revisions.constEnd());
418 if (className.isEmpty() && !resolved.isEmpty())
419 className = resolved.qualifiedClassName();
421 if (!sequenceValueType.isEmpty()) {
423 accessSemantics = DotQmltypes::S_SEQUENCE;
424 }
else if (isNamespace) {
426 accessSemantics = DotQmltypes::S_NONE;
427 }
else if (resolved.kind() == MetaType::Kind::Object) {
428 accessSemantics = DotQmltypes::S_REFERENCE;
433 if (elementNames.isEmpty()) {
436 accessSemantics = DotQmltypes::S_VALUE;
439 for (
auto elementName = elementNames.begin(); elementName != elementNames.end();) {
440 if (elementName->isEmpty() || elementName->front().isLower()) {
443 accessSemantics = DotQmltypes::S_VALUE;
452 <<
"Refusing to generate non-lowercase name"
453 << *elementName <<
"for unknown foreign type";
454 elementName = elementNames.erase(elementName);
456 if (elementNames.isEmpty()) {
459 accessSemantics = DotQmltypes::S_NONE;
463 }
else if (resolved.kind() == MetaType::Kind::Gadget) {
464 accessSemantics = DotQmltypes::S_VALUE;
466 accessSemantics = DotQmltypes::S_NONE;