]> git.cameronkatri.com Git - mandoc.git/blobdiff - eqn_html.c
Explicit block closure macros clobber next-line block head scope,
[mandoc.git] / eqn_html.c
index fd77b9193ffa157f999a139f03c6e44b0eeab601..f29733613bb96386809ffb89c2255c0d9118d17d 100644 (file)
@@ -1,6 +1,6 @@
-/*     $Id: eqn_html.c,v 1.5 2014/09/28 13:34:15 kristaps Exp $ */
+/*     $Id: eqn_html.c,v 1.10 2014/10/12 19:31:41 schwarze Exp $ */
 /*
- * Copyright (c) 2011 Kristaps Dzonsons <kristaps@bsd.lv>
+ * Copyright (c) 2011, 2014 Kristaps Dzonsons <kristaps@bsd.lv>
  *
  * Permission to use, copy, modify, and distribute this software for any
  * purpose with or without fee is hereby granted, provided that the above
 #include "out.h"
 #include "html.h"
 
-static const enum htmltag fontmap[EQNFONT__MAX] = {
-       TAG_SPAN, /* EQNFONT_NONE */
-       TAG_SPAN, /* EQNFONT_ROMAN */
-       TAG_B, /* EQNFONT_BOLD */
-       TAG_B, /* EQNFONT_FAT */
-       TAG_I /* EQNFONT_ITALIC */
-};
-
-static const struct eqn_box *
-       eqn_box(struct html *, const struct eqn_box *, int);
-
-
-void
-print_eqn(struct html *p, const struct eqn *ep)
-{
-       struct htmlpair  tag;
-       struct tag      *t;
-
-       PAIR_CLASS_INIT(&tag, "eqn");
-       t = print_otag(p, TAG_MATH, 1, &tag);
-
-       p->flags |= HTML_NONOSPACE;
-       eqn_box(p, ep->root, 1);
-       p->flags &= ~HTML_NONOSPACE;
-
-       print_tagq(p, t);
-}
-
-/*
- * This function is fairly brittle.
- * This is because the eqn syntax doesn't play so nicely with recusive
- * formats, e.g.,
- *     foo sub bar sub baz
- * ...needs to resolve into
- *     <msub> foo <msub> bar, baz </msub> </msub>
- * In other words, we need to embed some recursive work.
- * FIXME: this does NOT handle right-left associativity or precedence!
- */
-static const struct eqn_box *
-eqn_box(struct html *p, const struct eqn_box *bp, int next)
+static void
+eqn_box(struct html *p, const struct eqn_box *bp)
 {
-       struct tag      *post, *pilet, *tmp;
+       struct tag      *post, *row, *cell, *t;
        struct htmlpair  tag[2];
-       int              skiptwo;
+       const struct eqn_box *child, *parent;
+       size_t           i, j, rows;
 
        if (NULL == bp)
-               return(NULL);
+               return;
 
-       post = pilet = NULL;
-       skiptwo = 0;
-
-       /*
-        * If we're a "row" under a pile, then open up the piling
-        * context here.
-        * We do this first because the pile surrounds the content of
-        * the contained expression.
-        */
-       if (NULL != bp->parent && bp->parent->pile != EQNPILE_NONE) {
-               pilet = print_otag(p, TAG_MTR, 0, NULL);
-               print_otag(p, TAG_MTD, 0, NULL);
-       }
+       post = NULL;
 
        /*
-        * If we're establishing a pile, start the table mode now.
-        * If we've already in a pile row, then don't override "pilet",
-        * because we'll be closed out anyway.
+        * Special handling for a matrix, which is presented to us in
+        * column order, but must be printed in row-order.
         */
-       if (bp->pile != EQNPILE_NONE) {
-               tmp = print_otag(p, TAG_MTABLE, 0, NULL);
-               pilet = (NULL == pilet) ? tmp : pilet;
+       if (EQN_MATRIX == bp->type) {
+               if (NULL == bp->first)
+                       goto out;
+               if (EQN_LIST != bp->first->type) {
+                       eqn_box(p, bp->first);
+                       goto out;
+               }
+               if (NULL == (parent = bp->first->first))
+                       goto out;
+               /* Estimate the number of rows, first. */
+               if (NULL == (child = parent->first))
+                       goto out;
+               for (rows = 0; NULL != child; rows++)
+                       child = child->next;
+               /* Print row-by-row. */
+               post = print_otag(p, TAG_MTABLE, 0, NULL);
+               for (i = 0; i < rows; i++) {
+                       parent = bp->first->first;
+                       row = print_otag(p, TAG_MTR, 0, NULL);
+                       while (NULL != parent) {
+                               child = parent->first;
+                               for (j = 0; j < i; j++) {
+                                       if (NULL == child)
+                                               break;
+                                       child = child->next;
+                               }
+                               cell = print_otag
+                                       (p, TAG_MTD, 0, NULL);
+                               /*
+                                * If we have no data for this
+                                * particular cell, then print a
+                                * placeholder and continue--don't puke.
+                                */
+                               if (NULL != child)
+                                       eqn_box(p, child->first);
+                               print_tagq(p, cell);
+                               parent = parent->next;
+                       }
+                       print_tagq(p, row);
+               }
+               goto out;
        }
 
-       /*
-        * Positioning.
-        * This is the most complicated part, and actually doesn't quite
-        * work (FIXME) because it doesn't account for associativity.
-        * Setting "post" will mean that we're only going to process a
-        * single or double following expression.
-        */
        switch (bp->pos) {
+       case (EQNPOS_TO):
+               post = print_otag(p, TAG_MOVER, 0, NULL);
+               break;
        case (EQNPOS_SUP):
                post = print_otag(p, TAG_MSUP, 0, NULL);
                break;
        case (EQNPOS_FROM):
-               /* FALLTHROUGH */
+               post = print_otag(p, TAG_MUNDER, 0, NULL);
+               break;
        case (EQNPOS_SUB):
                post = print_otag(p, TAG_MSUB, 0, NULL);
                break;
        case (EQNPOS_OVER):
                post = print_otag(p, TAG_MFRAC, 0, NULL);
                break;
+       case (EQNPOS_FROMTO):
+               post = print_otag(p, TAG_MUNDEROVER, 0, NULL);
+               break;
        case (EQNPOS_SUBSUP):
-               /* This requires two elements. */
                post = print_otag(p, TAG_MSUBSUP, 0, NULL);
-               skiptwo = 1;
+               break;
+       case (EQNPOS_SQRT):
+               post = print_otag(p, TAG_MSQRT, 0, NULL);
                break;
        default:
                break;
        }
 
-       /*t = EQNFONT_NONE == bp->font ? NULL :
-           print_otag(p, fontmap[(int)bp->font], 0, NULL);*/
+       if (bp->top || bp->bottom) {
+               assert(NULL == post);
+               if (bp->top && NULL == bp->bottom)
+                       post = print_otag(p, TAG_MOVER, 0, NULL);
+               else if (bp->top && bp->bottom)
+                       post = print_otag(p, TAG_MUNDEROVER, 0, NULL);
+               else if (bp->bottom)
+                       post = print_otag(p, TAG_MUNDER, 0, NULL);
+       }
+
+       if (EQN_PILE == bp->type) {
+               assert(NULL == post);
+               if (bp->first != NULL && bp->first->type == EQN_LIST)
+                       post = print_otag(p, TAG_MTABLE, 0, NULL);
+       } else if (bp->type == EQN_LIST &&
+           bp->parent && bp->parent->type == EQN_PILE) {
+               assert(NULL == post);
+               post = print_otag(p, TAG_MTR, 0, NULL);
+               print_otag(p, TAG_MTD, 0, NULL);
+       }
 
        if (NULL != bp->text) {
-               assert(NULL == bp->first);
-               /*
-                * We have text.
-                * This can be a number, a function, a variable, or
-                * pretty much anything else.
-                * First, check for some known functions.
-                * If we're going to create a structural node (e.g.,
-                * sqrt), then set the "post" variable only if it's not
-                * already set.
-                */
-               if (0 == strcmp(bp->text, "sqrt")) {
-                       tmp = print_otag(p, TAG_MSQRT, 0, NULL);
-                       post = (NULL == post) ? tmp : post;
-               } else if (0 == strcmp(bp->text, "+") ||
-                          0 == strcmp(bp->text, "-") ||
-                          0 == strcmp(bp->text, "=") ||
-                          0 == strcmp(bp->text, "(") ||
-                          0 == strcmp(bp->text, ")") ||
-                          0 == strcmp(bp->text, "/")) {
-                       tmp = print_otag(p, TAG_MO, 0, NULL);
-                       print_text(p, bp->text);
-                       print_tagq(p, tmp);
-               } else {
-                       tmp = print_otag(p, TAG_MI, 0, NULL);
-                       print_text(p, bp->text);
-                       print_tagq(p, tmp);
-               }
-       } else if (NULL != bp->first) {
-               assert(NULL == bp->text);
-               /* 
-                * If we're a "fenced" component (i.e., having
-                * brackets), then process those brackets now.
-                * Otherwise, introduce a dummy row (if we're not
-                * already in a table context).
-                */
-               tmp = NULL;
+               assert(NULL == post);
+               post = print_otag(p, TAG_MI, 0, NULL);
+               print_text(p, bp->text);
+       } else if (NULL == post) {
                if (NULL != bp->left || NULL != bp->right) {
                        PAIR_INIT(&tag[0], ATTR_OPEN,
-                               NULL != bp->left ? bp->left : "");
+                           NULL == bp->left ? "" : bp->left);
                        PAIR_INIT(&tag[1], ATTR_CLOSE,
-                               NULL != bp->right ? bp->right : "");
-                       tmp = print_otag(p, TAG_MFENCED, 2, tag);
+                           NULL == bp->right ? "" : bp->right);
+                       post = print_otag(p, TAG_MFENCED, 2, tag);
+               }
+               if (NULL == post)
+                       post = print_otag(p, TAG_MROW, 0, NULL);
+               else
                        print_otag(p, TAG_MROW, 0, NULL);
-               } else if (NULL == pilet)
-                       tmp = print_otag(p, TAG_MROW, 0, NULL);
-               eqn_box(p, bp->first, 1);
-               if (NULL != tmp)
-                       print_tagq(p, tmp);
        }
 
-       /*
-        * If a positional context, invoke the "next" context.
-        * This is recursive and will return the end of the recursive
-        * chain of "next" contexts.
-        */
-       if (NULL != post) {
-               bp = eqn_box(p, bp->next, 0);
-               if (skiptwo)
-                       bp = eqn_box(p, bp->next, 0);
-               print_tagq(p, post);
+       eqn_box(p, bp->first);
+
+out:
+       if (NULL != bp->bottom) {
+               t = print_otag(p, TAG_MO, 0, NULL);
+               print_text(p, bp->bottom);
+               print_tagq(p, t);
+       }
+       if (NULL != bp->top) {
+               t = print_otag(p, TAG_MO, 0, NULL);
+               print_text(p, bp->top);
+               print_tagq(p, t);
        }
 
-       /* 
-        * If we're being piled (either directly, in the table, or
-        * indirectly in a table row), then close that out.
-        */
-       if (NULL != pilet)
-               print_tagq(p, pilet);
+       if (NULL != post)
+               print_tagq(p, post);
 
-       /*
-        * If we're normally processing, then grab the next node.
-        * If we're in a recursive context, then don't seek to the next
-        * node; further recursion has already been handled.
-        */
-       return(next ? eqn_box(p, bp->next, 1) : bp);
+       eqn_box(p, bp->next);
+}
+
+void
+print_eqn(struct html *p, const struct eqn *ep)
+{
+       struct htmlpair  tag;
+       struct tag      *t;
+
+       PAIR_CLASS_INIT(&tag, "eqn");
+       t = print_otag(p, TAG_MATH, 1, &tag);
+
+       p->flags |= HTML_NONOSPACE;
+       eqn_box(p, ep->root);
+       p->flags &= ~HTML_NONOSPACE;
+
+       print_tagq(p, t);
 }