Curl Global Community

Full Version: Curl macros introduction (part 1)
You're currently viewing a stripped down version of our content. View the full version with proper formatting.
This series is about Curl macros

Macros allow you to create both very simple and very complex results in Curl web pages, in Curl applications, and in Curl scripts. Most often they hide complexity from the user of the macro and keep markup simple.

You use a macro in Curl just as you would use any other Curl expression.
Examples: are any macros? Suppose We Wrote:{paragraph This is a text procedure }
{my-para This might be a macro or a user-defined text-format or text-proc }
{ml-para "Using quotes is a clue that it might be a proc, but it could be a macro"}
{paragraph Usually only by looking in the docs or user code will you know.}

Default Curl expressions (from the Curl library packages) which begin with {define- are macros. The Curl documentation clearly indicates which expression prefixes are macros both in the index and the respective macro pages. The live-code in the docs in built using the {example } macro.

Expressions with arguments not separated by commas or separated by other characters would be macros (but I suggest using commas in your macros and in what follows I try to show why.)

A macro is used to define new macros: it is {define-macro } and it gets its power from two other macros, those being {syntax-switch } and {expand-template }.

A macro call is replaced by the source code corresponding to the value which the macro returns (although macros do not declare a return value, they use {return } expressions.

In a package write:
Code:
{define-macro public {uc ?txt:text}
    {return {expand-template {{String ?txt}.to-upper-clone}}}
}

In an applet write:
Code:
{import uc from MY.MACROS}
{uc "this quoted text will print in caps without the quotes"}

In the browser we see Curl Wrote:THIS QUOTED TEXT WILL PRINT IN CAPS WITHOUT THE QUOTES

Note: Curl is an expression-based language, but macros may identify some expressions as statements. Examples are let and set expressions which were not surrounded by curly-braces for your convenience or other expressions where curl-braces were not required. Macro pattern case logic can recognize MyClass.myClassProc as a BinaryOp. But we will set all these details to one side for now.

When passed quoted characters, all of these variants have the same effect:
Code:
{define-macro public {uc ?txt:text}
    {return {expand-template {{String ?txt}.to-upper-clone}}}
}

{define-macro public {ucv ?txt:verbatim}
    {return {expand-template {{String ?txt}.to-upper-clone}}}
}

{define-macro public {uctk ?txt:token}
    {return {expand-template {{String ?txt}.to-upper-clone}}}
}

{define-macro public {uce ?txt:expression}
    {return {expand-template {{String ?txt}.to-upper-clone}}}
}

{define-macro public {ucs ?txt:statement}
    {return {expand-template {{String ?txt}.to-upper-clone}}}
}

When called as
Code:
{uc  "this quoted text will print in caps without the quotes"}
{br}
{ucv  "this quoted text will print in caps without any quotes"}
{br}
{uctk "this quoted text will print in caps without any quotes"}
{br}
{uce  "this quoted text will print in caps without any quotes"}
{br}
{ucs  "this quoted text will print in caps without any quotes"}
When passed a String identifier, all of these variants have the same effect (the proc is the clue that these simple macros are examples only):
Code:
{define-proc {up s:String}:String
    {return {s.to-upper-clone}}
}

{define-macro public {uc ?txt:text}
    {return {expand-template {up ?txt}}}
}

{define-macro public {ucv ?txt:verbatim}
    {return {expand-template {up ?txt}}}
}

{define-macro public {uctk ?txt:token}
    {return {expand-template {up ?txt}}}
}

{define-macro public {uce ?txt:expression}
    {return {expand-template {up ?txt}}}
}

{define-macro public {ucs ?txt:statement}
    {return {expand-template {up ?txt}}}
}

when called as
Code:
{def str:String = "testing macros"}

{uc   str}
{br}
{ucv  str}
{br}
{uctk str}
{br}
{uce  str}
{br}
{ucs  str}
But what happens if you attempt

Code:
{br}
{uc   "test " & str}
{br}
{ucv "test " & str}
{br}
{uce "test " & str}
{br}
{ucs "test " & str}
{br}
{uctk "test " & str}
{br}

???
The difficulty for understanding {value} is especially obvious when using at the top level of a curl applet file. To understand this, a user must know a DefaultDocument or PlainDocument will add an object without being defined with a varirable to it's layout container (a TextFlowBox) by default , while a variable definition will be kept as a variable only and value of this variable won't be added to it's layout container implicitly. So when you want a variable to be added to a DefaultDocument or PlainDocument, eithier use {value var} at the top level of the *Document, or wrap this variable with another Graphical container like a Fill or Frame or anything else, which is then instantiated directly at the top level of *Document without being assigned to any variable.