Dave Jarvis' Repositories

git clone https://repo.autonoma.ca/repo/keenwrite.git
/*
 * 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;
  }
}