This is a suite of libraries and tools to work with EFL#-based XML UI files.
It will be composed of:
- Helper tool to extract API information from EFL# based libraries.
- XML validation library (Mostly done. Extra work on this will be done as we get feedback from the compiler tasks)
- Code generation compiler
- Other tools
EXML# currently uses Dotnet Core and EFL# (the latter for testing purposes for now, builtin) for building.
$ dotnet build
$ dotnet test
EXML is a simple XML format inspired by Xamarin Android's AXML. Elements are either Containers (elements with children containers or widgets, like Boxes, Spotlights, etc) or Widgets (like Button, Radio, etc).
Sample EXML# file:
<?xml version="1.0" encoding="utf-8"?>
<exml xmlns:efl="http://www.enlightenment.org/EXML">
<Box efl:orientation="Vertical">
<Button efl:text="Ok" efl:clickedEvt="OnOkClicked"/>
<Button efl:text="Cancel" efl:clickedEvt="OnCancelClicked"/>
</Box>
</exml>
Tag names represents the widgets to be instantiated.The names can be represented
fully qualified or as local names to the Efl.Ui
namespace. In the example
below, both tags refer to the same Widget.
<Button />
<Efl.Ui.Button />
RULE: Tag names must resolve to valid widget names.
If an unkown tag name is encountered, an validation
Error
is generated.
Widgets that inherit from Efl.IPack
are considered containers and can have
others widgets as children, including other containers.
<Box>
<Text />
<Box>
<Button />
<Button />
</Box>
</Box>
RULE: Only container tags can have children tags.
If a non-container widget (e.g.
Button
) has children, an validationError
will be generated.
Attributes are used for both setting properties on the widgets and attaching events to callbacks.
EFL properties and events are namespaced with efl:
. Properties are
camelCased
in contrast with PascalCased
which is the style of C#. In the
example below, the Text
property will received the text "Click Me"
after
being instantiated.
<Button efl:text="Click Me"/>
Events are also camelCased
like properties:
<Button efl:text="Click Me" efl:clickedEvt="ClickButtonHandler" />
The tag above will have the callback ClickButtonHandler
attached to its
ClickedEvt
RULE: properties and event attributes must point to existing members.
If an attribute points to a non-existing member, an
Error
will be generated.
Also for properties, it only makes sense to assign values for writeable properties.
RULE: For properties, the property must be writeable to receive a value.
If an attribute points to a read-only property, a
Warning
will be generated.
For events, the callback to be attached must be a valid C# method name or the compilation would fail.
RULE: For events, the attribute value must be a valid C# method name.
If the callback of an event is not a valid name, a
Error
will be generated.
Here is a list of items to be defined as the format evolves and feedback both from users and code generation is received:
- Custom classes
- Non-ui classes (like
Spotlight.Manager
) - Constructor parameters
It is handled through reflection by the ApiDump
component. It loads
data from the binding efl-mono.dll
to an internal model akin to
the library Eolian-Cxx
used by the binding generator,
with top level collections for:
- Classes
- FunctionPointers
- Enums
- Structs
These items are tagged in the DLL with the Efl.Eo.BindingEntity
C#
attribute, which is also used by the binding itself.
For simpler loading and keeping the model class relationships relatively
simple, type references currently are simple strings and methods will be
added for querying it. For example, implemented interfaces information
is stored a list of TypeRef
(single strings) with the name of the
interfaces.
The data loaded from the DLL can be serialized and deserialized through
C#'s
BinaryFormatter
class. Currently the DLL built with the mono-beta
attribute stands
around 6MB with the serialized metadata around 8MB.
TBD
There are two main kinds of errors: Hard and soft errors.
Hard errors generally will mean a XML syntax errors where XMLReader
would raise a XmlException
exception and can’t continue.
Soft errors mean errors in the xml related to the meaning of the information, like wrong widgets, properties, etc.
- Line number
- So the plugin can report back to the user
- Malformed XML
- Typically raised as
XmlException
- Typically raised as
- Does the tag exist?
- Check if the tag is an instantiatable class
- If it has children, is it an Container?
- Check if the parent tag implements
Efl.IContainer
?
- Check if the parent tag implements
- All attributes refer to valid properties?
- All attributes are listed as properties in the class?
- Is the attribute read only?
- Read only attributes can’t receive values from XML;
- Naive property type checking
- Ints vs strings
- Maybe use C#'s
Convert
class?