blob: c23763fcce3ca2c94f4e5c08add67ba446b16f17 [file] [log] [blame]
/*
* Copyright 2014, Google Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google Inc. nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package org.jf.smalidea.psi.impl;
import com.intellij.lang.ASTNode;
import com.intellij.openapi.command.WriteCommandAction;
import com.intellij.psi.PsiElement;
import com.intellij.psi.PsiModifier.ModifierConstant;
import com.intellij.psi.PsiModifierList;
import com.intellij.psi.PsiModifierListOwner;
import com.intellij.psi.StubBasedPsiElement;
import com.intellij.psi.impl.source.tree.Factory;
import com.intellij.psi.impl.source.tree.TreeElement;
import com.intellij.util.IncorrectOperationException;
import org.jetbrains.annotations.NonNls;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
import org.jf.dexlib2.AccessFlags;
import org.jf.smalidea.SmaliTokens;
import org.jf.smalidea.psi.iface.SmaliModifierListOwner;
import org.jf.smalidea.psi.stub.SmaliModifierListStub;
import org.jf.smalidea.psi.stub.element.SmaliModifierListElementType;
import javax.annotation.Nonnull;
public class SmaliModifierList extends SmaliStubBasedPsiElement<SmaliModifierListStub>
implements StubBasedPsiElement<SmaliModifierListStub>, PsiModifierList {
public SmaliModifierList(@NotNull ASTNode node) {
super(node);
}
public SmaliModifierList(@NotNull SmaliModifierListStub stub) {
super(stub, SmaliModifierListElementType.INSTANCE);
}
public int getAccessFlags() {
SmaliModifierListStub stub = getStub();
if (stub != null) {
return stub.getAccessFlags();
}
int flags = 0;
for (PsiElement accessSpec: findChildrenByType(SmaliTokens.ACCESS_SPEC)) {
AccessFlags flag = AccessFlags.getAccessFlag(accessSpec.getText());
if (flag != null) {
flags |= flag.getValue();
}
}
return flags;
}
@Override public boolean hasModifierProperty(@ModifierConstant @NotNull @NonNls String name) {
return hasExplicitModifier(name);
}
@Override public boolean hasExplicitModifier(@ModifierConstant @NotNull @NonNls String name) {
SmaliModifierListStub stub = getStub();
if (stub != null) {
AccessFlags flag = AccessFlags.getAccessFlag(name);
if (flag == null) {
return false;
}
return (stub.getAccessFlags() & flag.getValue()) != 0;
}
for (PsiElement accessSpec: findChildrenByType(SmaliTokens.ACCESS_SPEC)) {
if (accessSpec.getText().equals(name)) {
return true;
}
}
return false;
}
@Override
public void setModifierProperty(@ModifierConstant @NotNull @NonNls String name, boolean addModifier)
throws IncorrectOperationException {
if (addModifier) {
final TreeElement leaf = Factory.createSingleLeafElement(SmaliTokens.ACCESS_SPEC, name, null, getManager());
new WriteCommandAction.Simple(getProject(), getContainingFile()) {
@Override protected void run() throws Throwable {
addInternal(leaf, leaf, null, null);
}
}.execute();
} else {
final PsiElement accessSpec = getAccessFlagElement(name);
if (accessSpec != null) {
new WriteCommandAction.Simple(getProject(), getContainingFile()) {
@Override protected void run() throws Throwable {
accessSpec.delete();
}
}.execute();
}
}
}
@Override
public void checkSetModifierProperty(@ModifierConstant @NotNull @NonNls String name, boolean addModifier)
throws IncorrectOperationException {
}
@Nonnull
private SmaliModifierListOwner getParentForAnnotations() {
SmaliModifierListOwner parent = (SmaliModifierListOwner)getStubOrPsiParentOfType(PsiModifierListOwner.class);
assert parent != null;
return parent;
}
@NotNull @Override public SmaliAnnotation[] getAnnotations() {
return getParentForAnnotations().getAnnotations();
}
@NotNull @Override public SmaliAnnotation[] getApplicableAnnotations() {
return getParentForAnnotations().getApplicableAnnotations();
}
@Nullable @Override public SmaliAnnotation findAnnotation(@NotNull @NonNls String qualifiedName) {
return getParentForAnnotations().findAnnotation(qualifiedName);
}
@NotNull @Override public SmaliAnnotation addAnnotation(@NotNull @NonNls String qualifiedName) {
return getParentForAnnotations().addAnnotation(qualifiedName);
}
@Nullable public PsiElement getAccessFlagElement(@NotNull String accessFlag) {
for (PsiElement accessSpec: findChildrenByType(SmaliTokens.ACCESS_SPEC)) {
if (accessSpec.getText().equals(accessFlag)) {
return accessSpec;
}
}
return null;
}
}