use of org.apache.flex.compiler.mxml.IMXMLTagAttributeData in project vscode-nextgenas by BowlerHatLLC.
the class ActionScriptTextDocumentService method mxmlReferences.
private CompletableFuture<List<? extends Location>> mxmlReferences(ReferenceParams params, IMXMLTagData offsetTag) {
IDefinition definition = getDefinitionForMXMLNameAtOffset(offsetTag, currentOffset);
if (definition != null) {
if (isInsideTagPrefix(offsetTag, currentOffset)) {
//ignore the tag's prefix
return CompletableFuture.completedFuture(Collections.emptyList());
}
ArrayList<Location> result = new ArrayList<>();
referencesForDefinition(definition, result);
return CompletableFuture.completedFuture(result);
}
//finally, check if we're looking for references to a tag's id
IMXMLTagAttributeData attributeData = getMXMLTagAttributeWithValueAtOffset(offsetTag, currentOffset);
if (attributeData == null || !attributeData.getName().equals(IMXMLLanguageConstants.ATTRIBUTE_ID)) {
//definition referenced at the current position.
return CompletableFuture.completedFuture(Collections.emptyList());
}
Path path = LanguageServerUtils.getPathFromLanguageServerURI(params.getTextDocument().getUri());
if (path == null) {
//this probably shouldn't happen, but check just to be safe
return CompletableFuture.completedFuture(Collections.emptyList());
}
ICompilationUnit unit = getCompilationUnit(path);
Collection<IDefinition> definitions = null;
try {
definitions = unit.getFileScopeRequest().get().getExternallyVisibleDefinitions();
} catch (Exception e) {
//safe to ignore
}
if (definitions == null || definitions.size() == 0) {
return CompletableFuture.completedFuture(Collections.emptyList());
}
IClassDefinition classDefinition = null;
for (IDefinition currentDefinition : definitions) {
if (currentDefinition instanceof IClassDefinition) {
classDefinition = (IClassDefinition) currentDefinition;
break;
}
}
if (classDefinition == null) {
//this probably shouldn't happen, but check just to be safe
return CompletableFuture.completedFuture(Collections.emptyList());
}
IASScope scope = classDefinition.getContainedScope();
for (IDefinition currentDefinition : scope.getAllLocalDefinitions()) {
if (currentDefinition.getBaseName().equals(attributeData.getRawValue())) {
definition = currentDefinition;
break;
}
}
if (definition == null) {
//definition referenced at the current position.
return CompletableFuture.completedFuture(Collections.emptyList());
}
ArrayList<Location> result = new ArrayList<>();
referencesForDefinition(definition, result);
return CompletableFuture.completedFuture(result);
}
use of org.apache.flex.compiler.mxml.IMXMLTagAttributeData in project vscode-nextgenas by BowlerHatLLC.
the class ActionScriptTextDocumentService method mxmlCompletion.
private CompletableFuture<CompletionList> mxmlCompletion(TextDocumentPositionParams position, IMXMLTagData offsetTag) {
CompletionList result = new CompletionList();
result.setIsIncomplete(false);
result.setItems(new ArrayList<>());
if (isInXMLComment(position)) {
//if we're inside a comment, no completion!
return CompletableFuture.completedFuture(result);
}
IMXMLTagData parentTag = offsetTag.getParentTag();
//for some reason, the attributes list includes the >, but that's not
//what we want here, so check if currentOffset isn't the end of the tag!
boolean isAttribute = offsetTag.isOffsetInAttributeList(currentOffset) && currentOffset < offsetTag.getAbsoluteEnd();
if (isAttribute && offsetTag.isCloseTag()) {
return CompletableFuture.completedFuture(result);
}
//inside <fx:Declarations>
if (isDeclarationsTag(offsetTag)) {
if (!isAttribute) {
autoCompleteTypesForMXML(result);
}
return CompletableFuture.completedFuture(result);
}
IDefinition offsetDefinition = getDefinitionForMXMLTag(offsetTag);
if (offsetDefinition == null) {
IDefinition parentDefinition = null;
if (parentTag != null) {
parentDefinition = getDefinitionForMXMLTag(parentTag);
}
if (parentDefinition != null) {
if (parentDefinition instanceof IClassDefinition) {
IClassDefinition classDefinition = (IClassDefinition) parentDefinition;
String offsetPrefix = offsetTag.getPrefix();
if (offsetPrefix.length() == 0 || parentTag.getPrefix().equals(offsetPrefix)) {
//only add members if the prefix is the same as the
//parent tag. members can't have different prefixes.
//also allow members when we don't have a prefix.
addMembersForMXMLTypeToAutoComplete(classDefinition, parentTag, offsetPrefix.length() == 0, result);
}
if (!isAttribute) {
//tags can't appear in attributes, so skip types
String defaultPropertyName = classDefinition.getDefaultPropertyName(currentProject);
if (defaultPropertyName != null) {
//only add types if the class defines [DefaultProperty]
//metadata
autoCompleteTypesForMXMLFromExistingTag(result, offsetTag);
}
}
} else {
//the parent is something like a property, so matching the
//prefix is not required
autoCompleteTypesForMXMLFromExistingTag(result, offsetTag);
}
return CompletableFuture.completedFuture(result);
} else if (isDeclarationsTag(parentTag)) {
autoCompleteTypesForMXMLFromExistingTag(result, offsetTag);
return CompletableFuture.completedFuture(result);
}
return CompletableFuture.completedFuture(result);
}
if (offsetDefinition instanceof IClassDefinition) {
IMXMLTagAttributeData attribute = getMXMLTagAttributeWithValueAtOffset(offsetTag, currentOffset);
if (attribute != null) {
return mxmlAttributeCompletion(offsetTag, result);
}
IClassDefinition classDefinition = (IClassDefinition) offsetDefinition;
addMembersForMXMLTypeToAutoComplete(classDefinition, offsetTag, !isAttribute, result);
String defaultPropertyName = classDefinition.getDefaultPropertyName(currentProject);
if (defaultPropertyName != null && !isAttribute) {
//if [DefaultProperty] is set, then we can instantiate
//types as child elements
//but we don't want to do that when in an attribute
autoCompleteTypesForMXML(result);
}
return CompletableFuture.completedFuture(result);
}
if (offsetDefinition instanceof IVariableDefinition || offsetDefinition instanceof IEventDefinition || offsetDefinition instanceof IStyleDefinition) {
if (!isAttribute) {
autoCompleteTypesForMXML(result);
}
return CompletableFuture.completedFuture(result);
}
System.err.println("Unknown definition for MXML completion: " + offsetDefinition.getClass());
return CompletableFuture.completedFuture(result);
}
use of org.apache.flex.compiler.mxml.IMXMLTagAttributeData in project vscode-nextgenas by BowlerHatLLC.
the class ActionScriptTextDocumentService method getOffsetMXMLTag.
private IMXMLTagData getOffsetMXMLTag(TextDocumentIdentifier textDocument, Position position) {
namespaceStartIndex = -1;
namespaceEndIndex = -1;
String uri = textDocument.getUri();
if (!uri.endsWith(MXML_EXTENSION)) {
// don't try to parse ActionScript files as MXML
return null;
}
currentOffset = -1;
importStartIndex = -1;
importEndIndex = -1;
Path path = LanguageServerUtils.getPathFromLanguageServerURI(uri);
if (path == null) {
return null;
}
if (!isInWorkspaceOrSourcePath(path)) {
//the path must be in the workspace or source-path
return null;
}
String code;
if (sourceByPath.containsKey(path)) {
code = sourceByPath.get(path);
} else {
System.err.println("Could not find source " + path.toAbsolutePath().toString());
System.err.println(sourceByPath.keySet().size());
return null;
}
//need to ensure that the compilation unit exists, even though we don't
//use it directly
ICompilationUnit unit = getCompilationUnit(path);
if (unit == null) {
//should have been logged already
return null;
}
IMXMLDataManager mxmlDataManager = currentWorkspace.getMXMLDataManager();
MXMLData mxmlData = (MXMLData) mxmlDataManager.get(fileSpecGetter.getFileSpecification(path.toAbsolutePath().toString()));
if (mxmlData == null) {
return null;
}
currentOffset = lineAndCharacterToOffset(new StringReader(code), position.getLine(), position.getCharacter());
if (currentOffset == -1) {
System.err.println("Could not find code at position " + position.getLine() + ":" + position.getCharacter() + " in file " + path.toAbsolutePath().toString());
return null;
}
//calculate the location for automatically generated xmlns tags
IMXMLTagData rootTag = mxmlData.getRootTag();
IMXMLTagAttributeData[] attributeDatas = rootTag.getAttributeDatas();
for (IMXMLTagAttributeData attributeData : attributeDatas) {
if (!attributeData.getName().startsWith("xmlns")) {
if (namespaceStartIndex == -1) {
namespaceStartIndex = attributeData.getStart();
namespaceEndIndex = namespaceStartIndex;
}
break;
}
int start = attributeData.getAbsoluteStart();
int end = attributeData.getValueEnd() + 1;
if (brokenMXMLValueEnd) {
end--;
}
if (namespaceStartIndex == -1 || namespaceStartIndex > start) {
namespaceStartIndex = start;
}
if (namespaceEndIndex == -1 || namespaceEndIndex < end) {
namespaceEndIndex = end;
}
}
IMXMLUnitData unitData = mxmlData.findContainmentReferenceUnit(currentOffset);
IMXMLUnitData currentUnitData = unitData;
while (currentUnitData != null) {
if (currentUnitData instanceof IMXMLTagData) {
IMXMLTagData tagData = (IMXMLTagData) currentUnitData;
if (tagData.getXMLName().equals(tagData.getMXMLDialect().resolveScript()) && unitData instanceof IMXMLTextData) {
IMXMLTextData textUnitData = (IMXMLTextData) unitData;
if (textUnitData.getTextType() == IMXMLTextData.TextType.CDATA) {
importStartIndex = textUnitData.getCompilableTextStart();
importEndIndex = textUnitData.getCompilableTextEnd();
}
}
return tagData;
}
currentUnitData = currentUnitData.getParentUnitData();
}
return null;
}
use of org.apache.flex.compiler.mxml.IMXMLTagAttributeData in project vscode-nextgenas by BowlerHatLLC.
the class ActionScriptTextDocumentService method getEmbeddedActionScriptNodeInMXMLTag.
private IASNode getEmbeddedActionScriptNodeInMXMLTag(IMXMLTagData tag, int offset, TextDocumentPositionParams position) {
IMXMLTagAttributeData attributeData = getMXMLTagAttributeWithValueAtOffset(tag, currentOffset);
if (attributeData != null) {
//some attributes can have ActionScript completion, such as
//events and properties with data binding
IClassDefinition tagDefinition = (IClassDefinition) currentProject.resolveXMLNameToDefinition(tag.getXMLName(), tag.getMXMLDialect());
IDefinition attributeDefinition = currentProject.resolveSpecifier(tagDefinition, attributeData.getShortName());
if (attributeDefinition instanceof IEventDefinition) {
IMXMLInstanceNode mxmlNode = (IMXMLInstanceNode) getOffsetNode(position);
IMXMLEventSpecifierNode eventNode = mxmlNode.getEventSpecifierNode(attributeData.getShortName());
for (IASNode asNode : eventNode.getASNodes()) {
IASNode containingNode = getContainingNodeIncludingStart(asNode, currentOffset);
if (containingNode != null) {
return containingNode;
}
}
return eventNode;
} else {
IASNode offsetNode = getOffsetNode(position);
if (offsetNode instanceof IMXMLInstanceNode) {
IMXMLInstanceNode instanceNode = (IMXMLInstanceNode) offsetNode;
IMXMLPropertySpecifierNode propertyNode = instanceNode.getPropertySpecifierNode(attributeData.getShortName());
if (propertyNode != null) {
for (int i = 0, count = propertyNode.getChildCount(); i < count; i++) {
IMXMLNode propertyChild = (IMXMLNode) propertyNode.getChild(i);
if (propertyChild instanceof IMXMLConcatenatedDataBindingNode) {
IMXMLConcatenatedDataBindingNode dataBinding = (IMXMLConcatenatedDataBindingNode) propertyChild;
for (int j = 0, childCount = dataBinding.getChildCount(); j < childCount; j++) {
IASNode dataBindingChild = dataBinding.getChild(i);
if (dataBindingChild.contains(currentOffset) && dataBindingChild instanceof IMXMLSingleDataBindingNode) {
//we'll parse this in a moment, as if it were
//a direct child of the property specifier
propertyChild = (IMXMLSingleDataBindingNode) dataBindingChild;
break;
}
}
}
if (propertyChild instanceof IMXMLSingleDataBindingNode) {
IMXMLSingleDataBindingNode dataBinding = (IMXMLSingleDataBindingNode) propertyChild;
IASNode containingNode = dataBinding.getExpressionNode().getContainingNode(currentOffset);
//we found the correct expression, but due to a bug
//in the compiler its line and column positions
//will be wrong. the resulting completion is too
//quirky, so this feature will be completed later.
//return containingNode;
}
}
}
}
//nothing possible for this attribute
}
}
return null;
}
use of org.apache.flex.compiler.mxml.IMXMLTagAttributeData in project vscode-nextgenas by BowlerHatLLC.
the class ActionScriptTextDocumentService method signatureHelp.
/**
* Displays a function's parameters, including which one is currently
* active. Called automatically by VSCode any time that the user types "(",
* so be sure to check that a function call is actually happening at the
* current position.
*/
@Override
public CompletableFuture<SignatureHelp> signatureHelp(TextDocumentPositionParams position) {
String textDocumentUri = position.getTextDocument().getUri();
if (!textDocumentUri.endsWith(AS_EXTENSION) && !textDocumentUri.endsWith(MXML_EXTENSION)) {
//we couldn't find a node at the specified location
return CompletableFuture.completedFuture(new SignatureHelp(Collections.emptyList(), -1, -1));
}
IASNode offsetNode = null;
IMXMLTagData offsetTag = getOffsetMXMLTag(position);
if (offsetTag != null) {
IMXMLTagAttributeData attributeData = getMXMLTagAttributeWithValueAtOffset(offsetTag, currentOffset);
if (attributeData != null) {
//some attributes can have ActionScript completion, such as
//events and properties with data binding
IClassDefinition tagDefinition = (IClassDefinition) currentProject.resolveXMLNameToDefinition(offsetTag.getXMLName(), offsetTag.getMXMLDialect());
IDefinition attributeDefinition = currentProject.resolveSpecifier(tagDefinition, attributeData.getShortName());
if (attributeDefinition instanceof IEventDefinition) {
IMXMLInstanceNode mxmlNode = (IMXMLInstanceNode) getOffsetNode(position);
IMXMLEventSpecifierNode eventNode = mxmlNode.getEventSpecifierNode(attributeData.getShortName());
for (IASNode asNode : eventNode.getASNodes()) {
IASNode containingNode = getContainingNodeIncludingStart(asNode, currentOffset);
if (containingNode != null) {
offsetNode = containingNode;
}
}
if (offsetNode == null) {
offsetNode = eventNode;
}
}
}
}
if (offsetNode == null) {
offsetNode = getOffsetNode(position);
}
if (offsetNode == null) {
//we couldn't find a node at the specified location
return CompletableFuture.completedFuture(new SignatureHelp(Collections.emptyList(), -1, -1));
}
IFunctionCallNode functionCallNode = getAncestorFunctionCallNode(offsetNode);
IFunctionDefinition functionDefinition = null;
if (functionCallNode != null) {
IExpressionNode nameNode = functionCallNode.getNameNode();
IDefinition definition = nameNode.resolve(currentUnit.getProject());
if (definition instanceof IFunctionDefinition) {
functionDefinition = (IFunctionDefinition) definition;
} else if (definition instanceof IClassDefinition) {
IClassDefinition classDefinition = (IClassDefinition) definition;
functionDefinition = classDefinition.getConstructor();
} else if (nameNode instanceof IIdentifierNode) {
//special case for super()
IIdentifierNode identifierNode = (IIdentifierNode) nameNode;
if (identifierNode.getName().equals(IASKeywordConstants.SUPER)) {
ITypeDefinition typeDefinition = nameNode.resolveType(currentProject);
if (typeDefinition instanceof IClassDefinition) {
IClassDefinition classDefinition = (IClassDefinition) typeDefinition;
functionDefinition = classDefinitionToConstructor(classDefinition);
}
}
}
}
if (functionDefinition != null) {
SignatureHelp result = new SignatureHelp();
List<SignatureInformation> signatures = new ArrayList<>();
SignatureInformation signatureInfo = new SignatureInformation();
signatureInfo.setLabel(getSignatureLabel(functionDefinition));
List<ParameterInformation> parameters = new ArrayList<>();
for (IParameterDefinition param : functionDefinition.getParameters()) {
ParameterInformation paramInfo = new ParameterInformation();
paramInfo.setLabel(param.getBaseName());
parameters.add(paramInfo);
}
signatureInfo.setParameters(parameters);
signatures.add(signatureInfo);
result.setSignatures(signatures);
result.setActiveSignature(0);
int index = getFunctionCallNodeArgumentIndex(functionCallNode, offsetNode);
IParameterDefinition[] params = functionDefinition.getParameters();
int paramCount = params.length;
if (paramCount > 0 && index >= paramCount) {
if (index >= paramCount) {
IParameterDefinition lastParam = params[paramCount - 1];
if (lastParam.isRest()) {
//functions with rest parameters may accept any
//number of arguments, so continue to make the rest
//parameter active
index = paramCount - 1;
} else {
//if there's no rest parameter, and we're beyond the
//final parameter, none should be active
index = -1;
}
}
}
if (index != -1) {
result.setActiveParameter(index);
}
return CompletableFuture.completedFuture(result);
}
return CompletableFuture.completedFuture(new SignatureHelp(Collections.emptyList(), -1, -1));
}
Aggregations