/* * Copyright 2016 White Magic Software, Ltd. * * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * o Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * o Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ package com.scrivenvar.definition.yaml; import com.fasterxml.jackson.databind.JsonNode; import com.scrivenvar.definition.VariableTreeItem; import java.util.Map.Entry; import javafx.scene.control.TreeItem; import javafx.scene.control.TreeView; /** * Transforms a JsonNode hierarchy into a tree that can be displayed in a user * interface. * * @author White Magic Software, Ltd. */ public class YamlTreeAdapter { private YamlParser yamlParser; public YamlTreeAdapter( final YamlParser parser ) { setYamlParser( parser ); } /** * Converts a YAML document to a TreeView based on the document keys. Only the * first document in the stream is adapted. * * @param name Root TreeItem node name. * * @return A TreeView populated with all the keys in the YAML document. */ public TreeView<String> adapt( final String name ){ final JsonNode rootNode = getYamlParser().getDocumentRoot(); final TreeItem<String> rootItem = createTreeItem( name ); rootItem.setExpanded( true ); adapt( rootNode, rootItem ); return new TreeView<>( rootItem ); } /** * Iterate over a given root node (at any level of the tree) and adapt each * leaf node. * * @param rootNode A JSON node (YAML node) to adapt. * @param rootItem The tree item to use as the root when processing the node. */ private void adapt( final JsonNode rootNode, final TreeItem<String> rootItem ) { rootNode.fields().forEachRemaining( (Entry<String, JsonNode> leaf) -> adapt( leaf, rootItem ) ); } /** * Recursively adapt each rootNode to a corresponding rootItem. * * @param rootNode The node to adapt. * @param rootItem The item to adapt using the node's key. */ private void adapt( final Entry<String, JsonNode> rootNode, final TreeItem<String> rootItem ) { final JsonNode leafNode = rootNode.getValue(); final String key = rootNode.getKey(); final TreeItem<String> leaf = createTreeItem( key ); if( leafNode.isValueNode() ) { leaf.getChildren().add( createTreeItem( rootNode.getValue().asText() ) ); } rootItem.getChildren().add( leaf ); if( leafNode.isObject() ) { adapt( leafNode, leaf ); } } /** * Creates a new tree item that can be added to the tree view. * * @param value The node's value. * * @return A new tree item node, never null. */ private TreeItem<String> createTreeItem( final String value ) { return new VariableTreeItem<>( value ); } private YamlParser getYamlParser() { return this.yamlParser; } private void setYamlParser( final YamlParser yamlParser ) { this.yamlParser = yamlParser; } }