All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.intellij.plugins.intelliLang.inject.xml.XmlLanguageInjector Maven / Gradle / Ivy

Go to download

A packaging of the IntelliJ Community Edition IntelliLang-xml library. This is release number 1 of trunk branch 142.

The newest version!
/*
 * Copyright 2006 Sascha Weinreuter
 *
 * 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 org.intellij.plugins.intelliLang.inject.xml;

import com.intellij.lang.Language;
import com.intellij.lang.injection.MultiHostInjector;
import com.intellij.lang.injection.MultiHostRegistrar;
import com.intellij.openapi.util.Ref;
import com.intellij.openapi.util.TextRange;
import com.intellij.openapi.util.Trinity;
import com.intellij.openapi.util.text.StringUtil;
import com.intellij.patterns.ElementPattern;
import com.intellij.psi.PsiElement;
import com.intellij.psi.PsiElementVisitor;
import com.intellij.psi.PsiFile;
import com.intellij.psi.PsiLanguageInjectionHost;
import com.intellij.psi.impl.source.tree.injected.InjectedLanguageUtil;
import com.intellij.psi.xml.*;
import com.intellij.util.PairProcessor;
import com.intellij.util.PatternValuesIndex;
import com.intellij.util.containers.ContainerUtil;
import gnu.trove.THashMap;
import org.intellij.plugins.intelliLang.Configuration;
import org.intellij.plugins.intelliLang.inject.InjectedLanguage;
import org.intellij.plugins.intelliLang.inject.InjectorUtils;
import org.intellij.plugins.intelliLang.inject.LanguageInjectionSupport;
import org.intellij.plugins.intelliLang.inject.config.AbstractTagInjection;
import org.intellij.plugins.intelliLang.inject.config.BaseInjection;
import org.intellij.plugins.intelliLang.inject.config.InjectionPlace;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

import java.util.*;
import java.util.regex.Pattern;

/**
 * This is the main part of the injection code. The component registers a language injector, the reference provider that
 * supplies completions for language-IDs and regular expression enum-values as well as the Quick Edit action.
 * 

* The injector obtains the static injection configuration for each XML tag, attribute or String literal and also * dynamically computes the prefix/suffix for the language fragment from binary expressions. *

* It also tries to deal with the "glued token" problem by removing or adding whitespace to the prefix/suffix. */ public final class XmlLanguageInjector implements MultiHostInjector { private final Configuration myConfiguration; private volatile Trinity> myXmlIndex; private final LanguageInjectionSupport mySupport; public XmlLanguageInjector(Configuration configuration) { myConfiguration = configuration; mySupport = InjectorUtils.findNotNullInjectionSupport(XmlLanguageInjectionSupport.XML_SUPPORT_ID); } @NotNull public List> elementsToInjectIn() { return Arrays.asList(XmlTag.class, XmlAttributeValue.class); } public void getLanguagesToInject(@NotNull final MultiHostRegistrar registrar, @NotNull PsiElement host) { final XmlElement xmlElement = (XmlElement) host; if (!isInIndex(xmlElement)) return; final TreeSet ranges = new TreeSet(InjectorUtils.RANGE_COMPARATOR); final PsiFile containingFile = xmlElement.getContainingFile(); final Ref unparsableRef = Ref.create(); getInjectedLanguage(xmlElement, unparsableRef, new PairProcessor>>() { public boolean process(final Language language, List> list) { for (Trinity trinity : list) { if (ranges.contains(trinity.third.shiftRight(trinity.first.getTextRange().getStartOffset()))) return true; } for (Trinity trinity : list) { final PsiLanguageInjectionHost host = trinity.first; if (host.getContainingFile() != containingFile) continue; final TextRange textRange = trinity.third; ranges.add(textRange.shiftRight(host.getTextRange().getStartOffset())); } InjectorUtils.registerInjection(language, list, containingFile, registrar); InjectorUtils.registerSupport(mySupport, true, registrar); if (Boolean.TRUE.equals(unparsableRef.get())) { InjectorUtils.putInjectedFileUserData(registrar, InjectedLanguageUtil.FRANKENSTEIN_INJECTION, Boolean.TRUE); } return true; } }); } void getInjectedLanguage(final PsiElement place, final Ref unparsableRef, final PairProcessor>> processor) { if (place instanceof XmlTag) { final XmlTag xmlTag = (XmlTag)place; List injections = myConfiguration.getInjections(XmlLanguageInjectionSupport.XML_SUPPORT_ID); //noinspection ForLoopReplaceableByForEach for (int i = 0, injectionsSize = injections.size(); i < injectionsSize; i++) { final BaseInjection injection = injections.get(i); if (injection.acceptsPsiElement(xmlTag)) { final Language language = InjectedLanguage.findLanguageById(injection.getInjectedLanguageId()); if (language == null) continue; final boolean separateFiles = !injection.isSingleFile() && StringUtil.isNotEmpty(injection.getValuePattern()); final List> result = ContainerUtil.newArrayList(); xmlTag.acceptChildren(new PsiElementVisitor() { @Override public void visitElement(final PsiElement element) { if (element instanceof XmlText) { if (!(element instanceof PsiLanguageInjectionHost) || element.getTextLength() == 0) return; final List list = injection.getInjectedArea(element); final InjectedLanguage l = InjectedLanguage.create(injection.getInjectedLanguageId(), injection.getPrefix(), injection.getSuffix(), false); for (TextRange textRange : list) { result.add(Trinity.create((PsiLanguageInjectionHost)element, l, textRange)); } } else if (element instanceof XmlTag) { if (!separateFiles) unparsableRef.set(Boolean.TRUE); if (injection instanceof AbstractTagInjection && ((AbstractTagInjection)injection).isApplyToSubTags()) { element.acceptChildren(this); } } } }); if (!result.isEmpty()) { if (separateFiles) { for (Trinity trinity : result) { processor.process(language, Collections.singletonList(trinity)); } } else { processor.process(language, result); } } if (injection.isTerminal()) { break; } } } } else if (place instanceof XmlAttributeValue && place.getParent() instanceof XmlAttribute) { final XmlAttribute attribute = (XmlAttribute)place.getParent(); final XmlAttributeValue value = (XmlAttributeValue)place; //if (value == null) return; // Check that we don't inject anything into embedded (e.g. JavaScript) content: // XmlToken: " // JSEmbeddedContent // XmlToken " // Actually IDEA shouldn't ask for injected languages at all in this case. final PsiElement[] children = value.getChildren(); if (children.length < 3 || !(children[1] instanceof XmlToken) || ((XmlToken)children[1]).getTokenType() != XmlTokenType.XML_ATTRIBUTE_VALUE_TOKEN) { return; } List injections = myConfiguration.getInjections(XmlLanguageInjectionSupport.XML_SUPPORT_ID); //noinspection ForLoopReplaceableByForEach for (int i = 0, size = injections.size(); i < size; i++) { BaseInjection injection = injections.get(i); if (injection.acceptsPsiElement(attribute)) { final Language language = InjectedLanguage.findLanguageById(injection.getInjectedLanguageId()); if (language == null) continue; final boolean separateFiles = !injection.isSingleFile() && StringUtil.isNotEmpty(injection.getValuePattern()); final List ranges = injection.getInjectedArea(value); if (ranges.isEmpty()) continue; final List> result = new ArrayList>(); final InjectedLanguage l = InjectedLanguage.create(injection.getInjectedLanguageId(), injection.getPrefix(), injection.getSuffix(), false); for (TextRange textRange : ranges) { result.add(Trinity.create((PsiLanguageInjectionHost)value, l, textRange)); } if (separateFiles) { for (Trinity trinity : result) { processor.process(language, Collections.singletonList(trinity)); } } else { processor.process(language, result); } if (injection.isTerminal()) { break; } } } } } // NOTE: local name of an xml entity or attribute value should match at least one string in the index private boolean isInIndex(XmlElement xmlElement) { final Trinity> index = getXmlAnnotatedElementsValue(); if (xmlElement instanceof XmlAttributeValue) xmlElement = (XmlElement)xmlElement.getParent(); final XmlTag tag; if (xmlElement instanceof XmlAttribute) { final XmlAttribute attribute = (XmlAttribute)xmlElement; if (areThereInjectionsWithText(attribute.getLocalName(), index)) return true; if (areThereInjectionsWithText(attribute.getValue(), index)) return true; //if (areThereInjectionsWithText(attribute.getNamespace(), index)) return false; tag = attribute.getParent(); } else if (xmlElement instanceof XmlTag) { tag = (XmlTag)xmlElement; } else { return false; } if (areThereInjectionsWithText(tag.getLocalName(), index)) return true; //if (areThereInjectionsWithText(tag.getNamespace(), index)) return false; return false; } private static boolean areThereInjectionsWithText(final String text, Trinity> index) { if (text == null) return false; if (index.third.contains(text)) return true; Pattern pattern = index.second; return pattern != null && pattern.matcher(text).matches(); } private Trinity> getXmlAnnotatedElementsValue() { Trinity> index = myXmlIndex; if (index == null || myConfiguration.getModificationCount() != index.first.longValue()) { final Map, BaseInjection> map = new THashMap, BaseInjection>(); for (BaseInjection injection : myConfiguration.getInjections(XmlLanguageInjectionSupport.XML_SUPPORT_ID)) { for (InjectionPlace place : injection.getInjectionPlaces()) { if (!place.isEnabled() || place.getElementPattern() == null) continue; map.put(place.getElementPattern(), injection); } } final Collection stringSet = PatternValuesIndex.buildStringIndex(map.keySet()); index = Trinity.create(myConfiguration.getModificationCount(), buildPattern(stringSet), stringSet); myXmlIndex = index; } return index; } @Nullable private static Pattern buildPattern(Collection stringSet) { final StringBuilder sb = new StringBuilder(); for (String s : stringSet) { if (!InjectorUtils.isRegexp(s)) continue; if (sb.length() > 0) sb.append('|'); sb.append("(?:").append(s).append(")"); } return sb.length() == 0 ? null : Pattern.compile(sb.toString()); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy