2d/2d-doc/scribblings/2d.scrbl
Robby Findler a350659906 export some readtable functions
and finish up the docs
2016-07-31 02:51:48 -05:00

350 lines
15 KiB
Racket

#lang scribble/doc
@(require scribble/base
scribble/manual
scribble/core
scribble/example
(for-label 2d/cond
2d/match
syntax-color/lexer-contract
racket/file
racket/contract
racket/base))
@title[#:tag "2d"]{2D Syntax}
@defmodulelang[2d]{The @racketmodname[2d] language installs
@litchar{#2d} reader support in the
@tech[#:doc '(lib "scribblings/reference/reference.scrbl")]{readtables},
and then chains to the reader of
another language that is specified immediately after
@racketmodname[2d].}
The @litchar{#2d} syntax extension adds the ability use
a two-dimensional grid syntax. That is, you can draw an ASCII-art
grid and then treat that as an expression. For example,
here is a simple equality function that operates on pairs and
numbers, written using a @litchar{#2d} conditional expression:
@codeblock{
#lang 2d racket
(require 2d/cond)
(define (same? a b)
#2dcond
╔═════════════╦═══════════════════════╦═════════════╗
(pair? a) (number? a)
╠═════════════╬═══════════════════════╬═════════════╣
(pair? b) (and (same? (car a) #f
(car b))
(same? (cdr a)
(cdr b)))
╠═════════════╬═══════════════════════╬═════════════╣
(number? b) #f (= a b)
╚═════════════╩═══════════════════════╩═════════════╝)
}
This notation works in two stages: reading, and parsing (just as in
Racket in general). The reading stage converts anything that begins
with @litchar{#2d} into a parenthesized expression (possibly signaling
errors if the @litchar{} and @litchar{} and @litchar{}
characters do not line up in the right places).
Since the first line contains @litchar{#2dcond}, the reader will
produce a sequence whose first position is the identifier @racket[2dcond].
That macro will take over and then expand into ordinary conditional
expressions, in this case figuring out whether or not the inputs
are pairs or numbers and evaluating the code in the appropriate cell.
At the reader level, the syntax @litchar{#2d} notation checks
the number of columns in the first row and uses that as a guide
for where subsequent rows may appear. Once that first row is set,
it serves as a guide to where the columns may appear in subsequent
rows, although following columns may be merged.
This merging can simplify
some uses of @litchar{#2d} expressions. For example, consider this
expression that captures subtyping relationships between a few of the
Typed Racket numeric types, this time using a @litchar{#2d} match
expression:
@codeblock{
#lang 2d racket
(require 2d/match)
(define (subtype? a b)
#2dmatch
╔══════════╦══════════╦═══════╦══════════╗
a b 'Integer 'Real 'Complex
╠══════════╬══════════╩═══════╩══════════╣
'Integer #t
╠══════════╬══════════╗
'Real
╠══════════╣ ╚═══════╗
'Complex #f
╚══════════╩══════════════════╩══════════╝)
}
There are a number of cell walls missing here, but this is still a
well-formed @litchar{#2d} expression. In this case, the @racket[2dmatch]
treats any of the situations that fall into the larger regions as
the same.
In general, a @litchar{#2d} expression, when read, turns into an expression
with at least two sub-pieces (not counting the initial name). The first is
a sequence of numbers giving the widths of the top row of cells;
the second is also a sequence of numbers, this time giving the heights
of the leftmost column of cells. The remaining sequence describe the cells
content. The first element of each is itself a sequence of coordinates,
one for each of the cells that are connected together. The remaining elements
are the subexpressions in the given cells.
For example, this:
@codeblock{
#lang 2d racket
'#2dex
╔══════════╦══════════╗
0 1
╠══════════╬══════════╣
2 3
╚══════════╩══════════╝
}
evaluates to
@racketblock['(2dex (10 10)
(2 2)
(((0 0)) 0)
(((0 1)) 2)
(((1 0)) 1)
(((1 1)) 3))]
and this
@codeblock{
#lang 2d racket
'#2dex
╔══════════╦══════════╦══════════╗
0 1 2 3 4
╠══════════╬══════════╩══════════╣
5 6
╚══════════╩═════════════════════╝
}
evaluates to
@racketblock['(2dex (10 10 10)
(2 2)
(((0 0)) 0)
(((0 1)) 5)
(((1 0)) 1 2)
(((1 1) (2 1)) 6)
(((2 0)) 3 4))]
In addition, the cells coordinates pairs have source locations of the first
character that is inside the corresponding cell. (Currently the span
is always @racket[1], but that may change.)
@section{Editing 2D}
DrRacket provides a number of keybindings to help editing @litchar{#2d} expressions.
See @seclink["Keyboard Shortcuts" #:doc '(lib "scribblings/drracket/drracket.scrbl") #:indirect? #t]{DrRacket's keyboard shortcuts}.
@section{2D Cond}
@defmodule[2d/cond]
@defform/subs[(2dcond cond-content)
([cond-content (code:line question-row
body-row
)
(code:line question-row
body-row
else-row)]
[question-row (code:line empty-cell question-cell )
(code:line empty-cell question-cell else-cell)]
[body-row (code:line question-cell exprs-cell )]
[else-row (code:line question-cell exprs-cell else-cell)]
[question-cell (code:line ╔═════════════╗
║question-expr║
╚═════════════╝)]
[empty-cell (code:line ╔═══╗
╚═══╝)]
[exprs-cell (code:line ╔═════════════╗
║expr expr ...║
╚═════════════╝)]
[else-cell (code:line ╔══════╗
else
╚══════╝)])]{
Evaluates the first row of question expressions until
one of them returns a true value (signaling an error if none do),
then evaluates the first column of question expressions until
one of them returns a true value (signaling an error if none do),
and then evaluates the cell in the middle where both point to,
returning the result of the last expression in that cell.
}
@section{2D Match}
@defmodule[2d/match]
@defform/subs[(2dmatch match-content)
([match-content (code:line match-first-row
match-row
)]
[match-first-row (code:line two-expr-cell match-pat-cell )]
[match-row (code:line match-pat-cell exprs-cell )]
[two-expr-cell (code:line ╔═════════════════╗
║col-expr row-expr║
╚═════════════════╝)]
[match-pat-cell (code:line ╔═════╗
pat
╚═════╝)]
[exprs-cell (code:line ╔═════════════╗
║expr expr ...║
╚═════════════╝)])]{
Matches @racket[col-expr] against each of patterns
in the first column of the table and matches @racket[row-expr]
against each of the patterns in the row row, and then evaluates
the corresponding @racket[exprs-cell], returning the value of the
last expression in that cell.
Within the top-left cell, the leftmost expression will count as
@racket[col-expr], and the rightmost as @racket[row-expr]. In case of a tie
(i.e., both expressions start at the same column, but on different lines),
the bottommost one will count as @racket[col-expr]. For example, all of
these are valid:
@racketblock[╔═════════════════╗
║col-expr row-expr║
╚═════════════════╝]
@racketblock[╔═════════════════╗
row-expr║
║col-expr
╚═════════════════╝]
@racketblock[╔════════╗
║row-expr║
║col-expr║
╚════════╝]
@history[#:changed "6.4"]{Made scrutinee parsing more flexible.}
}
@section{2D Tabular}
@defmodule[2d/tabular]
@defform/subs[(2dtabular tabular-content)
([tabular-content (code:line tabular-row
)
(code:line tabular-row
style-cell)]
[tabular-row (code:line tabular-cell )]
[tabular-cell (code:line ╔════════════════╗
║tabular-expr ...║
╚════════════════╝)]
[style-cell (code:line ╔═════════════════╗
║style-content ...║
╚═════════════════╝)]
[style-content (code:line #:style style-expr)
(code:line #:sep sep-expr)
#:ignore-first-row])
#:contracts ([style-expr style?]
[sep-expr (or/c block? content? #f)]
[tabular-expr (or/c block? content?)])]{
Constructs a @racket[tabular] matching the given cells.
If a cell spans multiple columns, then the resulting
@racket[tabular] has @racket['cont] in the corresponding
list element. No cells may span rows.
The @racket[#:style] and @racket[#:sep] arguments are just passed
to @racket[tabular].
If the @racket[#:ignore-first-row] keyword is provided, then the first
row of the @racket[2dtabular] expression is ignored. This can be used
in case the first row of the rendered table should not have all of the
columns (as @litchar{#2d} syntax requires that the first row contain
a cell for each column that appears in the table).
}
@section{2D Readtable}
@defmodule[2d/readtable]
@defproc[(make-readtable) readtable?]{
Builds a @racket[readtable?] that recognizes @litchar{#2d} and turns it into
a parenthesized form as discussed in @secref["2d"].
}
@defproc[(2d-readtable-dispatch-proc
[char char?]
[port input-port?]
[source any/c]
[line (or/c exact-positive-integer? #f)]
[column (or/c exact-nonnegative-integer? #f)]
[position (or/c exact-positive-integer? #f)]
[/recursive (-> input-port? any/c (or/c readtable? #f) any/c)]
[readtable (or/c #f readtable?)])
any/c]{
The function that implements @racket[make-readtable]'s functionality. The
@racket[/recursive] function is used to handle the content in the cells.
See the docs
on @tech[#:doc '(lib "scribblings/reference/reference.scrbl")]{readtables} for more information.
}
@section{2d Lexer}
@defmodule[2d/lexer]
@defproc[(2d-lexer [sub lexer/c]) lexer/c]{
Constructs a @racket[lexer/c] given one that handles
lexing inside the cells.
}
@section{2D Direction Chars}
@defmodule[2d/dir-chars]
@(define 2dchars-eval (make-base-eval '(require 2d/dir-chars)))
This library provides definitions of the characters that are looked for when
parsing 2d syntax.
@(define-syntax-rule
(doc-chars id . stuff)
(begin
@(defthing id (listof char?) . stuff)
@examples[#:label #f #:eval 2dchars-eval id]))
@doc-chars[adjustable-chars]{
These are the characters that are considered either to be part of 2d rectangle
or characters that could be part of one, possibly fixed by up a DrRacket keybinding.
}
@doc-chars[double-barred-chars]{
These are all of the @racket[adjustable-chars], except those that are regular ASCII.
}
@doc-chars[up-chars]{
All of the 2d chars that connect to the line above.
}
@doc-chars[dn-chars]{
All of the 2d chars that connect to the line below.
}
@doc-chars[lt-chars]{
All of the 2d chars that connect to the next char.
}
@doc-chars[rt-chars]{
All of the 2d chars that connect to the previous char.
}