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 const QAnyStringView classDefName = classDef.className();
219 const QList<QAnyStringView> namespaces = MetaTypesJsonProcessor::namespaces(classDef);
221 QAnyStringView foreignTypeName;
222 bool foreignIsNamespace =
false;
223 bool isConstructible =
false;
224 for (
const ClassInfo &obj : classDef.classInfos()) {
225 const QAnyStringView name = obj.name;
226 const QAnyStringView value = obj.value;
228 if (name == S_DEFAULT_PROPERTY) {
229 if (mode != RelatedType && defaultProp.isEmpty())
234 if (name == S_PARENT_PROPERTY) {
235 if (mode != RelatedType && parentProp.isEmpty())
240 if (name == S_REGISTER_ENUM_CLASSES_UNSCOPED) {
241 if (mode != RelatedType)
242 handleRegisterEnumClassesUnscoped(classDef, value);
246 if (name == S_ADDED_IN_VERSION) {
247 const QTypeRevision revision = handleInMinorVersion(
248 QTypeRevision::fromEncodedVersion(toInt(value)),
249 defaultRevision.majorVersion());
250 revisions.append(revision);
251 if (mode == TopLevel)
252 addedInRevision = revision;
256 if (mode != TopLevel)
259 if (name == S_REMOVED_IN_VERSION) {
260 removedInRevision = handleInMinorVersion(
261 QTypeRevision::fromEncodedVersion(toInt(value)),
262 defaultRevision.majorVersion());
267 if (name == S_ELEMENT) {
269 elementNames.append(classDefName);
270 else if (value != S_ANONYMOUS)
271 elementNames.append(value);
272 }
else if (name == S_CREATABLE) {
273 isCreatable = (value != S_FALSE);
274 }
else if (name == S_CREATION_METHOD) {
275 isStructured = (value == S_STRUCTURED);
276 isConstructible = isStructured || (value == S_CONSTRUCT);
277 }
else if (name == S_ATTACHED) {
278 if (
const FoundType attached = collectRelated(
279 value, types, foreign, defaultRevision, namespaces)) {
280 attachedType = attached.select(classDef,
"Attached").qualifiedClassName();
282 }
else if (name == S_EXTENDED) {
283 if (
const FoundType extension = collectRelated(
284 value, types, foreign, defaultRevision, namespaces)) {
285 javaScriptExtensionType = extension.javaScript.qualifiedClassName();
286 nativeExtensionType = extension.native.qualifiedClassName();
288 }
else if (name == S_EXTENSION_IS_JAVA_SCRIPT) {
290 extensionIsJavaScript =
true;
291 }
else if (name == S_EXTENSION_IS_NAMESPACE) {
293 extensionIsNamespace =
true;
294 }
else if (name == S_SEQUENCE) {
295 if (
const FoundType element = collectRelated(
296 value, types, foreign, defaultRevision, namespaces)) {
297 sequenceValueType = element.select(classDef,
"Sequence value").qualifiedClassName();
300 sequenceValueType = value;
302 }
else if (name == S_SINGLETON) {
305 }
else if (name == S_FOREIGN) {
306 foreignTypeName = value;
307 }
else if (name == S_FOREIGN_IS_NAMESPACE) {
308 foreignIsNamespace = (value == S_TRUE);
309 }
else if (name == S_PRIMITIVE_ALIAS) {
310 primitiveAliases.append(value);
311 }
else if (name == S_ROOT) {
312 isRootClass = (value == S_TRUE);
313 }
else if (name == S_HAS_CUSTOM_PARSER) {
315 hasCustomParser =
true;
316 }
else if (name == S_DEFERRED_PROPERTY_NAMES) {
317 deferredNames = split(value, QLatin1StringView(
","));
318 }
else if (name == S_IMMEDIATE_PROPERTY_NAMES) {
319 immediateNames = split(value, QLatin1StringView(
","));
323 if (addedInRevision.isValid() && !elementNames.isEmpty())
324 revisions.append(addedInRevision);
328 const bool isNamespace = foreignIsNamespace || classDef.kind() == MetaType::Kind::Namespace;
331 if (!foreignTypeName.isEmpty()) {
333 if (
const FoundType found = findType(foreign, types, foreignTypeName, namespaces)) {
334 resolved = found.select(classDef,
"Foreign");
342 for (
const ClassInfo &obj : resolved.classInfos()) {
343 const QAnyStringView foreignName = obj.name;
344 const QAnyStringView foreignValue = obj.value;
345 if (defaultProp.isEmpty() && foreignName == S_DEFAULT_PROPERTY) {
346 defaultProp = foreignValue;
347 }
else if (parentProp.isEmpty() && foreignName == S_PARENT_PROPERTY) {
348 parentProp = foreignValue;
349 }
else if (foreignName == S_REGISTER_ENUM_CLASSES_UNSCOPED) {
350 handleRegisterEnumClassesUnscoped(resolved, foreignValue);
351 }
else if (foreignName == S_ATTACHED) {
352 if (
const FoundType attached = collectRelated(
353 foreignValue, types, foreign, defaultRevision, namespaces)) {
354 attachedType = attached.select(resolved,
"Attached").qualifiedClassName();
356 }
else if (foreignName == S_EXTENDED) {
357 if (
const FoundType extension = collectRelated(
358 foreignValue, types, foreign, defaultRevision, namespaces)) {
359 nativeExtensionType = extension.native.qualifiedClassName();
360 javaScriptExtensionType = extension.javaScript.qualifiedClassName();
362 }
else if (foreignName == S_EXTENSION_IS_JAVA_SCRIPT) {
363 if (foreignValue == S_TRUE)
364 extensionIsJavaScript =
true;
365 }
else if (foreignName == S_EXTENSION_IS_NAMESPACE) {
366 if (foreignValue == S_TRUE)
367 extensionIsNamespace =
true;
368 }
else if (foreignName == S_SEQUENCE) {
369 if (
const FoundType element = collectRelated(
370 foreignValue, types, foreign, defaultRevision, namespaces)) {
372 = element.select(resolved,
"Sequence value").qualifiedClassName();
377 className = foreignTypeName;
383 if (file.isEmpty()) {
385 file = target.inputFile();
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;