Hi! Let's do some development with CuteFront. It is a pure-javascript and HTML framework. It is similar to the Qt desktop framework. Let's summarize how it works: Widgets are written in javascript. CSS and HTML from bootstrap version 5 are used. A Widget creates it's corresponding HTML code to the DOM with javascript. Widgets emit signals. Widgets have slots for receiving signals from other widgets. Code for each widget resides in a separate .js file. Signals and slots between widgets instances are explicitly connected in the main html file. A state of the widget is cached in the HTML elements (say, the state of a radio button). This is the preferred way to maintain the state. If needed, some part of the state can also be cached to internal member variables (say, this.some_boolean_flag, etc.). When user interacts with a widget (say, with a click or typing something), the widget's internal state is changed (say, contents of a text field or a radio button state). This interaction can result in a signal being emitted. When a widget receives a signal to a slot, this typically results in its internal state (and its HTML) being changed. Now I will give you an example how to define a basic widget class ``HateLike`` in file ``hatelike.js`` and it's accompanying html file ``hatelike.html`` that shows how to test a widget and how to connect signals to slots. ```javascript /*Beginning of file "hatelike.js" This file defines a widget class "HateLike" with a collapsable bootstrap accordeon. Within the accordeon, there is a radio button with two choises: "I like it" and "I hate it". There is also a text field that shows how many times the user has clicked either "I like it" or "I hate it". Each like increments the text field value by one, while each hate decreases it by one. Each time a user clicks any of the radio buttons, a text message signal is also emitted from the widget. This can be connected to other widget instances. The widget also has a slot that receives a text message signal from some other widget. If the incoming signal text includes the word "like" then the text field value is incremented by one, if it includes text "hate", then it is decreased by one. */ import { Widget, Signal, randomID } from '../lib/base/widget.js'; /* Widget is the base class for all widgets. Please, put always the methods described below in the same order: constructor, createSignals, _slot functions, createState, createElement The class, signals and slots should be documented with a string starting with "//DOC" to facilitate autodocumentation */ class HateLike extends Widget { /*//DOC Has two radio buttons with choices "I Like it" and "I hate it". Shows a text field how many times it has been liked or hated */ constructor(id) { // Usually, the constructor should only accept a single parameter (id). /* The id parameter is an html element's id attribute that is used to identify this.element in createElement() method default values for internal state should be set in the createState() method instead */ super(id); // calls createSignals() and sets this.id = id this.createElement(); this.createState(); } createSignals() { /* Define signals sent by this widget. After each signal definition, there should be a comment what the signal carries and what it is about. This method must always be defined, even it the widget didn't use any signals, in which case it is just an empty function. */ this.signals.message = new Signal(); /*//DOC Carries a string. Message that depends which radiobutton has been clicked. */ } /* Next define slots where this widget can receive signals from other widgets. Please use always the "snake-case" syntax for slots and always include "_slot" in the name of the function, for example in this case, instead of "inputTextSlot", the name is "input_text_slot". Each slot should have a comment that describes what kind of datatype and/or what kind a nested json object scheme it expects. You can even implement type and json scheme testing. */ input_text_slot(txt) { /*//DOC Receives a text message from another widget Input txt is a string */ // this.txt = structuredClone(txt) // if we would cache the data coming with the signal, we should make an immutable copy of it if (txt.includes("like")) { this.flike() } else if (txt.includes("hate")) { this.fhate() } } createState() { /* Initializes state variables created by createElement (see below). Creates additional state variables if necessary. */ if (this.element == null) { // this check must always be included return } // this.txt = null // if we would cache in the state the text received in function input_text_slot // state variables created in createElement (see below): this.counter_field.innerHTML='0' this.like.checked = true; // initial value: the "I Like it" radio button is enabled } createElement() { /* Hooks into an existing html element and creates dynamically the html of this particular widget into the DOM. Fields of the DOM elements of the dynamically created widget html are used as member state variables of the widget. Callbacks are also attached to buttons, etc. interactive elements. */ this.element = document.getElementById(this.id) if (this.element == null) { // this check must always be included this.err("could not find element with id", this.id) return } /* Generate some UUIDs. These are required for unique "id" attributes for the html elements. When a new widget is instantiated, each one of them should have their own unique set of html elements identified by a unique "id" attribute. Same goes for html # targets. */ let head=randomID(); let ctarget=randomID(); let name=randomID(); let uuid1=randomID(); let uuid2=randomID(); let uuid3=randomID(); // Create the HTML corresponding to this widget this.element.innerHTML=`
Overall points: