#region License Information /* HeuristicLab * Copyright (C) 2002-2016 Heuristic and Evolutionary Algorithms Laboratory (HEAL) * * This file is part of HeuristicLab. * * HeuristicLab is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * HeuristicLab is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with HeuristicLab. If not, see . */ #endregion using System; using System.Collections.Generic; using System.Drawing; using System.Globalization; using System.Linq; using System.Text; using HeuristicLab.Common; using HeuristicLab.Core; namespace HeuristicLab.Encodings.SymbolicExpressionTreeEncoding.Views { [Item("LaTeX/PDF Formatter", "Formatter for symbolic expression trees for use with latex package tikz.")] public class SymbolicExpressionTreeLatexFormatter : NamedItem, ISymbolicExpressionTreeStringFormatter { private readonly static Dictionary symbolNameMap = new Dictionary { {"ProgramRootSymbol", "Prog"}, {"StartSymbol","RPB"}, {"Multiplication", "$\\times$"}, {"Division", "$\\div$"}, {"Addition", "$+$"}, {"Subtraction", "$-$"}, {"Exponential", "$\\exp$"}, {"Logarithm", "$\\log$"} }; private readonly ReingoldTilfordLayoutEngine layoutEngine; public SymbolicExpressionTreeLatexFormatter() : base("LaTeX/PDF Formatter", "Formatter for symbolic expression trees for use with latex package tikz.") { layoutEngine = new ReingoldTilfordLayoutEngine(n => n.Subtrees) { HorizontalSpacing = 2, VerticalSpacing = 2, NodeWidth = 8, NodeHeight = 4 }; } protected SymbolicExpressionTreeLatexFormatter(SymbolicExpressionTreeLatexFormatter original, Cloner cloner) : base(original, cloner) { } public override IDeepCloneable Clone(Cloner cloner) { return new SymbolicExpressionTreeLatexFormatter(this, cloner); } public string Format(ISymbolicExpressionTree symbolicExpressionTree) { string s; Format(symbolicExpressionTree, out s); return s; } public Dictionary Format(ISymbolicExpressionTree symbolicExpressionTree, out string str, int offset = 0) { var nodeIds = new Dictionary(); var root = symbolicExpressionTree.Root; var actualRoot = root.SubtreeCount == 0 ? root.GetSubtree(0) : root; var nodeCoordinates = layoutEngine.CalculateLayout(actualRoot).ToDictionary(n => n.Content, n => new PointF(n.X, n.Y)); var sb = new StringBuilder(); var nl = Environment.NewLine; double ws = 1; double hs = 0.7; sb.Append("\\documentclass[class=minimal,border=0pt]{standalone}" + nl + "\\usepackage{tikz}" + nl + "\\begin{document}" + nl + "\\begin{tikzpicture}" + nl + "\\def\\ws{1}" + nl + "\\def\\hs{0.7}" + nl + "\\def\\offs{" + offset + "}" + nl); var nodes = symbolicExpressionTree.IterateNodesBreadth().ToList(); for (int i = 0; i < nodes.Count; ++i) { var node = nodes[i]; var id = Guid.NewGuid().ToString(); nodeIds[node] = id; var coord = nodeCoordinates[node]; var nodeName = symbolNameMap.ContainsKey(node.Symbol.Name) ? symbolNameMap[node.Symbol.Name] : node.ToString(); sb.AppendLine(string.Format(CultureInfo.InvariantCulture, "\\node ({0}) at (\\ws*{1} + \\offs,\\hs*{2}) {{{3}}};", nodeIds[node], ws * coord.X, -hs * coord.Y, EscapeLatexString(nodeName))); } for (int i = 0; i < nodes.Count; ++i) { var n = nodes[i]; foreach (var s in nodes[i].Subtrees) { sb.AppendLine(string.Format(CultureInfo.InvariantCulture, "\\draw ({0}) -- ({1});", nodeIds[n], nodeIds[s])); } } sb.Append("\\end{tikzpicture}" + nl + "\\end{document}" + nl); str = sb.ToString(); return nodeIds; } private static string EscapeLatexString(string s) { return s.Replace("\\", "\\\\").Replace("{", "\\{").Replace("}", "\\}").Replace("_", "\\_"); } } }