The processing #
Introduction #
Now that the interface has been created on the page dedicated to the interface, we can move on to coding the processing part of the application.
The application can be launched directly from this page at each stage of its development, for a better understanding of the purpose of the code introduced at each stage.
Interface displaying #
  Interface description (BODY)
  #
Let’s create a constant with the content that has been elaborated on the Interface page. By convention, with Python (and many other languages), the name of a constant consists only of uppercase letters, numbers and the _ character.
The name of the constant here is BODY, which is defined as follows:
BODY = """ <fieldset> <input placeholder="Name" value="World"> <button>Send</button> <hr> <fieldset> <output>Display area</output> </fieldset> </fieldset> """
The “”" sequence is a delimiter characterizing a string that spans several lines. In Python, you can also use the characters ' or " to delimit a string, but, in this case, the string cannot contain line breaks.
  Interface display function (atk)
  #
Let’s now write the function that will render (display the visual representation of) the contents of the BODY constant we’ve just created.
To do this, we’ll use a module called atlastk, a module being a kind of toolbox offering a full range of functions. The atlastk module is dedicated to the handling of the graphical user interface.
The first step is to declare to the runtime environment that we want to use this module. This is done using the import keyword. This declaration must be made before the first use of one of the module’s functionalities, so it is often placed at or near the beginning.
import atlastkBODY = """ <fieldset> <input placeholder="Name" value="World"> <button>Send</button> <hr> <fieldset> <output>Display area</output> </fieldset> </fieldset> """
Now let’s write the function that will actually perform the rendering. This function must be called atk. Here’s what it looks like:
async def atk(dom):
  await dom.inner("", BODY)
The definition (2nd line) of the atk function is set back from its declaration (1rst line); it is said to be “indented”. Generally, this indentation is not mandatory and is only used to make the code easier to read, but in Python, the indentation is part of the language’s syntax, and its absence will cause errors.
dom is a parameter received by the atk function. It is an object provided by the atlastk module. For technical reasons, most functions belonging to this object must be called with the await keyword. In addition, any function that contains at least one function called with await must be declared with the async keyword.
The use of the async and await keywords is solely due to the technology used to run Python applications in a web browser. They are rarely used in a Python program running in a traditional runtime environment, but are mandatory in this context; omitting them will not usually cause an error message to be displayed, but it will prevent applications from working properly.
The inner method of the dom object (a method is a function attached to an object) is used to replace the contents of a GUI element.
Here, the element is identified by an empty string (“”). This identifier is special and designates the entire interface. The entire GUI is thus made up of the contents of the BODY constant defined above.
As the atk function calls the atlastk module, but also uses the contents of the BODY constant, it will be placed as follows:
import atlastk
BODY = """
<fieldset>
  <input placeholder="Name" value="World">
  <button>Send</button>
  <hr>
  <fieldset>
    <output>Display area</output>
  </fieldset>
</fieldset>
"""
async def atk(dom):
  await dom.inner("", BODY)
Association display function / new connection #
Since it has the name atk, this function will automatically be associated with the event corresponding to a new connection, i.e. it will be called on each new connection. This allows the function, as in the example code, to display the interface of the application.
Now let’s call the function that will handle this association:
atlastk.launch(globals=globals())
atlastk refers to the module of the same name imported earlier, and launch is one of its functions to which we pass, in this case, the result of the globals() function. globals() is a special Python function, generally seldom used, but, in this case, the object it returns will enable the launch() function to launch the atk function and, more generally, to handle user interactions with the interface.
Below you’ll find all the code developed so far:
import atlastk
BODY = """
<fieldset>
  <input placeholder="Name" value="World">
  <button>Send</button>
  <hr>
  <fieldset>
    <output>Display area</output>
  </fieldset>
</fieldset>
"""
async def atk(dom):
  await dom.inner("", BODY)
atlastk.launch(globals=globals())
Clicking on Run will display the interface as elaborated on the Interface page. However, clicking on the Send button will have no effect, as the part of the code that supports this action is missing. It’s this code that we’re now going to develop.
Handling the Send button #
  Association click on Send button / action (xdh:onevent=“Submit”)
  #
Let’s modify the application code to indicate which action to associate with the clicking on the Send button. To do this, we’ll update the HTML code:
BODY = """
<fieldset>
  <input placeholder="Name" value="World">
  <button xdh:onevent="Submit">Send</button>
  <hr>
  <fieldset>
    <output>Display area</output>
  </fieldset>
</fieldset>
"""
With this code (xdh:onevent=“Submit”), we add an attribute to the HTML element button. For each HTML element, as we saw on the Interface page, we can specify a series of predefined attributes. But you can also extend HTML and create your own attributes (and elements too). These extensions to HTML code must be specified within a namespace. Here, the namespace is xdh and the : means that the identifier that follows (onevent) belongs to this namespace.
When parsing this code, the web browser, which is responsible for rendering HTML code, will ignore anything that belongs to a namespace it doesn’t know. It’s the atlastk module which, when parsing the HTML code in turn, detects the presence of components associated with the xdh namespace and will handle them.
Here, the attribute is xdh:onevent and its value is Submit. This means that the default action of the given button (clicking on it) is associated with an event named Submit.
Let’s take a look at the code in full:
import atlastk
BODY = """
<fieldset>
  <input placeholder="Name" value="World">
  <button xdh:onevent="Submit">Send</button>
  <hr>
  <fieldset>
    <output>Display area</output>
  </fieldset>
</fieldset>
"""
async def atk(dom):
  await dom.inner("", BODY)
atlastk.launch(globals=globals())
The application now displays the interface as before, but this time a click on the Send button causes an error message to be displayed. This indicates that a Submit action has been associated with the click on the Send button, but that there is no function associated to this action.
Let’s fix that.
  Function associated with the Send button (atkSubmit)
  #
For didactic purposes, clicking on the button will, at first, simply display a message indicating that the click has been taken into account.
Here’s what the code will look like:
async def atkSubmit(dom):
  await dom.alert('Click on the "Send" button detected!')
We find again the various elements (async, await, parameter dom…) we discovered with the atk function. The alert method of the object corresponding to parameter dom displays a dialog box containing the string passed as parameter.
  Association function atkSubmit / action Submit
  #
The toolkit Atlas will automatically associate an action defined with the xdh:onevent attribute with the function of the same name prefixed with atk. So, because it is associated with the Submit action, the above function is called atkSubmit.
Here is the complete code:
import atlastk
BODY = """
<fieldset>
  <input placeholder="Name" value="World">
  <button xdh:onevent="Submit">Send</button>
  <hr>
  <fieldset>
    <output>Display area</output>
  </fieldset>
</fieldset>
"""
async def atk(dom):
  await dom.inner("", BODY)
async def atkSubmit(dom):
  await dom.alert('Click on the "Send" button detected!')
atlastk.launch(globals=globals())
This time, clicking on the Send button displays the expected message (Click on the “Send” button detected!).
Let’s finalize the atkSubmit function.
Retrieving of user data #
  Input field identification for content retrieval (id=“Input”)
  #
Let’s modify the HTML code to add an identifier to the input field in order to retrieve its content:
BODY = """
<fieldset>
  <input placeholder="Name" value="World" id="Input">
  <button xdh:onevent="Submit">Send</button>
  <hr>
  <fieldset>
    <output>Display area</output>
  </fieldset>
</fieldset>
"""
The id attribute is used to assign an identifier to an HTML element so that it can be referred to, as we’ll see later. This identifier must, of course, be unique among all the elements making up a HTML document.
  Retrieving the content of a HTML element (dom.getValue(...))
  #
Let’s modify the atkSubmit function to retrieve the content of the input field and display it in the dialog box:
async def atkSubmit(dom): input = await dom.getValue("Input") await dom.alert('Click on tne "Send" button detected!') await dom.alert(f'Value of input field: "{input}"')
The getValue(...) method retrieves the value of an HTML element, in this case that of identifier Input corresponding to the input field. The code input = … is used to assign this value to the variable of identifier input.
The character string given as a parameter to the alert(...) method, because it is preceded by the f character, is an f-string and therefore undergoes special treatment. Indeed, any expression between braces ({}) present in an f-string is evaluated. In our case, the string contains the expression {input}. This expression will be replaced by the contents of the input variable when the application is run, i.e., as previously indicated, by the contents of the input field.
Let’s test the code with all the modifications:
import atlastk
BODY = """
<fieldset>
  <input placeholder="Name" value="World" id="Input">
  <button xdh:onevent="Submit">Send</button>
  <hr>
  <fieldset>
    <output>Display area</output>
  </fieldset>
</fieldset>
"""
async def atk(dom):
  await dom.inner("", BODY)
async def atkSubmit(dom):
  input = await dom.getValue("Input");
  await dom.alert(f'Value of input field: "{input}"')
atlastk.launch(globals=globals())
When you click on the Send button, the dialog box displays the value retrieved from the input field (World).
Displaying content in the interface #
  Display zone identification (id=“Output”)
  #
As with the input zone, we’ll assign the element corresponding to the display zone its own identifier (Output):
BODY = """
<fieldset>
  <input placeholder="Name" value="World" id="Input">
  <button xdh:onevent="Submit">Send</button>
  <hr>
  <fieldset>
    <output id="Output">Display area</output>
  </fieldset>
</fieldset>
"""
  Modifying the content of an HTML element (dom.setValue(...))
  #
Let’s alter the atkSubmit function to modify the content of the display zone to show a text containing the data entered by the user, which is stored in the input variable:
async def atkSubmit(dom):
  input = await dom.getValue("Input")
  await dom.alert(f'Value of input field: "{input}"')
  await dom.setValue("Output", f'Hello, {input}!')
The setValue(...) method is used to modify the value of an HTML element, in this case that of identifier Output, with the character string passed as parameter.
Let’s test the code with all the modifications:
import atlastk
BODY = """
<fieldset>
  <input placeholder="Name" value="World" id="Input">
  <button xdh:onevent="Submit">Send</button>
  <hr>
  <fieldset>
    <output id="Output">Display area</output>
  </fieldset>
</fieldset>
"""
async def atk(dom):
  await dom.inner("", BODY)
async def atkSubmit(dom):
  input = await dom.getValue("Input");
  await dom.setValue("Output", f'Hello, {input}!')
atlastk.launch(globals=globals())
Clicking on the Send button displays a greeting message containing the text in the input field. You can perform a few more tests by changing the content of this input field.
Improved ergonomics #
  Focus on an HTML element (dom.focus(...))
  #
You’ll notice that after clicking the Send button, you’ll need to click in the input field to be able to make a new entry. The input field is said to “lose focus”. On a smartphone, pressing the Send button closes the virtual keyboard.
An element is said to have the focus when it is the target of the user’s actions. So, when the input zone has the focus, it’s in this zone that the characters entered on the keyboard appear.
Let’s modify the code so that the input zone regains focus each time the input is validated, as well as when a session is opened. To do this, we modify the atk and atkSubmit functions:
async def atk(dom):
  await dom.inner("", BODY)
  await dom.focus("Input")
async def atkSubmit(dom):
  input = await dom.getValue("Input");
  await dom.setValue("Output", f'Bonjour, {input} !')
  await dom.focus("Input")
The focus(…) method is used to give focus to the element whose identifier is passed as a parameter. Note that only one element (or none at all) can have the focus at any given time.
Let’s test the result:
import atlastk
BODY = """
<fieldset>
  <input placeholder="Name" value="World" id="Input">
  <button xdh:onevent="Submit">Send</button>
  <hr>
  <fieldset>
    <output id="Output">Display area</output>
  </fieldset>
</fieldset>
"""
async def atk(dom):
  await dom.inner("", BODY)
  await dom.focus("Input")
async def atkSubmit(dom):
  input = await dom.getValue("Input");
  await dom.setValue("Output", f'Hello, {input}!')
  await dom.focus("Input")
atlastk.launch(globals=globals())
On launch, the cursor blinks in the input field, indicating that it has focus. On Android or iOs devices, depending on the browser you’re using, you’ll still need to click in the input field to enter text (displaying of the virtual keyboard), but once you’ve clicked on Send, the virtual keyboard will remain open and you’ll be able to enter text without having to click on the input field first.
Bonus #
We’re going to make two more small modifications using some of the elements we’ve already used. The first is in the HTML code, so that you can validate the entered data directly with the Enter key on a physical or virtual keyboard, and the second is in the code for the atkSubmit function, to empty the input field after each validation, clearing the way for a new entry.
These changes are as follows:
BODY = """ <fieldset> <input id="Input" xdh:onevent="Submit" placeholder="Name" value="World"> <button xdh:onevent="Submit">Send</button> <hr> <fieldset> <output>Display area</output> </fieldset> </fieldset> """ … async def atkSubmit(dom): input = await dom.getValue("Input"); await dom.setValue("Output", f'Bonjour, {input} !') await dom.focus("Input") await dom.setValue("Input", "")
Let’s see what happens:
import atlastk
BODY = """
<fieldset>
  <input id="Input" xdh:onevent="Submit" placeholder="Name" value="World">
  <button xdh:onevent="Submit">Send</button>
  <hr>
  <fieldset>
    <output id="Output">Display area</output>
  </fieldset>
</fieldset>
"""
async def atk(dom):
  await dom.inner("", BODY)
  await dom.focus("Input")
async def atkSubmit(dom):
  input = await dom.getValue("Input");
  await dom.setValue("Output", f'Hello, {input}!')
  await dom.focus("Input")
  await dom.setValue("Input", "")
atlastk.launch(globals=globals())
Note that you can now use the Enter key on the physical or virtual keyboard to validate your input, and that the input field empties after each validation.
Going on #
The Inspiration section offers a whole range of applications for discovering new functionalities, which you can launch and modify in situ to explore how they work.
The Create section is dedicated to creating your own applications.