blob: 4bf2f0860f4bdfc616d5a7c8bcc1b07384f16234 [file] [log] [blame]
/*=============================================================================#
# Copyright (c) 2021 Stephan Wahlbrink and others.
#
# This program and the accompanying materials are made available under the
# terms of the Eclipse Public License 2.0 which is available at
# https://www.eclipse.org/legal/epl-2.0, or the Apache License, Version 2.0
# which is available at https://www.apache.org/licenses/LICENSE-2.0.
#
# SPDX-License-Identifier: EPL-2.0 OR Apache-2.0
#
# Contributors:
# Stephan Wahlbrink <sw@wahlbrink.eu> - initial API and implementation
#=============================================================================*/
package org.eclipse.statet.ltk.ui.util;
import java.util.IdentityHashMap;
import java.util.List;
import org.eclipse.jface.viewers.ITreeContentProvider;
import org.eclipse.jface.viewers.Viewer;
import org.eclipse.statet.jcommons.lang.Disposable;
import org.eclipse.statet.jcommons.lang.NonNull;
import org.eclipse.statet.jcommons.lang.NonNullByDefault;
import org.eclipse.statet.jcommons.lang.Nullable;
import org.eclipse.statet.ltk.core.source.SourceModelStamp;
import org.eclipse.statet.ltk.model.core.LtkModels;
import org.eclipse.statet.ltk.model.core.element.EmbeddingForeignElement;
import org.eclipse.statet.ltk.model.core.element.LtkModelElement;
import org.eclipse.statet.ltk.model.core.element.LtkModelElementFilter;
import org.eclipse.statet.ltk.model.core.element.SourceStructElement;
import org.eclipse.statet.ltk.model.core.element.SourceUnitModelInfo;
import org.eclipse.statet.ltk.ui.ElementContentProvider;
@NonNullByDefault
public class ExtModelContentProvider implements ITreeContentProvider {
private static final @NonNull SourceStructElement<?, ?>[] NO_CHILDREN= new SourceStructElement[0];
protected static final ElementContentProvider NO_PROVIDER= new ElementContentProvider() {
};
private final IdentityHashMap<String, ElementContentProvider> providers= new IdentityHashMap<>(4);
public interface ModelContent {
@Nullable SourceUnitModelInfo getModelInfo(Object inputElement);
@Nullable LtkModelElementFilter<? super SourceStructElement<?, ?>> getContentFilter();
}
private final ModelContent content;
public ExtModelContentProvider(final ModelContent content) {
this.content= content;
}
@Override
public void dispose() {
for (final var provider : this.providers.values()) {
if (provider instanceof Disposable) {
((Disposable)provider).dispose();
}
}
this.providers.clear();
}
protected final ModelContent getContent() {
return this.content;
}
public @Nullable SourceModelStamp getStamp(final Object inputElement) {
final SourceUnitModelInfo modelInfo= getContent().getModelInfo(inputElement);
return (modelInfo != null) ? modelInfo.getStamp() : null;
}
@Override
public void inputChanged(final Viewer viewer, final @Nullable Object oldInput, final @Nullable Object newInput) {
}
protected @Nullable ElementContentProvider getElementContentProvider(final @Nullable String modelId) {
if (modelId == null) {
return null;
}
var provider= this.providers.get(modelId);
if (provider == null) {
provider= LtkModels.getModelAdapter(modelId, ElementContentProvider.class);
if (provider == null) {
provider= NO_PROVIDER;
}
this.providers.put(modelId, provider);
}
return (provider != NO_PROVIDER) ? provider : null;
}
protected void addProvider(final String modelId, final ElementContentProvider provider) {
this.providers.put(modelId, provider);
}
private boolean skipToForeignElement(final @Nullable SourceStructElement<?, ?> element) {
return (element != null
&& (element.getElementType() & LtkModelElement.MASK_C1) == LtkModelElement.C1_EMBEDDED );
}
protected boolean skipInTree(final SourceStructElement<?, ?> element) {
return false;
}
protected @NonNull SourceStructElement<?, ?>[] getElements(final @Nullable SourceUnitModelInfo modelInfo) {
if (modelInfo != null) {
final var children= modelInfo.getSourceElement()
.getSourceChildren(getContent().getContentFilter());
return children.toArray(new @NonNull SourceStructElement[children.size()]);
}
return NO_CHILDREN;
}
protected @Nullable SourceStructElement<?, ?> getParent(final SourceStructElement<?, ?> element) {
var parent= element.getSourceParent();
while (parent != null && (skipToForeignElement(parent) || skipInTree(parent))) {
parent= parent.getSourceParent();
}
return parent;
}
protected boolean hasChildren(@Nullable SourceStructElement<?, ?> element) {
if (skipToForeignElement(element)) {
element= ((EmbeddingForeignElement<?, ?>)element).getForeignElement();
}
ElementContentProvider provider;
if (element != null) {
if ((provider= getElementContentProvider(element.getModelTypeId())) != null) {
return provider.hasSourceChildren(element, getContent().getContentFilter());
}
else {
return element.hasSourceChildren(getContent().getContentFilter());
}
}
return false;
}
protected @NonNull SourceStructElement<?, ?>[] getChildren(@Nullable SourceStructElement<?, ?> element) {
if (skipToForeignElement(element)) {
element= ((EmbeddingForeignElement<?, ?>)element).getForeignElement();
}
ElementContentProvider provider;
if (element != null) {
List<? extends LtkModelElement<?>> children;
if ((provider= getElementContentProvider(element.getModelTypeId())) != null) {
children= provider.getSourceChildren(element, getContent().getContentFilter());
}
else {
children= element.getSourceChildren(getContent().getContentFilter());
}
return children.toArray(new @NonNull SourceStructElement[children.size()]);
}
return NO_CHILDREN;
}
@Override
public @NonNull Object[] getElements(final Object inputElement) {
return getElements(getContent().getModelInfo(inputElement));
}
@Override
public @Nullable Object getParent(final Object element) {
return getParent((SourceStructElement<?, ?>)element);
}
@Override
public boolean hasChildren(final Object element) {
return hasChildren((SourceStructElement<?, ?>)element);
}
@Override
public @NonNull Object[] getChildren(final Object element) {
return getChildren((SourceStructElement<?, ?>)element);
}
}