Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qqmlirbuilder_p.h
Go to the documentation of this file.
1// Copyright (C) 2016 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3#ifndef QQMLIRBUILDER_P_H
4#define QQMLIRBUILDER_P_H
5
6//
7// W A R N I N G
8// -------------
9//
10// This file is not part of the Qt API. It exists purely as an
11// implementation detail. This header file may change from version to
12// version without notice, or even be removed.
13//
14// We mean it.
15//
16
17#include <private/qqmljsast_p.h>
18#include <private/qqmljsengine_p.h>
19#include <private/qv4compiler_p.h>
20#include <private/qv4compileddata_p.h>
21#include <private/qqmljsmemorypool_p.h>
22#include <private/qqmljsfixedpoolarray_p.h>
23#include <private/qv4codegen_p.h>
24#include <private/qv4compiler_p.h>
25#include <QTextStream>
26#include <QCoreApplication>
27
28QT_BEGIN_NAMESPACE
29
30class QQmlPropertyCache;
31class QQmlContextData;
32class QQmlTypeNameCache;
33struct QQmlIRLoader;
34
35namespace QmlIR {
36
37struct Document;
38
39template <typename T>
41{
43 : first(nullptr)
44 , last(nullptr)
45 {}
46
48 T *last;
49 int count = 0;
50
51 int append(T *item) {
52 item->next = nullptr;
53 if (last)
54 last->next = item;
55 else
56 first = item;
57 last = item;
58 return count++;
59 }
60
61 void prepend(T *item) {
62 item->next = first;
63 first = item;
64 if (!last)
65 last = first;
66 ++count;
67 }
68
69 template <typename Sortable, typename Base, Sortable Base::*sortMember>
70 T *findSortedInsertionPoint(T *item) const
71 {
72 T *insertPos = nullptr;
73
74 for (T *it = first; it; it = it->next) {
75 if (!(it->*sortMember <= item->*sortMember))
76 break;
77 insertPos = it;
78 }
79
80 return insertPos;
81 }
82
83 void insertAfter(T *insertionPoint, T *item) {
84 if (!insertionPoint) {
85 prepend(item);
86 } else if (insertionPoint == last) {
87 append(item);
88 } else {
89 item->next = insertionPoint->next;
90 insertionPoint->next = item;
91 ++count;
92 }
93 }
94
95 T *unlink(T *before, T *item) {
96 T * const newNext = item->next;
97
98 if (before)
99 before->next = newNext;
100 else
101 first = newNext;
102
103 if (item == last) {
104 if (newNext)
105 last = newNext;
106 else
107 last = first;
108 }
109
110 --count;
111 return newNext;
112 }
113
114 T *slowAt(int index) const
115 {
116 T *result = first;
117 while (index > 0 && result) {
118 result = result->next;
119 --index;
120 }
121 return result;
122 }
123
124 struct Iterator {
125 // turn Iterator into a proper iterator
127 using value_type = T;
129 using pointer = T *;
130 using reference = T &;
131
132 T *ptr;
133
134 explicit Iterator(T *p) : ptr(p) {}
135
136 T *operator->() {
137 return ptr;
138 }
139
140 const T *operator->() const {
141 return ptr;
142 }
143
144 T &operator*() {
145 return *ptr;
146 }
147
148 const T &operator*() const {
149 return *ptr;
150 }
151
153 ptr = ptr->next;
154 return *this;
155 }
156
158 Iterator that {ptr};
159 ptr = ptr->next;
160 return that;
161 }
162
163 bool operator==(const Iterator &rhs) const {
164 return ptr == rhs.ptr;
165 }
166
167 bool operator!=(const Iterator &rhs) const {
168 return ptr != rhs.ptr;
169 }
170
171 operator T *() { return ptr; }
172 operator const T *() const { return ptr; }
173 };
174
176 Iterator end() { return Iterator(nullptr); }
177
179};
180
181struct Object;
182
184{
186};
187
200
201
203{
205
206 template<typename IdGenerator>
207 static bool initType(
208 QV4::CompiledData::ParameterType *type, const IdGenerator &idGenerator,
209 const QQmlJS::AST::Type *annotation)
210 {
211 using Flag = QV4::CompiledData::ParameterType::Flag;
212
213 if (!annotation)
214 return initType(type, QString(), idGenerator(QString()), Flag::NoFlag);
215
216 const QString typeId = annotation->typeId->toString();
217 const QString typeArgument =
218 annotation->typeArgument ? annotation->typeArgument->toString() : QString();
219
220 if (typeArgument.isEmpty())
221 return initType(type, typeId, idGenerator(typeId), Flag::NoFlag);
222
223 if (typeId == QLatin1String("list"))
224 return initType(type, typeArgument, idGenerator(typeArgument), Flag::List);
225
226 const QString annotationString = annotation->toString();
227 return initType(type, annotationString, idGenerator(annotationString), Flag::NoFlag);
228 }
229
230 static QV4::CompiledData::CommonType stringToBuiltinType(const QString &typeName);
231
232private:
233 static bool initType(
234 QV4::CompiledData::ParameterType *paramType, const QString &typeName,
235 int typeNameIndex, QV4::CompiledData::ParameterType::Flag listFlag);
236};
237
238struct Signal
239{
243
244 QStringList parameterStringList(const QV4::Compiler::StringTableGenerator *stringPool) const;
245
246 int parameterCount() const { return parameters->count; }
247 PoolList<Parameter>::Iterator parametersBegin() const { return parameters->begin(); }
249
251};
252
254{
256};
257
259{
260 // The offset in the source file where the binding appeared. This is used for sorting to ensure
261 // that assignments to list properties are done in the correct order. We use the offset here instead
262 // of Binding::location as the latter has limited precision.
264 // Binding's compiledScriptIndex is index in object's functionsAndExpressions
266};
267
272
273struct Alias : public QV4::CompiledData::Alias
274{
276};
277
282
284{
288 quint32 index = 0; // index in parsedQML::functions
291
292 // --- QQmlPropertyCacheCreator interface
293 const Parameter *formalsBegin() const { return formals.begin(); }
294 const Parameter *formalsEnd() const { return formals.end(); }
295 // ---
296
298};
299
300struct Q_QML_COMPILER_EXPORT CompiledFunctionOrExpression
301{
304
305 QQmlJS::AST::Node *parentNode = nullptr; // FunctionDeclaration, Statement or Expression
306 QQmlJS::AST::Node *node = nullptr; // FunctionDeclaration, Statement or Expression
309};
310
311struct Q_QML_COMPILER_EXPORT Object
312{
313 Q_DECLARE_TR_FUNCTIONS(Object)
314public:
315 quint32 inheritedTypeNameIndex;
317 int id;
321
324
325 const Property *firstProperty() const { return properties->first; }
326 int propertyCount() const { return properties->count; }
327 Alias *firstAlias() const { return aliases->first; }
328 int aliasCount() const { return aliases->count; }
329 const Enum *firstEnum() const { return qmlEnums->first; }
330 int enumCount() const { return qmlEnums->count; }
331 const Signal *firstSignal() const { return qmlSignals->first; }
332 int signalCount() const { return qmlSignals->count; }
333 Binding *firstBinding() const { return bindings->first; }
334 int bindingCount() const { return bindings->count; }
335 const Function *firstFunction() const { return functions->first; }
336 int functionCount() const { return functions->count; }
337 const InlineComponent *inlineComponent() const { return inlineComponents->first; }
338 int inlineComponentCount() const { return inlineComponents->count; }
339 const RequiredPropertyExtraData *requiredPropertyExtraData() const {return requiredPropertyExtraDatas->first; }
340 int requiredPropertyExtraDataCount() const { return requiredPropertyExtraDatas->count; }
342
343 PoolList<Binding>::Iterator bindingsBegin() const { return bindings->begin(); }
344 PoolList<Binding>::Iterator bindingsEnd() const { return bindings->end(); }
345 PoolList<Property>::Iterator propertiesBegin() const { return properties->begin(); }
346 PoolList<Property>::Iterator propertiesEnd() const { return properties->end(); }
347 PoolList<Alias>::Iterator aliasesBegin() const { return aliases->begin(); }
348 PoolList<Alias>::Iterator aliasesEnd() const { return aliases->end(); }
349 PoolList<Enum>::Iterator enumsBegin() const { return qmlEnums->begin(); }
350 PoolList<Enum>::Iterator enumsEnd() const { return qmlEnums->end(); }
351 PoolList<Signal>::Iterator signalsBegin() const { return qmlSignals->begin(); }
352 PoolList<Signal>::Iterator signalsEnd() const { return qmlSignals->end(); }
353 PoolList<Function>::Iterator functionsBegin() const { return functions->begin(); }
354 PoolList<Function>::Iterator functionsEnd() const { return functions->end(); }
355 PoolList<InlineComponent>::Iterator inlineComponentsBegin() const { return inlineComponents->begin(); }
356 PoolList<InlineComponent>::Iterator inlineComponentsEnd() const { return inlineComponents->end(); }
357 PoolList<RequiredPropertyExtraData>::Iterator requiredPropertyExtraDataBegin() const {return requiredPropertyExtraDatas->begin(); }
358 PoolList<RequiredPropertyExtraData>::Iterator requiredPropertyExtraDataEnd() const {return requiredPropertyExtraDatas->end(); }
359
360 // If set, then declarations for this object (and init bindings for these) should go into the
361 // specified object. Used for declarations inside group properties.
363
364 void init(QQmlJS::MemoryPool *pool, int typeNameIndex, int idIndex, const QV4::CompiledData::Location &location);
365
366 QString appendEnum(Enum *enumeration);
367 QString appendSignal(Signal *signal);
368 QString appendProperty(Property *prop, const QString &propertyName, bool isDefaultProperty, const QQmlJS::SourceLocation &defaultToken, QQmlJS::SourceLocation *errorLocation);
369 QString appendAlias(Alias *prop, const QString &aliasName, bool isDefaultProperty, const QQmlJS::SourceLocation &defaultToken, QQmlJS::SourceLocation *errorLocation);
370 void appendFunction(QmlIR::Function *f);
371 void appendInlineComponent(InlineComponent *ic);
372 void appendRequiredPropertyExtraData(RequiredPropertyExtraData *extraData);
373
374 QString appendBinding(Binding *b, bool isListBinding);
375 Binding *findBinding(quint32 nameIndex) const;
376 Binding *unlinkBinding(Binding *before, Binding *binding) { return bindings->unlink(before, binding); }
377 void insertSorted(Binding *b);
378 QString bindingAsString(Document *doc, int scriptIndex) const;
379
382
384 int namedObjectsInComponentCount() const { return namedObjectsInComponent.size(); }
385 const quint32 *namedObjectsInComponentTable() const { return namedObjectsInComponent.begin(); }
386
387 bool hasFlag(QV4::CompiledData::Object::Flag flag) const { return flags & flag; }
388 qint32 objectId() const { return id; }
390
391private:
392 friend struct ::QQmlIRLoader;
393
394 PoolList<Property> *properties;
395 PoolList<Alias> *aliases;
396 PoolList<Enum> *qmlEnums;
397 PoolList<Signal> *qmlSignals;
398 PoolList<Binding> *bindings;
399 PoolList<Function> *functions;
400 PoolList<InlineComponent> *inlineComponents;
401 PoolList<RequiredPropertyExtraData> *requiredPropertyExtraDatas;
402};
403
404struct Q_QML_COMPILER_EXPORT Pragma
405{
406 enum PragmaType
407 {
408 Singleton,
409 Strict,
410 ListPropertyAssignBehavior,
411 ComponentBehavior,
412 FunctionSignatureBehavior,
413 NativeMethodBehavior,
414 ValueTypeBehavior,
415 Translator,
416 };
417
418 enum ListPropertyAssignBehaviorValue
419 {
420 Append,
421 Replace,
422 ReplaceIfNotDefault,
423 };
424
425 enum ComponentBehaviorValue
426 {
427 Unbound,
428 Bound
429 };
430
431 enum FunctionSignatureBehaviorValue
432 {
433 Ignored,
434 Enforced
435 };
436
437 enum NativeMethodBehaviorValue
438 {
439 AcceptThisObject,
440 RejectThisObject
441 };
442
443 enum ValueTypeBehaviorValue
444 {
445 Copy = 0x1,
446 Addressable = 0x2,
447 Assertable = 0x4,
448 };
449 Q_DECLARE_FLAGS(ValueTypeBehaviorValues, ValueTypeBehaviorValue);
450
451 PragmaType type;
452
453 union {
454 ListPropertyAssignBehaviorValue listPropertyAssignBehavior;
455 ComponentBehaviorValue componentBehavior;
456 FunctionSignatureBehaviorValue functionSignatureBehavior;
457 NativeMethodBehaviorValue nativeMethodBehavior;
458 ValueTypeBehaviorValues::Int valueTypeBehavior;
459 uint translationContextIndex;
460 };
461
462 QV4::CompiledData::Location location;
463};
464
465struct Q_QML_COMPILER_EXPORT Document
466{
467 Document(const QString &fileName, const QString &finalUrl, bool debugMode);
468 QString code;
469 QQmlJS::Engine jsParserEngine;
470 QV4::Compiler::Module jsModule;
471 QList<const QV4::CompiledData::Import *> imports;
472 QList<Pragma*> pragmas;
473 QQmlJS::AST::UiProgram *program;
474 QVector<Object*> objects;
475 QV4::Compiler::JSUnitGenerator jsGenerator;
476
477 QQmlRefPointer<QV4::CompiledData::CompilationUnit> javaScriptCompilationUnit;
478
479 bool isSingleton() const {
480 return std::any_of(pragmas.constBegin(), pragmas.constEnd(), [](const Pragma *pragma) {
481 return pragma->type == Pragma::Singleton;
482 });
483 }
484
485 int registerString(const QString &str) { return jsGenerator.registerString(str); }
486 QString stringAt(int index) const { return jsGenerator.stringForIndex(index); }
487
488 int objectCount() const {return objects.size();}
489 Object* objectAt(int i) const {return objects.at(i);}
490};
491
492class Q_QML_COMPILER_EXPORT ScriptDirectivesCollector : public QQmlJS::Directives
493{
494 QmlIR::Document *document;
495 QQmlJS::Engine *engine;
496 QV4::Compiler::JSUnitGenerator *jsGenerator;
497
498public:
499 ScriptDirectivesCollector(QmlIR::Document *doc);
500
501 void pragmaLibrary() override;
502 void importFile(const QString &jsfile, const QString &module, int lineNumber, int column) override;
503 void importModule(const QString &uri, const QString &version, const QString &module, int lineNumber, int column) override;
504};
505
506struct Q_QML_COMPILER_EXPORT IRBuilder : public QQmlJS::AST::Visitor
507{
508 Q_DECLARE_TR_FUNCTIONS(QQmlCodeGenerator)
509public:
510 IRBuilder();
511 bool generateFromQml(const QString &code, const QString &url, Document *output);
512
513 using QQmlJS::AST::Visitor::visit;
514 using QQmlJS::AST::Visitor::endVisit;
515
516 bool visit(QQmlJS::AST::UiImport *ast) override;
517 bool visit(QQmlJS::AST::UiPragma *ast) override;
518 bool visit(QQmlJS::AST::UiProgram *) override;
519 bool visit(QQmlJS::AST::UiArrayBinding *ast) override;
520 bool visit(QQmlJS::AST::UiObjectBinding *ast) override;
521 bool visit(QQmlJS::AST::UiObjectDefinition *ast) override;
522 bool visit(QQmlJS::AST::UiInlineComponent *ast) override;
523 bool visit(QQmlJS::AST::UiEnumDeclaration *ast) override;
524 bool visit(QQmlJS::AST::UiPublicMember *ast) override;
525 bool visit(QQmlJS::AST::UiScriptBinding *ast) override;
526 bool visit(QQmlJS::AST::UiSourceElement *ast) override;
527 bool visit(QQmlJS::AST::UiRequired *ast) override;
528
529 void throwRecursionDepthError() override
530 {
531 recordError(QQmlJS::SourceLocation(),
532 QStringLiteral("Maximum statement or expression depth exceeded"));
533 }
534
535 void accept(QQmlJS::AST::Node *node);
536
537 // returns index in _objects
538 bool defineQMLObject(
539 int *objectIndex, QQmlJS::AST::UiQualifiedId *qualifiedTypeNameId,
540 const QV4::CompiledData::Location &location,
541 QQmlJS::AST::UiObjectInitializer *initializer, Object *declarationsOverride = nullptr);
542
543 bool defineQMLObject(
544 int *objectIndex, QQmlJS::AST::UiObjectDefinition *node,
545 Object *declarationsOverride = nullptr)
546 {
547 const QQmlJS::SourceLocation location = node->qualifiedTypeNameId->firstSourceLocation();
548 return defineQMLObject(
549 objectIndex, node->qualifiedTypeNameId,
550 { location.startLine, location.startColumn }, node->initializer,
551 declarationsOverride);
552 }
553
554 static QString asString(QQmlJS::AST::UiQualifiedId *node);
555 QStringView asStringRef(QQmlJS::AST::Node *node);
556 static QTypeRevision extractVersion(QStringView string);
557 QStringView textRefAt(const QQmlJS::SourceLocation &loc) const
558 { return QStringView(sourceCode).mid(loc.offset, loc.length); }
559 QStringView textRefAt(const QQmlJS::SourceLocation &first,
560 const QQmlJS::SourceLocation &last) const;
561
562 virtual void setBindingValue(QV4::CompiledData::Binding *binding,
563 QQmlJS::AST::Statement *statement, QQmlJS::AST::Node *parentNode);
564 void tryGeneratingTranslationBinding(QStringView base, QQmlJS::AST::ArgumentList *args, QV4::CompiledData::Binding *binding);
565
566 void appendBinding(QQmlJS::AST::UiQualifiedId *name, QQmlJS::AST::Statement *value,
567 QQmlJS::AST::Node *parentNode);
568 void appendBinding(QQmlJS::AST::UiQualifiedId *name, int objectIndex, bool isOnAssignment = false);
569 void appendBinding(const QQmlJS::SourceLocation &qualifiedNameLocation,
570 const QQmlJS::SourceLocation &nameLocation, quint32 propertyNameIndex,
571 QQmlJS::AST::Statement *value, QQmlJS::AST::Node *parentNode);
572 void appendBinding(const QQmlJS::SourceLocation &qualifiedNameLocation,
573 const QQmlJS::SourceLocation &nameLocation, quint32 propertyNameIndex,
574 int objectIndex, bool isListItem = false, bool isOnAssignment = false);
575
576 bool appendAlias(QQmlJS::AST::UiPublicMember *node);
577
578 enum class IsQmlFunction { Yes, No };
579 virtual void registerFunctionExpr(QQmlJS::AST::FunctionExpression *fexp, IsQmlFunction);
580
581 Object *bindingsTarget() const;
582
583 bool setId(const QQmlJS::SourceLocation &idLocation, QQmlJS::AST::Statement *value);
584
585 // resolves qualified name (font.pixelSize for example) and returns the last name along
586 // with the object any right-hand-side of a binding should apply to.
587 bool resolveQualifiedId(QQmlJS::AST::UiQualifiedId **nameToResolve, Object **object, bool onAssignment = false);
588
589 void recordError(const QQmlJS::SourceLocation &location, const QString &description);
590
591 quint32 registerString(const QString &str) const { return jsGenerator->registerString(str); }
592 template <typename _Tp> _Tp *New() { return pool->New<_Tp>(); }
593
594 QString stringAt(int index) const { return jsGenerator->stringForIndex(index); }
595
596 static bool isStatementNodeScript(QQmlJS::AST::Statement *statement);
597 static bool isRedundantNullInitializerForPropertyDeclaration(Property *property, QQmlJS::AST::Statement *statement);
598
599 QString sanityCheckFunctionNames(Object *obj, QQmlJS::SourceLocation *errorLocation);
600
601 QList<QQmlJS::DiagnosticMessage> errors;
602
603 QSet<QString> inlineComponentsNames;
604
605 QList<const QV4::CompiledData::Import *> _imports;
606 QList<Pragma*> _pragmas;
607 QVector<Object*> _objects;
608
609 QV4::CompiledData::TypeReferenceMap _typeReferences;
610
611 Object *_object;
612 Property *_propertyDeclaration;
613
614 QQmlJS::MemoryPool *pool;
615 QString sourceCode;
616 QV4::Compiler::JSUnitGenerator *jsGenerator;
617
618 bool insideInlineComponent = false;
619};
620
621struct Q_QML_COMPILER_EXPORT QmlUnitGenerator
622{
623 void generate(Document &output, const QV4::CompiledData::DependentTypesHasher &dependencyHasher = QV4::CompiledData::DependentTypesHasher());
624
625private:
626 typedef bool (Binding::*BindingFilter)() const;
627 char *writeBindings(char *bindingPtr, const Object *o, BindingFilter filter) const;
628};
629
630struct Q_QML_COMPILER_EXPORT JSCodeGen : public QV4::Compiler::Codegen
631{
632 JSCodeGen(Document *document,
633 QV4::Compiler::CodegenWarningInterface *iface =
634 QV4::Compiler::defaultCodegenWarningInterface(),
635 bool storeSourceLocations = false);
636
637 // Returns mapping from input functions to index in IR::Module::functions / compiledData->runtimeFunctions
638 QVector<int>
639 generateJSCodeForFunctionsAndBindings(const QList<CompiledFunctionOrExpression> &functions);
640
641 bool generateRuntimeFunctions(QmlIR::Object *object);
642
643private:
644 Document *document;
645};
646
647// RegisterStringN ~= std::function<int(QStringView)>
648// FinalizeTranlationData ~= std::function<void(QV4::CompiledData::Binding::ValueType, QV4::CompiledData::TranslationData)>
649/*
650 \internal
651 \a base: name of the potential translation function
652 \a args: arguments to the function call
653 \a registerMainString: Takes the first argument passed to the translation function, and it's
654 result will be stored in a TranslationData's stringIndex for translation bindings and in numbeIndex
655 for string bindings.
656 \a registerCommentString: Takes the comment argument passed to some of the translation functions.
657 Result will be stored in a TranslationData's commentIndex
658 \a finalizeTranslationData: Takes the type of the binding and the previously set up TranslationData
659 */
660template<
661 typename RegisterMainString,
662 typename RegisterCommentString,
663 typename RegisterContextString,
664 typename FinalizeTranslationData>
665void tryGeneratingTranslationBindingBase(QStringView base, QQmlJS::AST::ArgumentList *args,
666 RegisterMainString registerMainString,
667 RegisterCommentString registerCommentString,
668 RegisterContextString registerContextString,
669 FinalizeTranslationData finalizeTranslationData
670 )
671{
672 if (base == QLatin1String("qsTr")) {
673 QV4::CompiledData::TranslationData translationData;
674 translationData.number = -1;
675
676 // empty string
677 translationData.commentIndex = 0;
678
679 // No context (not empty string)
680 translationData.contextIndex = QV4::CompiledData::TranslationData::NoContextIndex;
681
682 if (!args || !args->expression)
683 return; // no arguments, stop
684
685 QStringView translation;
686 if (QQmlJS::AST::StringLiteral *arg1 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression)) {
687 translation = arg1->value;
688 } else {
689 return; // first argument is not a string, stop
690 }
691
692 translationData.stringIndex = registerMainString(translation);
693
694 args = args->next;
695
696 if (args) {
697 QQmlJS::AST::StringLiteral *arg2 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression);
698 if (!arg2)
699 return; // second argument is not a string, stop
700 translationData.commentIndex = registerCommentString(arg2->value);
701
702 args = args->next;
703 if (args) {
704 if (QQmlJS::AST::NumericLiteral *arg3 = QQmlJS::AST::cast<QQmlJS::AST::NumericLiteral *>(args->expression)) {
705 translationData.number = int(arg3->value);
706 args = args->next;
707 } else {
708 return; // third argument is not a translation number, stop
709 }
710 }
711 }
712
713 if (args)
714 return; // too many arguments, stop
715
716 finalizeTranslationData(QV4::CompiledData::Binding::Type_Translation, translationData);
717 } else if (base == QLatin1String("qsTrId")) {
718 QV4::CompiledData::TranslationData translationData;
719 translationData.number = -1;
720
721 // empty string, but unused
722 translationData.commentIndex = 0;
723
724 // No context (not empty string)
725 translationData.contextIndex = QV4::CompiledData::TranslationData::NoContextIndex;
726
727 if (!args || !args->expression)
728 return; // no arguments, stop
729
730 QStringView id;
731 if (QQmlJS::AST::StringLiteral *arg1 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression)) {
732 id = arg1->value;
733 } else {
734 return; // first argument is not a string, stop
735 }
736 translationData.stringIndex = registerMainString(id);
737
738 args = args->next;
739
740 if (args) {
741 if (QQmlJS::AST::NumericLiteral *arg3 = QQmlJS::AST::cast<QQmlJS::AST::NumericLiteral *>(args->expression)) {
742 translationData.number = int(arg3->value);
743 args = args->next;
744 } else {
745 return; // third argument is not a translation number, stop
746 }
747 }
748
749 if (args)
750 return; // too many arguments, stop
751
752 finalizeTranslationData(QV4::CompiledData::Binding::Type_TranslationById, translationData);
753 } else if (base == QLatin1String("QT_TR_NOOP") || base == QLatin1String("QT_TRID_NOOP")) {
754 if (!args || !args->expression)
755 return; // no arguments, stop
756
757 QStringView str;
758 if (QQmlJS::AST::StringLiteral *arg1 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression)) {
759 str = arg1->value;
760 } else {
761 return; // first argument is not a string, stop
762 }
763
764 args = args->next;
765 // QT_TR_NOOP can have a disambiguation string, QT_TRID_NOOP can't
766 if (args && base == QLatin1String("QT_TR_NOOP")) {
767 // we have a disambiguation string; we don't need to do anything with it
768 if (QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression))
769 args = args->next;
770 else // second argument is not a string, stop
771 return;
772 }
773
774 if (args)
775 return; // too many arguments, stop
776
777 QV4::CompiledData::TranslationData translationData;
778 translationData.number = registerMainString(str);
779 finalizeTranslationData(QV4::CompiledData::Binding::Type_String, translationData);
780 } else if (base == QLatin1String("QT_TRANSLATE_NOOP")) {
781 if (!args || !args->expression)
782 return; // no arguments, stop
783
784 args = args->next;
785 if (!args || !args->expression)
786 return; // no second arguments, stop
787
788 QStringView str;
789 if (QQmlJS::AST::StringLiteral *arg2 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression)) {
790 str = arg2->value;
791 } else {
792 return; // first argument is not a string, stop
793 }
794
795 args = args->next;
796 if (args) {
797 // we have a disambiguation string; we don't need to do anything with it
798 if (QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression))
799 args = args->next;
800 else // third argument is not a string, stop
801 return;
802 }
803
804 if (args)
805 return; // too many arguments, stop
806
807 QV4::CompiledData::TranslationData fakeTranslationData;
808 fakeTranslationData.number = registerMainString(str);
809 finalizeTranslationData(QV4::CompiledData::Binding::Type_String, fakeTranslationData);
810 } else if (base == QLatin1String("qsTranslate")) {
811 QV4::CompiledData::TranslationData translationData;
812 translationData.number = -1;
813 translationData.commentIndex = 0; // empty string
814
815 if (!args || !args->next)
816 return; // less than 2 arguments, stop
817
818 QStringView translation;
819 if (QQmlJS::AST::StringLiteral *arg1
820 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression)) {
821 translation = arg1->value;
822 } else {
823 return; // first argument is not a string, stop
824 }
825
826 translationData.contextIndex = registerContextString(translation);
827
828 args = args->next;
829 Q_ASSERT(args);
830
831 QQmlJS::AST::StringLiteral *arg2
832 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression);
833 if (!arg2)
834 return; // second argument is not a string, stop
835 translationData.stringIndex = registerMainString(arg2->value);
836
837 args = args->next;
838 if (args) {
839 QQmlJS::AST::StringLiteral *arg3
840 = QQmlJS::AST::cast<QQmlJS::AST::StringLiteral *>(args->expression);
841 if (!arg3)
842 return; // third argument is not a string, stop
843 translationData.commentIndex = registerCommentString(arg3->value);
844
845 args = args->next;
846 if (args) {
847 if (QQmlJS::AST::NumericLiteral *arg4
848 = QQmlJS::AST::cast<QQmlJS::AST::NumericLiteral *>(args->expression)) {
849 translationData.number = int(arg4->value);
850 args = args->next;
851 } else {
852 return; // fourth argument is not a translation number, stop
853 }
854 }
855 }
856
857 if (args)
858 return; // too many arguments, stop
859
860 finalizeTranslationData(QV4::CompiledData::Binding::Type_Translation, translationData);
861 }
862}
863
864} // namespace QmlIR
865
866QT_END_NAMESPACE
867
868#endif // QQMLIRBUILDER_P_H
void tryGeneratingTranslationBindingBase(QStringView base, QQmlJS::AST::ArgumentList *args, RegisterMainString registerMainString, RegisterCommentString registerCommentString, RegisterContextString registerContextString, FinalizeTranslationData finalizeTranslationData)
static const quint32 emptyStringIndex
#define COMPILE_EXCEPTION(location, desc)
static QStringList astNodeToStringList(QQmlJS::AST::Node *node)
static bool run(IRBuilder *builder, QQmlJS::AST::UiPragma *node, Pragma *pragma)
CompiledFunctionOrExpression * next
QV4::CompiledData::Location location
PoolList< EnumValue >::Iterator enumValuesEnd() const
PoolList< EnumValue > * enumValues
int enumValueCount() const
PoolList< EnumValue >::Iterator enumValuesBegin() const
QV4::CompiledData::Location location
const Parameter * formalsBegin() const
QV4::CompiledData::ParameterType returnType
QQmlJS::FixedPoolArray< Parameter > formals
const Parameter * formalsEnd() const
InlineComponent * next
bool hasFlag(QV4::CompiledData::Object::Flag flag) const
int indexOfDefaultPropertyOrAlias
PoolList< Property >::Iterator propertiesEnd() const
PoolList< Property >::Iterator propertiesBegin() const
int requiredPropertyExtraDataCount() const
QString appendEnum(Enum *enumeration)
PoolList< Signal >::Iterator signalsEnd() const
int inlineComponentCount() const
const quint32 * namedObjectsInComponentTable() const
void appendRequiredPropertyExtraData(RequiredPropertyExtraData *extraData)
PoolList< RequiredPropertyExtraData >::Iterator requiredPropertyExtraDataEnd() const
void insertSorted(Binding *b)
PoolList< Binding >::Iterator bindingsEnd() const
int enumCount() const
PoolList< Function >::Iterator functionsBegin() const
void simplifyRequiredProperties()
QString appendBinding(Binding *b, bool isListBinding)
QV4::CompiledData::Location location
void appendInlineComponent(InlineComponent *ic)
Alias * firstAlias() const
int bindingCount() const
PoolList< Function >::Iterator functionsEnd() const
PoolList< Alias >::Iterator aliasesBegin() const
PoolList< RequiredPropertyExtraData >::Iterator requiredPropertyExtraDataBegin() const
PoolList< Enum >::Iterator enumsEnd() const
qint32 objectId() const
QString appendSignal(Signal *signal)
Binding * firstBinding() const
const Property * firstProperty() const
QString bindingAsString(Document *doc, int scriptIndex) const
const Enum * firstEnum() const
void init(QQmlJS::MemoryPool *pool, int typeNameIndex, int idIndex, const QV4::CompiledData::Location &location)
Object * declarationsOverride
QString appendAlias(Alias *prop, const QString &aliasName, bool isDefaultProperty, const QQmlJS::SourceLocation &defaultToken, QQmlJS::SourceLocation *errorLocation)
int signalCount() const
int functionCount() const
QString appendProperty(Property *prop, const QString &propertyName, bool isDefaultProperty, const QQmlJS::SourceLocation &defaultToken, QQmlJS::SourceLocation *errorLocation)
PoolList< Enum >::Iterator enumsBegin() const
PoolList< CompiledFunctionOrExpression > * functionsAndExpressions
Binding * unlinkBinding(Binding *before, Binding *binding)
const RequiredPropertyExtraData * requiredPropertyExtraData() const
bool hasAliasAsDefaultProperty() const
PoolList< InlineComponent >::Iterator inlineComponentsEnd() const
PoolList< Signal >::Iterator signalsBegin() const
int aliasCount() const
void appendFunction(QmlIR::Function *f)
const Function * firstFunction() const
PoolList< Binding >::Iterator bindingsBegin() const
PoolList< InlineComponent >::Iterator inlineComponentsBegin() const
const InlineComponent * inlineComponent() const
QQmlJS::FixedPoolArray< int > runtimeFunctionIndices
const Signal * firstSignal() const
QV4::CompiledData::Location locationOfIdProperty
QQmlJS::FixedPoolArray< quint32 > namedObjectsInComponent
Binding * findBinding(quint32 nameIndex) const
int propertyCount() const
int namedObjectsInComponentCount() const
PoolList< Alias >::Iterator aliasesEnd() const
static bool initType(QV4::CompiledData::ParameterType *type, const IdGenerator &idGenerator, const QQmlJS::AST::Type *annotation)
static QV4::CompiledData::CommonType stringToBuiltinType(const QString &typeName)
bool operator!=(const Iterator &rhs) const
const T * operator->() const
bool operator==(const Iterator &rhs) const
const T & operator*() const
T * findSortedInsertionPoint(T *item) const
void prepend(T *item)
T * unlink(T *before, T *item)
T * slowAt(int index) const
void insertAfter(T *insertionPoint, T *item)
int append(T *item)
RequiredPropertyExtraData * next
int parameterCount() const
PoolList< Parameter > * parameters
PoolList< Parameter >::Iterator parametersEnd() const
QStringList parameterStringList(const QV4::Compiler::StringTableGenerator *stringPool) const
PoolList< Parameter >::Iterator parametersBegin() const
QV4::CompiledData::Location location