Form events
While the application is running, there is a number of events for each form that occur as a result of certain user actions:
Global
QUERYCLOSE
- the user tries to close the form by clicking on the corresponding button in the form header.QUERYOK
- the user tries to close a modal form by double-clicking on any row in the table.
For objects on the form
CHANGE
- user changed the current value of an object.
For properties or actions on the form
CHANGE
- the user initiated a property change or action call.CHANGEWYS
- the user initiated a WYSIWYG property change using the PASTE operation or a special input mechanismGROUPCHANGE
- the user initiated a property change for all objects in the tableEDIT
- the user initiated editing of an objectCONTEXTMENU
- the user selected the specified item in the context menu of a property (action) on the form
There are also several so-called derivative events that are nothing more than syntactic sugar, but at the same time allow you more effectively to solve typical problems that arise when working with forms:
Global
INIT
- occurs immediately after the form is opened.APPLY
- occurs when the form session is saved to the database (inside the transaction, at the very beginning, before global event handling is performed).APPLY BEFORE
- occurs immediately before the form session is saved to the database (before the start of the transaction).APPLY AFTER
- occurs immediately after the form session is successfully saved to the database (after the end of the transaction).CANCEL
- occurs when the form session changes are canceled.CLOSE
- occurs when theSystem.formClose
action is executed.DROP
- occurs when theSystem.formDrop
action is executed.
If the form is the session owner (meaning that when the System.formOk
action is executed the form session is saved):
OK
,OK BEFORE
,OK AFTER
- occurs when theSystem.formOk
action is executed, at the moments similar to the corresponding moments for theAPPLY
event (i.e. inside, before and after the transaction). In this case,OK
andOK BEFORE
handlers are executed before executingAPPLY
andAPPLY BEFORE
handlers, andOK AFTER
is executed afterAPPLY AFTER
.
If the form is not the session owner:
OK
- occurs when theSystem.formOk
action is executed
If OK BEFORE
and OK AFTER
handlers are defined, but the form is not the session owner, these handlers are still executed, respectively before and after the OK
event handler.
If, after the execution of event handlers with the BEFORE
postfix, the System.applyCanceled
property value equals TRUE
, further execution of the action that led to the occurrence of this event is stopped (for example, with APPLY BEFORE
, saving the session is interrupted as if one of the existing constraints had been violated).
For the remainder of the section, the behavior of properties and actions is exactly the same and so we will use only the term property (behavior is absolutely identical for actions).
The developer can execute certain actions (handlers) when any of the events described above occurs. In the current implementation there can be several handlers for global events and form object events, but only one for form property events. In the first case, the handler is added to the corresponding list, in the second case, the handler replaces the existing one.
For property, it is possible to define its event handlers for the whole logics at once. In that case, these handlers will be automatically added to all forms where these properties are displayed.
You can also specify a keyboard shortcut for each property, pressing which triggers the property's CHANGE
event. If several properties on the form correspond to one key combination, the event will be triggered only for the property whose component is closest to the current active component in the component hierarchy.
Default handlers
For some events, the platform automatically creates default handlers:
-
QUERYCLOSE
Calls the
System.formClose
action -
QUERYOK
Calls the
System.formOk
action -
CHANGE
The user is requested for an object of the changed property value class, after which the received object is written to this property. If the property is not mutable, the user filter mechanism for this property (or for the property specified using the corresponding option) is automatically called.
-
GROUPCHANGE
Calls the
CHANGE
event handler for all objects that meet the filter conditions of the object group in which the changed property is displayed.
If property event handler uses (even implicitly) the value request operator, then default handler can be created in a different way.
For an object request from the user, depending on the type of class, the following operators are used:
- Builtin classes - the input primitive operator.
- Custom classes - the open form operator. The form is the list form for this class.
Standard handlers
For properties and actions on the form, it is also possible to define the following standard change event handlers (CHANGE
, CHANGEWYS
, GROUPCHANGE
, EDIT
):
- Read Only (
READONLY
) - if the property is displayed in the table, the handler will be similar toCHANGE
default handler when the property is not mutable (that is, the user filter mechanism will be called). If the property is not displayed in the table, nothing will happen. You can also make this option conditional (READONLYIF
) (that is, change only if the value of some property is notNULL
). - Selector (
SELECTOR
) - when you try to make a change, a dialog will be shown in which the user will be asked to change the current value of the object.
Language
To define the form event handlers, use the ON
option in the FORM
statement (events block, properties and actions block, objects block), as well as in property options.
Examples
showImpossibleMessage() { MESSAGE 'It\'s impossible'; };
posted = DATA BOOLEAN (Invoice);
FORM invoice 'Invoice' // creating a form for editing an invoice
OBJECTS i = Invoice PANEL // creating an object of the invoice class
// ... setting the rest of the form behavior
EVENTS
// specifying that when the user clicks OK, an action should be executed that will execute
// actions to "conduction" this invoice
ON OK { posted(i) <- TRUE; },
// by clicking the formDrop button, showing a message that this cannot be, since this button
// by default will be shown only in the form for choosing an invoice, and this form is basically
// an invoice edit form
ON DROP showImpossibleMessage()
;
CLASS Shift;
currentShift = DATA Shift();
CLASS Cashier;
currentCashier = DATA Cashier();
CLASS Receipt;
shift = DATA Shift (Receipt);
cashier = DATA Cashier (Receipt);
FORM POS 'POS' // declaring the form for product sale to the customer in the salesroom
OBJECTS r = Receipt PANEL // adding an object that will store the current receipt
// ... declaring the behavior of the form
;
createReceipt () {
NEW r = Receipt {
shift(r) <- currentShift();
cashier(r) <- currentCashier();
SEEK POS.r = r;
}
}
// adding a property through the form extension so that SEEK could be applied
// to the already created object on the form
EXTEND FORM POS
EVENTS
// when opening the form, executing the action to create a new receipt,
// which fills in the shift, cashier and other information
ON INIT createReceipt()
;