All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.citygml4j.cityjson.writer.CityJSONFeatureWriter Maven / Gradle / Ivy

/*
 * citygml4j - The Open Source Java API for CityGML
 * https://github.com/citygml4j
 *
 * Copyright 2013-2023 Claus Nagel 
 *
 * 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.
 */

package org.citygml4j.cityjson.writer;

import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.util.MinimalPrettyPrinter;
import com.fasterxml.jackson.databind.node.ObjectNode;
import org.citygml4j.cityjson.adapter.Fields;
import org.citygml4j.cityjson.adapter.geometry.serializer.VerticesBuilder;
import org.citygml4j.cityjson.model.CityJSONType;
import org.citygml4j.cityjson.model.geometry.Transform;
import org.citygml4j.cityjson.model.geometry.Vertex;
import org.citygml4j.core.model.cityobjectgroup.CityObjectGroup;
import org.citygml4j.core.model.core.AbstractFeature;
import org.xmlobjects.gml.model.geometry.Envelope;
import org.xmlobjects.gml.visitor.Visitable;

import java.io.IOException;
import java.util.ArrayDeque;
import java.util.Deque;
import java.util.List;
import java.util.Objects;

public class CityJSONFeatureWriter extends AbstractCityJSONWriter {
    private final Deque topLevelObjects = new ArrayDeque<>();

    CityJSONFeatureWriter(JsonGenerator writer) {
        super(writer);
        writer.setPrettyPrinter(new MinimalPrettyPrinter("\n"));
    }

    public boolean isSetExternalExtension(String name) {
        return helper.isSetExternalExtension(name);
    }

    public void addExternalExtension(String name, String url, String version) {
        helper.addExternalExtension(name, url, version);
    }

    public Vertex getTranslation() {
        return helper.getGeometrySerializer().getVerticesBuilder().getTranslation();
    }

    public CityJSONFeatureWriter withTranslation(Vertex translation) {
        Objects.requireNonNull(translation, "Translation must not be null.");
        helper.getGeometrySerializer().getVerticesBuilder().setTranslation(translation);
        return this;
    }

    private void writeStartDocument(AbstractFeature feature) throws CityJSONWriteException {
        if (state != State.INITIAL) {
            throw new CityJSONWriteException("The document has already been started.");
        }

        try {
            writer.writeStartObject();
            writer.writeStringField(Fields.TYPE, CityJSONType.CITYJSON.toTypeName());
            writer.writeStringField(Fields.VERSION, helper.getVersion().toValue());

            // write empty CityObjects field
            writer.writeObjectFieldStart(Fields.CITY_OBJECTS);
            writer.writeEndObject();

            // write empty vertices field
            writer.writeArrayFieldStart(Fields.VERTICES);
            writer.writeEndArray();

            writeTransform(computeTransform(feature));
            writeExtensions();
            getAndSetReferenceSystem(feature);
            writeMetadata();

            writer.writeEndObject();
        } catch (IOException e) {
            throw new CityJSONWriteException("Caused by:", e);
        } finally {
            state = State.DOCUMENT_STARTED;
        }
    }

    public void writeCityObject(AbstractFeature feature) throws CityJSONWriteException {
        switch (state) {
            case CLOSED:
                throw new CityJSONWriteException("Illegal to write city objects after writer has been closed.");
            case INITIAL:
                writeStartDocument(feature);
        }

        super.writeCityObject(feature);
    }

    @Override
    void beginTopLevelObject() {
        topLevelObjects.push(helper.createObject());
    }

    @Override
    void writeCityObject(String id, ObjectNode node) throws CityJSONWriteException {
         ObjectNode topLevelObject = topLevelObjects.pop();
         if (topLevelObject != null) {
             try {
                 topLevelObject.set(id, node);

                 writer.writeStartObject();
                 writer.writeStringField(Fields.TYPE, CityJSONType.CITYJSON_FEATURE.toTypeName());
                 writer.writeStringField(Fields.ID, id);
                 writer.writeObjectField(Fields.CITY_OBJECTS, topLevelObject);

                 writeVertices(false);
                 writeAppearance();

                 writer.writeEndObject();
             } catch (IOException e) {
                 throw new CityJSONWriteException("Caused by:", e);
             } finally {
                 helper.reset();
             }
         }
    }

    @Override
    void writeChildObject(String id, ObjectNode node) throws CityJSONWriteException {
        ObjectNode topLevelObject = topLevelObjects.peek();
        if (topLevelObject != null) {
            topLevelObject.set(id, node);
        }
    }

    private void writeEndDocument() throws CityJSONWriteException {
        if (state == State.INITIAL) {
            writeStartDocument(null);
        }

        if (state == State.DOCUMENT_STARTED) {
            for (Visitable visitable : resolveScopes) {
                if (visitable instanceof CityObjectGroup) {
                    if (state == State.INITIAL) {
                        writeStartDocument((CityObjectGroup) visitable);
                    }

                    writeCityObject((CityObjectGroup) visitable);
                }
            }
        }
    }

    @Override
    public void close() throws CityJSONWriteException {
        if (state == State.CLOSED) {
            throw new CityJSONWriteException("The writer has already been closed.");
        }

        try {
            writeEndDocument();
            super.close();
        } finally {
            state = State.CLOSED;
        }
    }

    private Transform computeTransform(AbstractFeature feature) {
        VerticesBuilder builder = helper.getGeometrySerializer().getVerticesBuilder();
        Transform transform = new Transform();

        double scale = 1 / Math.pow(10, builder.getPrecision());
        transform.setScale(Vertex.of(scale, scale, scale));

        if (feature != null && builder.getTranslation() == null) {
            Envelope envelope = feature.computeEnvelope();
            List lowerCorner = envelope.getLowerCorner().toCoordinateList3D();
            if (!lowerCorner.isEmpty()) {
                transform.setTranslate(Vertex.of(
                        builder.round(lowerCorner.get(0)),
                        builder.round(lowerCorner.get(1)),
                        builder.round(lowerCorner.get(2))));
            }

            builder.setTranslation(transform.getTranslate());
        }

        return transform;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy