NAME
Class::Declarative - Provides a declarative framework for Perl
VERSION
Version 0.07
SYNOPSIS
This module is a framework for writing Perl code in a declarative manner. What that means right now is that instead of seeing a script as a series of actions to be carried out, you can view the script as a set of objects to be instantiated, then invoked. The syntax for building these objects is intended to be concise and flexible, mostly staying out of your way. Perl code is used to declare actions to be taken once the structure is built, as well as any actions to be taken interactively as the script runs.
The original motivation for designing this framework was to provide a more rational way of defining a Wx user interface. As it is, the data structures making up a Wx GUI are built with painstakingly detailed (and boring) imperative code. There are XML-based GUI specification frameworks, but I wanted to write my own that wasn't XML-based because I hate typing XML even more than I hate writing setup code.
Back when I did a lot of GUI work, I'd usually write some pseudocode to describe parts of the UI, then translate it into code by hand. So this year, while noodling around about some tools I'd find useful in my translation business, I thought, well, why not just write a class to interpret that pseudocode description directly?
Once I started getting into that in earnest, I realized that the Wx-specific functionality could be spun out into an application-specific (in my new parlance, a "semantic") domain, leaving a core set of functionality that was a general declarative framework. I then realized that the same framework could easily be used to work with domains other than Wx GUIs, such as building PDFs, building Flash applications, doing things with Word documents... All kinds of things. All of those things are currently in pieces on the workbench - except for the Word module, which is ready, if not for prime time, then at least for deep cable midnight airing.
Here's a GUI example using something like the Wx domain. This is a pretty simple example, but it gives you a taste of what I'm talking about. Since Class::Declarative runs as a source filter, the example below is a working Perl script that replaces roughly 80 lines of the Wx example code it was adapted from. And yes, it runs in my test suite right now.
use Wx::Declarative;
dialog (xsize=250, ysize=110) "Wx::Declarative dialog sample"
field celsius (size=100, x=20, y=20) "0"
button celsius (x=130, y=20) "Celsius" { $^fahrenheit = ($^celsius / 100.0) * 180 + 32; }
field fahrenheit (size=100, x=20, y=50) "32"
button fahrenheit (x=130, y=50) "Fahrenheit" { $^celsius = (($^fahrenheit - 32) / 180.0) * 100; }
The main things to look at are as follows: first, yes - syntactically significant indentation. I know it's suspiciously Pythonic, I know all the arguments citing the danger of getting things to line up, and I don't care; this is the way I have always written my pseudocode, and odds are you're no different and you know it. If it makes you feel better, the indentation detection algorithm is pretty flexible, and Perl code within curly braces is exempt from indentation significance. (Not that this example has any multiline code, but you see what I mean.)
Second, fields are declared here and their content is exposed as magic variables in the code snippets. You will immediately see that code embedded in a declarative structure goes through a modification pass before being eval
'd into a sub. So there is a possibility that I have screwed that modification pass up. I don't have an answer for this right now; the point is quick and easy, not perfection (yet). Caveat emptor. It's still a neat feature.
There is a standard parser and standard data structure available for tags to use if it suits your purpose - but there's no mandate to use them, and the parser tools are open for use. They're still a little raw, but pretty powerful.
A declarative object can report its own source code, and that source code can compile into an equivalent declarative object. This means that dynamically constructed objects or applications can be written out as executable code, and code has introspective capability while in the loaded state. Class::Declarative
also has a macro system that allows the construction of code during the build phase; a macro always dumps as its source, not the result of the expansion, so you can capture dynamic behavior that runs dynamically every time.
TUTORIAL
For more information about how to use Class::Declarative
, you'll probably want to see Class::Declarative::Tutorial instead of this file; the rest of this presentation is devoted to the internal workings of Class::Declarative
. (Old literate programming habits, I guess.) Honestly, you can probably just stop here, because if you're not reading the source with the POD it probably won't make any sense anyway. Go read the tutorial. Not that I've finished it.
SETTING UP THE CLASS STRUCTURE
import, yes_i_am_declarative
The import
function is called when the package is imported. It's used for the filter support; don't call it.
If semantic classes are supplied in the use
command, we're going to instantiate and scan them here. They'll be used to decorate the parse tree appropriately.
class_build_handler ($string, $hash), app_build_handler ($string, $hash), build_handler ($string)
Given a tag name, class_build_handler
returns a hashref of information about how the tag expects to be treated:
* The class its objects should be blessed into, as a coderef to generate the object ('Class::Declarative::Node' is the default) * Its line parser, by name ('default-line' is the default) * Its body parser, by name ('default-body' is the default) * A second-level hashref of hashrefs providing overriding semantics for descendants of this tag.
If you also provide a hashref, it is assigned to the tag name.
The app_build_handler
does the same thing, but specific to the given application - this allows dynamic tag definition.
Finally, build_handler
is a read-only lookup for a tag in the context of its ancestry that climbs the tree to find the contextual semantics for the tag.
makenode($ancestry, $code)
Finds the right build handler for the tag in question, then builds the right class of node with the code given.
known_tags()
Returns the keys of the build handler hash. This is probably a fossil.
FILTERING SOURCE CODE
By default, Class::Declarative
runs as a filter. That means it intercepts code coming in and can change it before Perl starts parsing. Needless to say, filters act very cautiously, because the only thing that can parse Perl correctly is Perl (and sometimes even Perl has doubts). So this filter basically just wraps the entire input source in a call to new
, which is then parsed and called after the filter returns.
filter
The filter
function is called by the source code filtering process. You probably don't want to call it. But if you've ever wondered how difficult it is to write a source code filter, read it. Hint: it really isn't difficult.
PARSERS
The parsing process in Class::Declarative
is recursive. The basic form is a tagged line followed by indented text, followed by another tagged line with indented text, and so on. Alternatively, the indented part can be surrounded by brackets.
tag [rest of line]
indented text
indented text
indented text
tag [rest of line] {
bracketed text
bracketed text
}
By default, each tag parses its indented text in the same way, and it's turtles all the way down. Bracketed text, however, is normally not parsed as declarative (or "nodal") structure, but is left untouched for special handling, typically being parsed by Perl and wrapped as a closure.
However, all this is merely the default. Any tag may also specify a different parser for its own indented text, or may carry out some transformation on the text before invoking the parser. It's up to the tag. The data
tag, for instance, treats each indented line as a row in a table.
Once the body is handled, the "rest of line" is also parsed into data useful for the node. Again, there is a default parser, which takes a line of the following form:
tag name (parameter, parameter=value) [option, option=value] "label or other string text" parser < { bracketed text }
Any element of that line may be omitted, except for the tag.
init_parsers(), including locally defined is_blank, is_blank_or_comment, and line_indentation
Sets up the registry and builds our default line and body parsers.
init_default_line_parser(), init_default_body_parser(), init_locator_parser()
These are called by init_parsers
to initialize our various sublanguage parsers. You don't need to call them.
parser($name)
Retrieves a parser from the registry.
parse_line ($node)
Given a node, finds the line parser for it, and runs it on the node's line.
parse($node, $body)
Given a node and body text for it, finds the body parser appropriate to the node's tag and runs it on the node and the body text specified.
parse_using($string, $parser)
Given a string and the name of a parser, calls the parser on the string and returns the result.
BUILDING AND MANAGING THE APPLICATION
You'd think this would be up at the top, but we had to do a lot of work just to be ready to instantiate a Class::Declarative
object.
new
The new
function is of course called to create a new Class::Declarative
object. If you pass it some code, it will load that code immediately.
semantic_handler ($tag)
Returns the instance of a semantic module, such as 'core' or 'wx'.
start
This is called from outside to kick off the process defined in this application. The way we handle this is just to ask the first semantic class to start itself. The idea there being that it's probably going to be Wx or something that provides the interface. (It could also be a Web server or something.)
The core semantics just execute all the top-level items that are flagged callable.
id($idstring)
Wx works with numeric IDs for events, and I presume the other event-based systems do, too. I don't like numbers; they're hard to read and tell apart. So Class::Declarative
registers event names for you, assigning application-wide unique numeric IDs you can use in your payload objects.
root()
Returns $self; for nodes, returns the parent. The upshot is that by calling root
we can get the root of the tree, fast.
describe([$use])
Returns a reconstructed set of source code used to compile this present Class::Declarative
object. If it was assembled in parts, you still get the whole thing back. Macro results are not included in this dump (they're presumed to be the result of macros in the tree itself, so they should be regenerated the next time anyway).
If you specify a true value for $use, the dump will include a "use" statement at the start in order to make the result an executable Perl script. The dump is always in filter format (if you built it with -nofilter) and contains Class::Declarative
's best guess of the semantic modules used. If you're using a "use lib" to affect your %INC, the result won't work right unless you modify it, but if it's all standard modules, the dump result, after loading, should work the same as the original entry.
find_data
The find_data
function finds a top-level data node.
AUTHOR
Michael Roberts, <michael at vivtek.com>
BUGS
Please report any bugs or feature requests to bug-class-declarative at rt.cpan.org
, or through the web interface at http://rt.cpan.org/NoAuth/ReportBug.html?Queue=Class-Declarative. I will be notified, and then you'll automatically be notified of progress on your bug as I make changes.
SUPPORT
You can find documentation for this module with the perldoc command.
perldoc Class::Declarative
You can also look for information at:
RT: CPAN's request tracker
AnnoCPAN: Annotated CPAN documentation
CPAN Ratings
Search CPAN
ACKNOWLEDGEMENTS
LICENSE AND COPYRIGHT
Copyright 2010 Michael Roberts.
This program is free software; you can redistribute it and/or modify it under the terms of either: the GNU General Public License as published by the Free Software Foundation; or the Artistic License.
See http://dev.perl.org/licenses/ for more information.