/*
* citygml4j - The Open Source Java API for CityGML
* https://github.com/citygml4j
*
* Copyright 2013-2017 Claus Nagel <claus.nagel@gmail.com>
*
* 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.
*/
import java.io.File;
import java.text.SimpleDateFormat;
import java.util.Date;
import org.citygml4j.CityGMLContext;
import org.citygml4j.builder.CityGMLBuilder;
import org.citygml4j.model.citygml.bridge.AbstractBridge;
import org.citygml4j.model.citygml.core.AbstractCityObject;
import org.citygml4j.model.citygml.core.CityModel;
import org.citygml4j.model.citygml.core.LodRepresentation;
import org.citygml4j.model.citygml.generics.GenericCityObject;
import org.citygml4j.model.citygml.tunnel.AbstractTunnel;
import org.citygml4j.model.gml.feature.AbstractFeature;
import org.citygml4j.model.gml.geometry.GeometryProperty;
import org.citygml4j.model.gml.geometry.aggregates.MultiGeometry;
import org.citygml4j.model.module.citygml.CityGMLVersion;
import org.citygml4j.model.module.citygml.CoreModule;
import org.citygml4j.util.walker.FeatureWalker;
import org.citygml4j.xml.io.CityGMLInputFactory;
import org.citygml4j.xml.io.CityGMLOutputFactory;
import org.citygml4j.xml.io.reader.CityGMLReader;
import org.citygml4j.xml.io.reader.FeatureReadMode;
import org.citygml4j.xml.io.writer.CityModelWriter;
public class DowngradeConverter {
public static void main(String[] args) throws Exception {
final SimpleDateFormat df = new SimpleDateFormat("[HH:mm:ss] ");
System.out.println(df.format(new Date()) + "setting up citygml4j context and JAXB builder");
CityGMLContext ctx = new CityGMLContext();
CityGMLBuilder builder = ctx.createCityGMLBuilder();
System.out.println(df.format(new Date()) + "reading CityGML 2.0.0 file LOD3_Railway_v200.gml chunk-wise");
CityGMLInputFactory in = builder.createCityGMLInputFactory();
in.setProperty(CityGMLInputFactory.FEATURE_READ_MODE, FeatureReadMode.SPLIT_PER_COLLECTION_MEMBER);
CityGMLReader reader = in.createCityGMLReader(new File("../datasets/LOD3_Railway_v200.gml"));
CityGMLVersion version = CityGMLVersion.v1_0_0;
System.out.println(df.format(new Date()) + "writing citygml4j object tree as CityGML " + version + " document");
CityGMLOutputFactory out = builder.createCityGMLOutputFactory();
out.setCityGMLVersion(version);
CityModelWriter writer = out.createCityModelWriter(new File("LOD3_Railway_v100.gml"));
writer.setPrefixes(version);
writer.setDefaultNamespace(CoreModule.v1_0_0);
writer.setSchemaLocations(version);
writer.setIndentString(" ");
// we iterate through all <cityObjectMember>s of the <CityModel>
while (reader.hasNext()) {
AbstractFeature feature = (AbstractFeature)reader.nextFeature();
// if we find a tunnel or a bridge we convert it to a generic city object as proxy
if (feature instanceof AbstractTunnel || feature instanceof AbstractBridge) {
final GenericCityObject proxy = new GenericCityObject();
proxy.setLod3Geometry(new GeometryProperty<MultiGeometry>(new MultiGeometry()));
System.out.println(df.format(new Date()) + "converting " + ((AbstractCityObject)feature).getCityGMLClass() + " and nested features to " + proxy.getCityGMLClass());
// we use a feature walker to visit the tunnel/bridge and all its nested city objects
FeatureWalker walker = new FeatureWalker() {
public void visit(AbstractCityObject abstractCityObject) {
// simply collect all LOD3 geometries and add them to the gml:MultiGeometry of the proxy
System.out.println(df.format(new Date()) + "adding geometry of " + abstractCityObject.getCityGMLClass());
LodRepresentation lods = abstractCityObject.getLodRepresentation();
if (lods.isSetLod3Geometry()) {
MultiGeometry multiGeometry = (MultiGeometry)proxy.getLod3Geometry().getGeometry();
multiGeometry.getGeometryMember().addAll(lods.getLodGeometry(3));
}
}
public void visit(AbstractBridge abstractBridge) {
// copy attributes of the brigde
proxy.setClazz(abstractBridge.getClazz());
proxy.setFunction(abstractBridge.getFunction());
proxy.setUsage(abstractBridge.getUsage());
super.visit(abstractBridge);
}
public void visit(AbstractTunnel abstractTunnel) {
// copy attributes of the tunnel
proxy.setClazz(abstractTunnel.getClazz());
proxy.setFunction(abstractTunnel.getFunction());
proxy.setUsage(abstractTunnel.getUsage());
super.visit(abstractTunnel);
}
};
feature.accept(walker);
// copy common attributes
proxy.setId(feature.getId());
proxy.setName(feature.getName());
proxy.setRelativeToTerrain(((AbstractCityObject)feature).getRelativeToTerrain());
proxy.setRelativeToWater(((AbstractCityObject)feature).getRelativeToWater());
// let's swap the bridge/tunnel with the proxy object
feature = proxy;
}
if (!(feature instanceof CityModel))
writer.writeFeatureMember(feature);
}
reader.close();
writer.close();
System.out.println(df.format(new Date()) + "CityGML file LOD3_Railway_v100.gml written");
System.out.println(df.format(new Date()) + "sample citygml4j application successfully finished");
}
}