/****************************************************************************** * Copyright (c) 2016 Oracle * All rights reserved. This program and the accompanying materials * are made available under the terms of the Eclipse Public License v1.0 * which accompanies this distribution, and is available at * http://www.eclipse.org/legal/epl-v10.html * * Contributors: * Konstantin Komissarchik - initial implementation and ongoing maintenance ******************************************************************************/ package org.eclipse.sapphire.internal; import org.eclipse.sapphire.Element; import org.eclipse.sapphire.Event; import org.eclipse.sapphire.Listener; import org.eclipse.sapphire.LoggingService; import org.eclipse.sapphire.PropertyDef; import org.eclipse.sapphire.Sapphire; import org.eclipse.sapphire.Since; import org.eclipse.sapphire.Version; import org.eclipse.sapphire.VersionCompatibility; import org.eclipse.sapphire.VersionCompatibilityService; import org.eclipse.sapphire.VersionCompatibilityTargetService; import org.eclipse.sapphire.VersionConstraint; import org.eclipse.sapphire.modeling.el.ConcatFunction; import org.eclipse.sapphire.modeling.el.FailSafeFunction; import org.eclipse.sapphire.modeling.el.Function; import org.eclipse.sapphire.modeling.el.FunctionResult; import org.eclipse.sapphire.modeling.el.Literal; import org.eclipse.sapphire.modeling.el.ModelElementFunctionContext; import org.eclipse.sapphire.modeling.el.parser.ExpressionLanguageParser; import org.eclipse.sapphire.modeling.util.MiscUtil; import org.eclipse.sapphire.services.ServiceCondition; import org.eclipse.sapphire.services.ServiceContext; /** * Implementation of VersionCompatibilityService that derives its behavior from @VersionCompatibility and @Since annotations. * * @author <a href="mailto:konstantin.komissarchik@oracle.com">Konstantin Komissarchik</a> */ public final class DeclarativeVersionCompatibilityService extends VersionCompatibilityService { private VersionCompatibilityTargetService versionCompatibilityTargetService; private Listener versionCompatibilityTargetServiceListener; private FunctionResult functionResult; @Override protected void initVersionCompatibilityService() { final Element element = context( Element.class ); final PropertyDef property = context( PropertyDef.class ); this.versionCompatibilityTargetService = VersionCompatibilityTargetService.find( element, property ); this.versionCompatibilityTargetServiceListener = new Listener() { @Override public void handle( final Event event ) { refresh(); } }; this.versionCompatibilityTargetService.attach( this.versionCompatibilityTargetServiceListener ); Function function = null; final VersionCompatibility versionCompatibilityAnnotation = property.getAnnotation( VersionCompatibility.class ); if( versionCompatibilityAnnotation != null ) { try { function = ExpressionLanguageParser.parse( versionCompatibilityAnnotation.value() ); } catch( Exception e ) { Sapphire.service( LoggingService.class ).log( e ); } } else { final Since sinceAnnotation = property.getAnnotation( Since.class ); try { function = ExpressionLanguageParser.parse( sinceAnnotation.value() ); } catch( Exception e ) { Sapphire.service( LoggingService.class ).log( e ); } if( function != null ) { function = ConcatFunction.create( "[", function ); } } if( function == null ) { function = Literal.NULL; } else { function = FailSafeFunction.create( function, Literal.create( VersionConstraint.class ), null ); this.functionResult = function.evaluate( new ModelElementFunctionContext( element ) ); this.functionResult.attach ( new Listener() { @Override public void handle( final Event event ) { refresh(); } } ); } } @Override protected Data compute() { final Version version = this.versionCompatibilityTargetService.version(); final String versioned = this.versionCompatibilityTargetService.versioned(); boolean compatible = false; final VersionConstraint constraint = (VersionConstraint) this.functionResult.value(); if( constraint != null && version != null ) { compatible = constraint.check( version ); } return new Data( versioned, version, constraint, compatible ); } public VersionConstraint constraint() { final Data data = (Data) data(); return ( data == null ? null : data.constraint() ); } @Override public void dispose() { super.dispose(); if( this.versionCompatibilityTargetService != null ) { this.versionCompatibilityTargetService.detach( this.versionCompatibilityTargetServiceListener ); } if( this.functionResult != null ) { try { this.functionResult.dispose(); } catch( Exception e ) { Sapphire.service( LoggingService.class ).log( e ); } } } public static final class Data extends VersionCompatibilityService.Data { private final VersionConstraint constraint; public Data( final String versioned, final Version version, final VersionConstraint constraint, final boolean compatible ) { super( compatible, version, versioned ); this.constraint = constraint; } public VersionConstraint constraint() { return this.constraint; } @Override public boolean equals( final Object obj ) { if( obj instanceof Data ) { final Data data = (Data) obj; return super.equals( obj ) && MiscUtil.equal( this.constraint, data.constraint ); } return false; } @Override public int hashCode() { return super.hashCode() + ( this.constraint == null ? 0 : this.constraint.hashCode() ); } } public static final class Condition extends ServiceCondition { @Override public boolean applicable( final ServiceContext context ) { final Element element = context.find( Element.class ); final PropertyDef property = context.find( PropertyDef.class ); return ( property.hasAnnotation( VersionCompatibility.class ) || property.hasAnnotation( Since.class ) ) && ( VersionCompatibilityTargetService.find( element, property ) != null ); } } }