use of com.intellij.javascript.flex.mxml.schema.ClassBackedElementDescriptor in project intellij-plugins by JetBrains.
the class MxmlWriter method processTagChildren.
boolean processTagChildren(final XmlTag tag, @NotNull final Context context, @Nullable final Context parentContext, final boolean propertiesExpected, @Nullable PropertyKind propertyKind) {
int lengthPosition = propertyKind != null && propertyKind.isList() ? out.allocateShort() : 0;
int explicitContentOccurred = -1;
int validAndStaticChildrenCount = 0;
final XmlTagChild[] children = tag.getValue().getChildren();
// if we process property tag value - if we cannot set value due to invalid content, so, we don't write property,
// otherwise if there is no content, we write explicit null
boolean invalidValue = false;
for (XmlTagChild child : children) {
if (child instanceof XmlTag) {
XmlTag childTag = (XmlTag) child;
XmlElementDescriptor descriptor = childTag.getDescriptor();
if (descriptor == null) {
LOG.warn("Descriptor is null, skip " + child);
invalidValue = true;
continue;
}
assert descriptor != null;
if (descriptor instanceof ClassBackedElementDescriptor) {
final ClassBackedElementDescriptor classBackedDescriptor = (ClassBackedElementDescriptor) descriptor;
if (classBackedDescriptor.isPredefined()) {
if (MxmlUtil.isObjectLanguageTag(tag)) {
// IDEA-73482
processPropertyTag(childTag, new AnyXmlAttributeDescriptorWrapper(descriptor), context);
} else if (descriptor.getQualifiedName().equals(FlexPredefinedTagNames.DECLARATIONS)) {
injectedASWriter.readDeclarations(this, childTag);
}
continue;
} else if (MxmlUtil.isAbstract(classBackedDescriptor)) {
addProblem(child, "abstract.class", classBackedDescriptor.getQualifiedName());
continue;
}
if (explicitContentOccurred == 1) {
LOG.warn("Default content already processed, skip " + child);
continue;
}
if (propertiesExpected && explicitContentOccurred == -1) {
explicitContentOccurred = 0;
final PropertyKind defaultPropertyKind = processDefaultProperty(tag, valueProviderFactory.create(childTag), classBackedDescriptor, children.length, context);
if (defaultPropertyKind == null) {
continue;
} else if (defaultPropertyKind.isList()) {
lengthPosition = out.allocateShort();
propertyKind = defaultPropertyKind;
} else if (defaultPropertyKind == PropertyKind.PRIMITIVE) {
validAndStaticChildrenCount++;
continue;
}
}
if (processClassBackedSubTag(childTag, classBackedDescriptor, context, propertyKind != null && propertyKind.isList())) {
validAndStaticChildrenCount++;
}
} else if (propertiesExpected && descriptor instanceof AnnotationBackedDescriptor) {
AnnotationBackedDescriptor annotationBackedDescriptor = (AnnotationBackedDescriptor) descriptor;
// skip invalid, contiguous child elements already processed and explicit content (i.e. AnnotationBackedDescriptor, property childTag) was occurred
if (explicitContentOccurred == 0) {
explicitContentOccurred = 1;
if (propertyKind != null && propertyKind.isList()) {
endList(validAndStaticChildrenCount, lengthPosition);
}
}
if (childTag.getNamespace().equals(MxmlJSClass.MXML_URI4) && childTag.getLocalName().equals(FlexStateElementNames.STATES)) {
if (childTag.getSubTags().length != 0) {
hasStates = true;
assert parentContext == null;
if (stateWriter == null) {
stateWriter = new StateWriter(writer);
}
stateWriter.readDeclaration(childTag);
}
} else {
processPropertyTag(childTag, annotationBackedDescriptor, context);
}
}
} else if (child instanceof XmlText && !MxmlUtil.containsOnlyWhitespace(child)) {
if (explicitContentOccurred == 1) {
LOG.warn("Default content already processed, skip '" + child.getText().trim() + "'");
continue;
}
if (context.getChildrenType() != null && !context.getChildrenType().equals(JSCommonTypeNames.STRING_CLASS_NAME)) {
LOG.warn("Illegal child type, skip '" + child.getText().trim() + "'");
continue;
}
if (propertiesExpected && explicitContentOccurred == -1) {
explicitContentOccurred = 0;
final XmlElementValueProvider valueProvider = valueProviderFactory.create((XmlText) child);
final PropertyKind defaultPropertyKind = processDefaultProperty(tag, valueProvider, null, children.length, context);
if (defaultPropertyKind == PropertyKind.IGNORE) {
explicitContentOccurred = -1;
continue;
} else if (defaultPropertyKind == null) {
continue;
} else if (defaultPropertyKind.isList()) {
lengthPosition = out.allocateShort();
propertyKind = defaultPropertyKind;
} else if (defaultPropertyKind == PropertyKind.PRIMITIVE) {
validAndStaticChildrenCount++;
continue;
} else {
final ValueWriter valueWriter;
try {
valueWriter = propertyProcessor.processXmlTextAsDefaultPropertyWithComplexType(valueProvider, tag, context);
} catch (InvalidPropertyException e) {
// we don't need any out rollback - nothing is written yet
problemsHolder.add(e);
continue;
}
if (valueWriter == null) {
throw new IllegalArgumentException("unexpected default property kind " + defaultPropertyKind);
} else if (valueWriter == InjectedASWriter.IGNORE) {
continue;
}
}
} else if (propertyKind == PropertyKind.VECTOR) {
LOG.warn("skip " + child + " due to IDEA-73478");
// IDEA-73478, XmlText allowed only for fx:Array, but not for fx:Vector (even with type String)
break;
}
if (propertyKind != null && propertyKind == PropertyKind.COMPLEX) {
invalidValue = true;
LOG.warn("Text is not expected" + child);
} else {
writer.string(((XmlText) child).getValue());
validAndStaticChildrenCount++;
}
}
}
if (propertyKind != null && propertyKind.isList()) {
endList(validAndStaticChildrenCount, lengthPosition);
} else if (!propertiesExpected && validAndStaticChildrenCount == 0) {
if (invalidValue) {
return false;
}
// PropertyAsTagWithCommentedValueAsTag, replace Amf3Types.OBJECT to Amf3Types.NULL
out.putByte(Amf3Types.NULL, out.size() - 1);
}
return true;
}
use of com.intellij.javascript.flex.mxml.schema.ClassBackedElementDescriptor in project intellij-plugins by JetBrains.
the class MxmlWriter method processDefaultProperty.
// descriptor will be null if child is XmlText
@Nullable
private PropertyKind processDefaultProperty(XmlTag parentTag, XmlElementValueProvider valueProvider, @Nullable ClassBackedElementDescriptor descriptor, int childrenLength, @NotNull Context context) {
final ClassBackedElementDescriptor parentDescriptor = (ClassBackedElementDescriptor) parentTag.getDescriptor();
assert parentDescriptor != null;
final AnnotationBackedDescriptor defaultDescriptor = parentDescriptor.getDefaultPropertyDescriptor();
final boolean isXmlText = descriptor == null;
if (defaultDescriptor == null) {
final String className = parentDescriptor.getQualifiedName();
final JSClass jsClass;
if (parentDescriptor instanceof MxmlBackedElementDescriptor) {
jsClass = (JSClass) ActionScriptClassResolver.findClassByQNameStatic(className, parentTag);
} else {
jsClass = (JSClass) parentDescriptor.getDeclaration();
}
final boolean isDirectContainerImpl = className.equals(FlexCommonTypeNames.ICONTAINER);
if (isDirectContainerImpl || ActionScriptClassResolver.isParentClass(jsClass, FlexCommonTypeNames.ICONTAINER)) {
if (isXmlText) {
addProblem(parentTag, "initializer.cannot.be.represented.in.text", parentTag.getLocalName());
return null;
}
if (!isDirectContainerImpl && isHaloNavigator(className, jsClass) && !ActionScriptClassResolver.isParentClass((JSClass) descriptor.getDeclaration(), FlexCommonTypeNames.INAVIGATOR_CONTENT)) {
addProblem(parentTag, "children.must.be", parentTag.getLocalName(), FlexCommonTypeNames.INAVIGATOR_CONTENT);
return null;
}
writer.classOrPropertyName("0");
out.write(AmfExtendedTypes.MX_CONTAINER_CHILDREN);
return PropertyKind.ARRAY;
} else {
// http://youtrack.jetbrains.net/issue/IDEA-66565
addProblem(parentTag, "default.property.not.found", parentTag.getLocalName());
}
} else {
// xmlText as default property with injection, see BindingToDeferredInstanceFromBytesBase
if (isXmlText && writeXmlTextAsDefaultPropertyInjectedValue(parentTag, valueProvider, defaultDescriptor, context)) {
return null;
}
if (defaultDescriptor.getType().equals(JSCommonTypeNames.ARRAY_CLASS_NAME) && defaultDescriptor.getArrayType() != null) {
final String elementType = defaultDescriptor.getArrayType();
context.setChildrenType(elementType);
final boolean isString = elementType.equals(JSCommonTypeNames.STRING_CLASS_NAME);
if (isString) {
if (descriptor != null && !descriptor.getQualifiedName().equals(JSCommonTypeNames.STRING_CLASS_NAME)) {
addProblem(parentTag, "children.must.be", parentTag.getLocalName(), elementType);
return null;
}
} else {
if (isXmlText) {
addProblem(parentTag, "initializer.cannot.be.represented.in.text", parentTag.getLocalName());
return PropertyKind.IGNORE;
}
}
}
writer.classOrPropertyName(defaultDescriptor.getName());
if (defaultDescriptor.isDeferredInstance()) {
writer.newInstance("com.intellij.flex.uiDesigner.flex.DeferredInstanceFromArray", 1, false).typeMarker(Amf3Types.ARRAY);
return PropertyKind.ARRAY;
} else {
final String type = defaultDescriptor.getType();
if (type.equals(JSCommonTypeNames.STRING_CLASS_NAME) || (isXmlText && childrenLength == 1 && (type.equals(JSCommonTypeNames.OBJECT_CLASS_NAME) || type.equals(JSCommonTypeNames.ANY_TYPE)))) {
writeSubstitutedString(valueProvider.getSubstituted());
} else if (defaultDescriptor.contentIsArrayable()) {
out.write(type.equals(JSCommonTypeNames.ARRAY_CLASS_NAME) ? Amf3Types.ARRAY : AmfExtendedTypes.ARRAY_IF_LENGTH_GREATER_THAN_1);
return PropertyKind.ARRAY;
} else if (defaultDescriptor.getArrayType() != null) /* Vector */
{
if (isXmlText) {
LOG.warn("skip " + valueProvider.getElement() + " due to IDEA-73478");
return null;
}
writer.vectorHeader(defaultDescriptor.getArrayType());
return PropertyKind.VECTOR;
} else if (type.equals(JSCommonTypeNames.NUMBER_CLASS_NAME)) {
out.writeAmfDouble(valueProvider.getTrimmed());
} else if (type.equals(JSCommonTypeNames.BOOLEAN_CLASS_NAME)) {
out.writeAmfBoolean(valueProvider.getTrimmed());
} else {
out.write(Amf3Types.OBJECT);
return PropertyKind.COMPLEX;
}
}
}
return PropertyKind.PRIMITIVE;
}
use of com.intellij.javascript.flex.mxml.schema.ClassBackedElementDescriptor in project intellij-plugins by JetBrains.
the class PropertyProcessor method write.
@SuppressWarnings("StatementWithEmptyBody")
@Override
public PropertyKind write(AnnotationBackedDescriptor descriptor, XmlElementValueProvider valueProvider, PrimitiveAmfOutputStream out, BaseWriter writer, boolean isStyle, @Nullable Context parentContext) throws InvalidPropertyException {
final String type = descriptor.getType();
if (isStyle) {
int flags = 0;
if (isEffect()) {
flags |= StyleFlags.EFFECT;
out.write(flags);
out.write(Amf3Types.OBJECT);
return COMPLEX_STYLE;
} else {
out.write(flags);
}
} else if (isEffect()) {
out.write(Amf3Types.OBJECT);
return COMPLEX;
}
if (writeIfPrimitive(valueProvider, type, out, descriptor, isStyle, false)) {
} else if (type.equals(JSCommonTypeNames.ARRAY_CLASS_NAME)) {
if (!descriptor.isRichTextContent() && valueProvider instanceof XmlAttributeValueProvider && isInlineArray(valueProvider.getTrimmed())) {
writeInlineArray(valueProvider);
} else {
out.write(Amf3Types.ARRAY);
return ARRAY;
}
} else if (descriptor.getArrayType() != null) {
if (valueProvider.getElement() instanceof XmlTag) {
final XmlTag propertyTag = (XmlTag) valueProvider.getElement();
final XmlTag[] subTags = propertyTag.getSubTags();
if (subTags.length == 1) {
final XmlTag contentTag = subTags[0];
final XmlElementDescriptor contentTagDescriptor = contentTag.getDescriptor();
if (contentTagDescriptor instanceof ClassBackedElementDescriptor && CodeContext.AS3_VEC_VECTOR_QUALIFIED_NAME.equals(contentTagDescriptor.getQualifiedName())) {
if (!mxmlWriter.processMxmlVector(contentTag, parentContext, false)) {
throw new InvalidPropertyException(contentTag, "invalid.vector.value");
}
return isStyle ? PRIMITIVE_STYLE : PRIMITIVE;
}
}
}
writer.vectorHeader(descriptor.getArrayType());
return VECTOR;
} else if (type.equals(JSCommonTypeNames.OBJECT_CLASS_NAME) || type.equals(JSCommonTypeNames.ANY_TYPE)) {
final PropertyKind propertyKind = writeUntypedPropertyValue(out, valueProvider, descriptor, isStyle, parentContext);
if (propertyKind != null) {
return propertyKind;
}
} else if (type.equals(FlexCommonTypeNames.IFACTORY)) {
writeClassFactory(valueProvider);
} else {
out.write(Amf3Types.OBJECT);
return isStyle ? COMPLEX_STYLE : COMPLEX;
}
return isStyle ? PRIMITIVE_STYLE : PRIMITIVE;
}
use of com.intellij.javascript.flex.mxml.schema.ClassBackedElementDescriptor in project intellij-plugins by JetBrains.
the class PropertyProcessor method processXmlTextAsDefaultPropertyWithComplexType.
public ValueWriter processXmlTextAsDefaultPropertyWithComplexType(XmlElementValueProvider valueProvider, XmlTag parent, Context context) throws InvalidPropertyException {
ClassBackedElementDescriptor classBackedElementDescriptor = (ClassBackedElementDescriptor) parent.getDescriptor();
LOG.assertTrue(classBackedElementDescriptor != null);
AnnotationBackedDescriptor defaultPropertyDescriptor = classBackedElementDescriptor.getDefaultPropertyDescriptor();
LOG.assertTrue(defaultPropertyDescriptor != null);
return processInjected(valueProvider, defaultPropertyDescriptor, defaultPropertyDescriptor.isStyle(), context);
}
Aggregations