/******************************************************************************* * Copyright (c) 2012, 2013 Association for Decentralized Information Management in * Industry THTH ry. * 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: * VTT Technical Research Centre of Finland - initial API and implementation *******************************************************************************/ package org.simantics.objmap.internal; import gnu.trove.map.hash.THashMap; import java.util.ArrayList; import java.util.Collection; import java.util.Collections; import java.util.Set; import org.simantics.db.ReadGraph; import org.simantics.db.WriteGraph; import org.simantics.objmap.bidirectional.IBidirectionalLinkType; import org.simantics.objmap.bidirectional.IBidirectionalMapping; import org.simantics.objmap.bidirectional.IBidirectionalMappingSchema; import org.simantics.objmap.exceptions.MappingException; public class BidirectionalMapping implements IBidirectionalMapping { IBidirectionalMappingSchema schema; THashMap> forwardMap = new THashMap>(); THashMap> backwardMap = new THashMap>(); ArrayList> modifiedDomain = new ArrayList>(); ArrayList> modifiedRange = new ArrayList>(); private void markDomainModified(BidirectionalLink link) { if(!link.domainModified) { link.domainModified = true; modifiedDomain.add(link); } } private void markRangeModified(BidirectionalLink link) { if(!link.rangeModified) { link.rangeModified = true; modifiedRange.add(link); } } public BidirectionalMapping(IBidirectionalMappingSchema schema) { this.schema = schema; } private BidirectionalLink addLink(IBidirectionalLinkType linkType, Domain domainElement, Range rangeElement) { BidirectionalLink link = new BidirectionalLink(linkType, domainElement, rangeElement); forwardMap.put(domainElement, link); backwardMap.put(rangeElement, link); return link; } @Override public Set getDomain() { return Collections.unmodifiableSet(forwardMap.keySet()); } @Override public Range get(Domain domainElement) { BidirectionalLink link = forwardMap.get(domainElement); if(link == null) return null; return link.rangeElement; } @Override public Range map(ReadGraph graph, Domain domainElement) throws MappingException { Range result = get(domainElement); if(result == null) { IBidirectionalLinkType linkType = schema.linkTypeOfDomainElement(graph, domainElement); Range rangeElement = linkType.createRangeElement(graph, domainElement); addLink(linkType, domainElement, rangeElement); linkType.createRange(graph, this, domainElement, rangeElement); } return result; } public Collection updateRange(ReadGraph graph) throws MappingException { ArrayList updated = new ArrayList(Math.max(10, modifiedDomain.size())); for(BidirectionalLink link : modifiedDomain) { link.domainModified = false; if(link.linkType.updateRange(graph, this, link.domainElement, link.rangeElement)) updated.add(link.rangeElement); } modifiedDomain.clear(); return updated; } @Override public Set getRange() { return Collections.unmodifiableSet(backwardMap.keySet()); } @Override public Domain inverseGet(Range rangeElement) { BidirectionalLink link = backwardMap.get(rangeElement); if(link == null) return null; return link.domainElement; } @Override public Domain inverseMap(WriteGraph graph, Range rangeElement) throws MappingException { Domain result = inverseGet(rangeElement); if(result == null) { IBidirectionalLinkType linkType = schema.linkTypeOfRangeElement(graph, rangeElement); Domain domainElement = linkType.createDomainElement(graph, rangeElement); addLink(linkType, domainElement, rangeElement); linkType.createDomain(graph, this, domainElement, rangeElement); } return result; } public Collection updateDomain(WriteGraph graph) throws MappingException { ArrayList updated = new ArrayList(Math.max(10, modifiedRange.size())); for(BidirectionalLink link : modifiedRange) { link.rangeModified = false; if(link.linkType.updateDomain(graph, this, link.domainElement, link.rangeElement)) updated.add(link.domainElement); } modifiedDomain.clear(); return updated; } }