| /******************************************************************************* |
| * Copyright (c) 2011-2018 EclipseSource Muenchen GmbH and others. |
| * |
| * All rights reserved. This program and the accompanying materials |
| * are made available under the terms of the Eclipse Public License 2.0 |
| * which accompanies this distribution, and is available at |
| * https://www.eclipse.org/legal/epl-2.0/ |
| * |
| * SPDX-License-Identifier: EPL-2.0 |
| * |
| * Contributors: |
| * Lucas Koehler - initial API and implementation |
| ******************************************************************************/ |
| package org.eclipse.emfforms.internal.core.services.segments.index; |
| |
| import org.eclipse.core.runtime.IStatus; |
| import org.eclipse.emf.common.notify.Notification; |
| import org.eclipse.emf.common.util.EList; |
| import org.eclipse.emf.ecore.EObject; |
| import org.eclipse.emf.ecore.EReference; |
| import org.eclipse.emf.ecore.EStructuralFeature.Setting; |
| import org.eclipse.emf.ecp.common.spi.asserts.Assert; |
| import org.eclipse.emf.ecp.view.spi.model.ModelChangeNotification; |
| import org.eclipse.emf.ecp.view.spi.model.VDomainModelReferenceSegment; |
| import org.eclipse.emfforms.spi.common.report.AbstractReport; |
| import org.eclipse.emfforms.spi.common.report.ReportService; |
| import org.eclipse.emfforms.spi.core.services.databinding.DatabindingFailedException; |
| import org.eclipse.emfforms.spi.core.services.databinding.DatabindingFailedReport; |
| import org.eclipse.emfforms.spi.core.services.databinding.emf.EMFFormsSegmentResolver; |
| import org.eclipse.emfforms.spi.core.services.structuralchange.StructuralChangeSegmentTester; |
| import org.eclipse.emfforms.spi.view.indexsegment.model.VIndexDomainModelReferenceSegment; |
| import org.osgi.service.component.annotations.Component; |
| import org.osgi.service.component.annotations.Reference; |
| |
| /** |
| * A {@link StructuralChangeSegmentTester} implementation to check {@link VIndexDomainModelReferenceSegment |
| * VIndexDomainModelReferenceSegments} with corresponding domain objects for structural changes. |
| * |
| * @author Lucas Koehler |
| * |
| */ |
| @Component(name = "IndexSegmentStructuralChangeTester") |
| public class IndexSegmentStructuralChangeTester implements StructuralChangeSegmentTester { |
| private EMFFormsSegmentResolver segmentResolver; |
| private ReportService reportService; |
| |
| /** |
| * Sets the {@link EMFFormsSegmentResolver}. |
| * |
| * @param segmentResolver The {@link EMFFormsSegmentResolver} |
| */ |
| @Reference(unbind = "-") |
| protected void setEMFFormsSegmentResolver(EMFFormsSegmentResolver segmentResolver) { |
| this.segmentResolver = segmentResolver; |
| } |
| |
| /** |
| * Sets the {@link ReportService}. |
| * |
| * @param reportService The {@link ReportService} |
| */ |
| @Reference(unbind = "-") |
| protected void setReportService(ReportService reportService) { |
| this.reportService = reportService; |
| } |
| |
| // BEGIN COMPLEX CODE |
| @SuppressWarnings("unchecked") |
| @Override |
| public boolean isStructureChanged(VDomainModelReferenceSegment segment, EObject domainObject, |
| ModelChangeNotification notification) { |
| Assert.create(segment).notNull(); |
| Assert.create(domainObject).notNull(); |
| Assert.create(segment).notNull().ofClass(VIndexDomainModelReferenceSegment.class); |
| |
| if (notification.getRawNotification().isTouch()) { |
| return false; |
| } |
| |
| final VIndexDomainModelReferenceSegment indexSegment = (VIndexDomainModelReferenceSegment) segment; |
| |
| Setting setting; |
| try { |
| setting = segmentResolver.resolveSegment(indexSegment, domainObject); |
| } catch (final DatabindingFailedException ex) { |
| reportService.report(new DatabindingFailedReport(ex)); |
| return false; |
| } |
| |
| /* |
| * Check whether the notifying EObject and the EReference match the notification. If this is the case, check if |
| * the relevant element was changed. |
| * If the EStructuralFeature of the resolved Setting is an EAttribute, its change is irrelevant. |
| */ |
| if (EReference.class.isInstance(setting.getEStructuralFeature())) { |
| final EReference eReference = (EReference) setting.getEStructuralFeature(); |
| if (eReference.equals(notification.getStructuralFeature()) |
| && notification.getNotifier() == setting.getEObject()) { |
| |
| final int event = notification.getRawNotification().getEventType(); |
| final int position = notification.getRawNotification().getPosition(); |
| final int index = indexSegment.getIndex(); |
| final EList<Object> list = (EList<Object>) setting.getEObject().eGet(setting.getEStructuralFeature()); |
| |
| /* |
| * For ADD and REMOVE only changes at the index's or a lower position in the list affect the indexed |
| * position. |
| * For ADD, after the element has been added, the list's size must be at least index+1. Otherwise, the |
| * element at the indexed position still does not exist after the change. |
| */ |
| if (event == Notification.ADD && position <= index && index < list.size()) { |
| return true; |
| } |
| |
| /* |
| * For REMOVE, after the element has been removed, the list's size must be at least equal to the index. |
| * Otherwise, the element at the indexed position did not exist before and after the change. |
| */ |
| if (event == Notification.REMOVE && position <= index && index <= list.size()) { |
| return true; |
| } |
| |
| /* |
| * For ADD_MANY, REMOVE_MANY or MOVE events the position of the notification will not be the segment's |
| * index even if the element at the index changed. Therefore, we have to be overly careful and |
| * always indicate a change when one of the events was triggered. |
| */ |
| if (event == Notification.ADD_MANY || event == Notification.REMOVE_MANY |
| || event == Notification.MOVE) { |
| return true; |
| } |
| } |
| } |
| return false; |
| } |
| // END COMPLEX CODE |
| |
| @Override |
| public double isApplicable(VDomainModelReferenceSegment segment) { |
| if (segment == null) { |
| reportService |
| .report(new AbstractReport("The given domain model reference segment was null.", IStatus.WARNING)); //$NON-NLS-1$ |
| return NOT_APPLICABLE; |
| } |
| if (VIndexDomainModelReferenceSegment.class.isInstance(segment)) { |
| return 5d; |
| } |
| return NOT_APPLICABLE; |
| } |
| |
| } |