blob: 61513d1e4c4fff47b0f61c21e2db9e3b6673cad3 [file] [log] [blame]
/*
* Copyright 2000-2013 JetBrains s.r.o.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.jetbrains.python.psi.impl;
import com.intellij.lang.ASTNode;
import com.intellij.psi.PsiElement;
import com.intellij.psi.PsiPolyVariantReference;
import com.intellij.psi.tree.IElementType;
import com.intellij.psi.util.PsiTreeUtil;
import com.intellij.psi.util.QualifiedName;
import com.intellij.util.IncorrectOperationException;
import com.jetbrains.python.PyElementTypes;
import com.jetbrains.python.PyNames;
import com.jetbrains.python.psi.PyBinaryExpression;
import com.jetbrains.python.psi.PyElementType;
import com.jetbrains.python.psi.PyElementVisitor;
import com.jetbrains.python.psi.PyExpression;
import com.jetbrains.python.psi.impl.references.PyOperatorReference;
import com.jetbrains.python.psi.resolve.PyResolveContext;
import com.jetbrains.python.psi.types.*;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
/**
* @author yole
*/
public class PyBinaryExpressionImpl extends PyElementImpl implements PyBinaryExpression {
public PyBinaryExpressionImpl(ASTNode astNode) {
super(astNode);
}
@Override
protected void acceptPyVisitor(PyElementVisitor pyVisitor) {
pyVisitor.visitPyBinaryExpression(this);
}
@Nullable
public PyExpression getLeftExpression() {
return PsiTreeUtil.getChildOfType(this, PyExpression.class);
}
public PyExpression getRightExpression() {
return PsiTreeUtil.getNextSiblingOfType(getLeftExpression(), PyExpression.class);
}
@Nullable
public PyElementType getOperator() {
final PsiElement psiOperator = getPsiOperator();
return psiOperator != null ? (PyElementType)psiOperator.getNode().getElementType() : null;
}
@Nullable
public PsiElement getPsiOperator() {
ASTNode node = getNode();
final ASTNode child = node.findChildByType(PyElementTypes.BINARY_OPS);
if (child != null) return child.getPsi();
return null;
}
public boolean isOperator(String chars) {
ASTNode child = getNode().getFirstChildNode();
StringBuilder buf = new StringBuilder();
while (child != null) {
IElementType elType = child.getElementType();
if (elType instanceof PyElementType && PyElementTypes.BINARY_OPS.contains(elType)) {
buf.append(child.getText());
}
child = child.getTreeNext();
}
return buf.toString().equals(chars);
}
@Nullable
public PyExpression getOppositeExpression(PyExpression expression) throws IllegalArgumentException {
PyExpression right = getRightExpression();
PyExpression left = getLeftExpression();
if (expression.equals(left)) {
return right;
}
if (expression.equals(right)) {
return left;
}
throw new IllegalArgumentException("expression " + expression + " is neither left exp or right exp");
}
@Override
public void deleteChildInternal(@NotNull ASTNode child) {
PyExpression left = getLeftExpression();
PyExpression right = getRightExpression();
if (left == child.getPsi() && right != null) {
replace(right);
}
else if (right == child.getPsi() && left != null) {
replace(left);
}
else {
throw new IncorrectOperationException("Element " + child.getPsi() + " is neither left expression or right expression");
}
}
@NotNull
@Override
public PsiPolyVariantReference getReference() {
return getReference(PyResolveContext.noImplicits());
}
@NotNull
@Override
public PsiPolyVariantReference getReference(PyResolveContext context) {
return new PyOperatorReference(this, context);
}
public PyType getType(@NotNull TypeEvalContext context, @NotNull TypeEvalContext.Key key) {
if (isOperator("and") || isOperator("or")) {
final PyExpression left = getLeftExpression();
final PyType leftType = left != null ? context.getType(left) : null;
final PyExpression right = getRightExpression();
final PyType rightType = right != null ? context.getType(right) : null;
if (leftType == null && rightType == null) {
return null;
}
return PyUnionType.union(leftType, rightType);
}
final PyTypeChecker.AnalyzeCallResults results = PyTypeChecker.analyzeCall(this, context);
if (results != null) {
final PyType type = results.getCallable().getCallType(context, this);
if (!PyTypeChecker.isUnknown(type) && !(type instanceof PyNoneType)) {
return type;
}
}
if (PyNames.COMPARISON_OPERATORS.contains(getReferencedName())) {
return PyBuiltinCache.getInstance(this).getBoolType();
}
return null;
}
@Override
public PyExpression getQualifier() {
return getLeftExpression();
}
@Nullable
@Override
public QualifiedName asQualifiedName() {
return PyPsiUtils.asQualifiedName(this);
}
@Override
public boolean isQualified() {
return getQualifier() != null;
}
@Override
public String getReferencedName() {
final PyElementType t = getOperator();
return t != null ? t.getSpecialMethodName() : null;
}
@Override
public ASTNode getNameElement() {
final PsiElement op = getPsiOperator();
return op != null ? op.getNode() : null;
}
}