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.processors;

import com.scrivenvar.AbstractFileFactory;
import com.scrivenvar.FileEditorTab;
import com.scrivenvar.preview.HTMLPreviewPane;
import java.nio.file.Path;
import java.util.Map;
import javafx.beans.value.ObservableValue;

/**
 * Responsible for creating processors capable of parsing, transforming,
 * interpolating, and rendering known file types.
 *
 * @author White Magic Software, Ltd.
 */
public class ProcessorFactory extends AbstractFileFactory {

  private final HTMLPreviewPane previewPane;
  private final Map<String, String> resolvedMap;

  private Processor<String> terminalProcessChain;

  /**
   * Constructs a factory with the ability to create processors that can perform
   * text and caret processing to generate a final preview.
   *
   * @param previewPane
   * @param resolvedMap
   */
  public ProcessorFactory(
    final HTMLPreviewPane previewPane,
    final Map<String, String> resolvedMap ) {
    this.previewPane = previewPane;
    this.resolvedMap = resolvedMap;
  }

  /**
   * Creates a processor suitable for parsing and rendering the file opened at
   * the given tab.
   *
   * @param tab The tab containing a text editor, path, and caret position.
   *
   * @return A processor that can render the given tab's text.
   */
  public Processor<String> createProcessor( final FileEditorTab tab ) {
    final Path path = tab.getPath();
    final Processor<String> processor;

    switch( lookup( path ) ) {
      case RMARKDOWN:
        processor = createRProcessor( tab );
        break;

      case SOURCE:
        processor = createMarkdownProcessor( tab );
        break;

      case XML:
        processor = createXMLProcessor( tab );
        break;

      case RXML:
        processor = createRXMLProcessor( tab );
        break;

      default:
        processor = createIdentityProcessor( tab );
        break;
    }

    return processor;
  }

  /**
   * Returns a processor common to all processors: markdown, caret position
   * token replacer, and an HTML preview renderer.
   *
   * @return Processors at the end of the processing chain.
   */
  private Processor<String> getCommonProcessor() {
    if( this.terminalProcessChain == null ) {
      this.terminalProcessChain = createCommonProcessor();
    }

    return this.terminalProcessChain;
  }

  /**
   * Creates and links the processors at the end of the processing chain.
   *
   * @return A markdown, caret replacement, and preview pane processor chain.
   */
  private Processor<String> createCommonProcessor() {
    final Processor<String> hpp = new HTMLPreviewProcessor( getPreviewPane() );
    final Processor<String> mcrp = new CaretReplacementProcessor( hpp );
    final Processor<String> mpp = new MarkdownProcessor( mcrp );

    return mpp;
  }
  
  protected Processor<String> createIdentityProcessor( final FileEditorTab tab ) {
    final Processor<String> hpp = new HTMLPreviewProcessor( getPreviewPane() );
    final Processor<String> ip = new IdentityProcessor( hpp );
    
    return ip;
  }

  protected Processor<String> createMarkdownProcessor( final FileEditorTab tab ) {
    final ObservableValue<Integer> caret = tab.caretPositionProperty();
    final Processor<String> tpc = getCommonProcessor();
    final Processor<String> cip = createMarkdownInsertionProcessor( tpc, caret );
    final Processor<String> dvp = new DefaultVariableProcessor( cip, getResolvedMap() );

    return dvp;
  }

  protected Processor<String> createXMLProcessor( final FileEditorTab tab ) {
    final ObservableValue<Integer> caret = tab.caretPositionProperty();
    final Processor<String> tpc = getCommonProcessor();
    final Processor<String> xmlp = new XMLProcessor( tpc, tab.getPath() );
    final Processor<String> dvp = new DefaultVariableProcessor( xmlp, getResolvedMap() );
    final Processor<String> xcip = createXMLInsertionProcessor( dvp, caret );

    return xcip;
  }

  protected Processor<String> createRProcessor( final FileEditorTab tab ) {
    final ObservableValue<Integer> caret = tab.caretPositionProperty();
    final Processor<String> tpc = getCommonProcessor();
    final Processor<String> rp = new InlineRProcessor( tpc, getResolvedMap(), tab.getPath() );
    final Processor<String> rvp = new RVariableProcessor( rp, getResolvedMap() );
    final Processor<String> cip = createRInsertionProcessor( rvp, caret );

    return cip;
  }

  protected Processor<String> createRXMLProcessor( final FileEditorTab tab ) {
    final ObservableValue<Integer> caret = tab.caretPositionProperty();
    final Processor<String> tpc = getCommonProcessor();
    final Processor<String> xmlp = new XMLProcessor( tpc, tab.getPath() );
    final Processor<String> rp = new InlineRProcessor( xmlp, getResolvedMap(), tab.getPath() );
    final Processor<String> rvp = new RVariableProcessor( rp, getResolvedMap() );
    final Processor<String> xcip = createXMLInsertionProcessor( rvp, caret );

    return xcip;
  }

  private Processor<String> createMarkdownInsertionProcessor(
    final Processor<String> tpc, final ObservableValue<Integer> caret ) {
    return new MarkdownCaretInsertionProcessor( tpc, caret );
  }

  /**
   * Create an insertion processor that is aware of R statements and will insert
   * a caret outside of any statement the caret falls within.
   *
   * @param processor Another link in the processor chain.
   * @param caret The caret insertion point.
   *
   * @return A processor that can insert a caret token without disturbing any R
   * code.
   */
  private Processor<String> createRInsertionProcessor(
    final Processor<String> processor, final ObservableValue<Integer> caret ) {
    return new RMarkdownCaretInsertionProcessor( processor, caret );
  }

  private Processor<String> createXMLInsertionProcessor(
    final Processor<String> tpc, final ObservableValue<Integer> caret ) {
    return new XMLCaretInsertionProcessor( tpc, caret );
  }

  private HTMLPreviewPane getPreviewPane() {
    return this.previewPane;
  }

  /**
   * Returns the variable map of interpolated definitions.
   *
   * @return A map to help dereference variables.
   */
  private Map<String, String> getResolvedMap() {
    return this.resolvedMap;
  }
}