šablona/funkce/src/cz/frantovo/xmlWebGenerator/makra/Diagram.java
author František Kučera <franta-hg@frantovo.cz>
Thu Oct 24 22:06:44 2019 +0200 (2019-10-24)
changeset 136 d5feb9d8ebc3
parent 90 ae439159d833
permissions -rw-r--r--
fix license version: GNU GPLv3
     1 /**
     2  * XML Web generátor – program na generování webových stránek
     3  * Copyright © 2012 František Kučera (frantovo.cz)
     4  * 
     5  * This program is free software: you can redistribute it and/or modify
     6  * it under the terms of the GNU General Public License as published by
     7  * the Free Software Foundation, version 3 of the License.
     8  * 
     9  * This program is distributed in the hope that it will be useful,
    10  * but WITHOUT ANY WARRANTY; without even the implied warranty of
    11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
    12  * GNU General Public License for more details.
    13  * 
    14  * You should have received a copy of the GNU General Public License
    15  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
    16  */
    17 package cz.frantovo.xmlWebGenerator.makra;
    18 
    19 import java.io.File;
    20 import java.io.IOException;
    21 import java.io.PrintStream;
    22 import java.net.URLDecoder;
    23 import java.nio.charset.Charset;
    24 import static cz.frantovo.xmlWebGenerator.NástrojeCLI.*;
    25 
    26 /**
    27  * Diagramy
    28  * 
    29  * @author František Kučera (frantovo.cz)
    30  */
    31 public class Diagram {
    32 
    33 	private static final String ADRESÁŘ_VÝSTUPNÍ = "výstup";
    34 	private static final String PŘÍKAZ_DOT = "dot";
    35 	private static int počítadloDiagramů = 0;
    36 	private static String počítadloDiagramůKontext = ""; // aktuálně zpracovávaná stránka, při změně vynulujeme počítadlo
    37 
    38 	/**
    39 	 * Vytvoří obrázek s diagramem.
    40 	 * @param zadání definice diagramu ve formátu dot
    41 	 * @param vodorovně zda má být graf orientovaný vodorovně (funguje jen při <code>kompletní = false</code>)
    42 	 * @param kompletní false, pokud k zadání chceme doplnit <code>digraph d {…}</code>
    43 	 * @param kontext kam diagram patří – typicky název stránky, do které je vložen
    44 	 * diagramy se pak budou číslovat v rámci tohoto kontextu 
    45 	 * → nebude docházet k přepisování diagramů jiných stránek při částečném přegenerování webu.
    46 	 * @param souborZadání null pokud chceme automatické číslování | nebo zadáme název souboru se zadáním diagramu – vygenerovaný diagram se pak bude jmenovat stejně
    47 	 * @return název souboru bez přípony, který byl vytvořen, nebo null, pokud došlo k chybě.
    48 	 */
    49 	public static String vytvořDiagram(String zadání, boolean vodorovně, boolean kompletní, String kontext, String souborZadání) throws IOException, InterruptedException {
    50 		if (isPříkazDostupný(PŘÍKAZ_DOT)) {
    51 
    52 			String soubor;
    53 			if (souborZadání == null) {
    54 				if (kontext == null) {
    55 					počítadloDiagramů++;
    56 					soubor = "diagram-" + počítadloDiagramů;
    57 				} else {
    58 					// TODO: tohle by se mělo udělat v XSLT
    59 					kontext = URLDecoder.decode(kontext, Charset.defaultCharset().name());
    60 
    61 					// Každá stránka bude mít svoje diagramy číslované od 1
    62 					if (!počítadloDiagramůKontext.equals(kontext)) {
    63 						počítadloDiagramůKontext = kontext;
    64 						počítadloDiagramů = 0;
    65 					}
    66 
    67 					počítadloDiagramů++;
    68 					soubor = "diagram-" + kontext + "-" + počítadloDiagramů;
    69 				}
    70 			} else {
    71 				soubor = souborZadání;
    72 			}
    73 			String souborSložka = ADRESÁŘ_VÝSTUPNÍ + File.separator + soubor;
    74 
    75 			String zdroják;
    76 			if (kompletní) {
    77 				zdroják = zadání;
    78 			} else {
    79 				StringBuilder b = new StringBuilder(zadání.length() + 200);
    80 				b.append("digraph d {\n");
    81 				b.append("\tbgcolor=\"transparent\";\n");
    82 				if (vodorovně) {
    83 					b.append("\trankdir=LR;");
    84 				}
    85 				b.append(zadání);
    86 				b.append("}\n");
    87 				zdroják = b.toString();
    88 			}
    89 
    90 			Runtime r = Runtime.getRuntime();
    91 			Process p = r.exec(new String[]{PŘÍKAZ_DOT, "-T", "svg", "-o", souborSložka + ".svg"});
    92 
    93 			/**
    94 			 * TODO: generovat i PNG bitmapu
    95 			 */
    96 			PrintStream vstupProcesu = new PrintStream(p.getOutputStream());
    97 			vstupProcesu.print(zdroják);
    98 			vstupProcesu.close();
    99 
   100 			String chyby = načtiProud(p.getErrorStream());
   101 
   102 			p.waitFor();
   103 
   104 			if (chyby.length() == 0) {
   105 				return soubor;
   106 			} else {
   107 				System.err.print("Při vytváření diagramu došlo k chybě: " + chyby);
   108 				return null;
   109 			}
   110 		} else {
   111 			System.err.println("Příkaz " + PŘÍKAZ_DOT + " není na vašem systému dostupný → diagramy nelze vygreslit.");
   112 			System.err.println("Můžete ho nainstalovat pomocí:");
   113 			System.err.println("\t$ aptitude install graphviz   # (Debian/Ubuntu)");
   114 			System.err.println("\t$ yum install graphviz        # (Fedora/RedHat)");
   115 			return null;
   116 		}
   117 	}
   118 }
   119