Skip to content

Latest commit

 

History

History
247 lines (199 loc) · 8.67 KB

basic-concepts.md

File metadata and controls

247 lines (199 loc) · 8.67 KB

Creating a Component

Each component in jaseci webkit is rendered within a jsc-app. After setting up your code, and assuming your jsc-app component is placed somewhere in your html tree, the next step is to create the markup. We won't be using HTML to create the structure of our webpage, however, in this case, we will use our jsc-app component to generate the markup using JSON.

Creating a component is simple: at the bare minimum we need to create an object with a component, props, and sections properties. Here's an example of how we can render a Navbar component.

	[
		{
			"component": "Navbar",
			"sections": {
				"links": [
					{
						"component": 'NavLink',
						"props": {"label": "Home"}
					}
				]
			},
			"props": {
				"label": "Jaseci",
				"background": "red"
			},
		}
	]

In the code above we asked for a Navbar component with a single link element, we set the label of our navbar to 'Jaseci' and the background to 'red'. We also asked for a link within the navbar and for it to be rendered within the links section.

Names

Names are unique values we attach to a component that will allow us to reference it in the future to get the value of its properties.

Sections

Sections allow us to place components within another component. It also allows us to place components at specific locations within another component. In our example above, with the navbar component, links is a section specific to the navbar component that allows us to add Link components within the navbar. Some components have sections and some do not, so be sure to review the available sections for each component to know when and where you can place components within another.

Events

Events allow us to interact with our markup. Just like we can generate the structure of our webpage using props to control the styling and content of our components we can use events to add logic and functionality to make our site interactive. Let's have a look at how we can add an event to our navbar component.

Let's assume we want to display an alert message when a link is clicked: we can modify our link component to listen to an onClick event we can perform an action when we click on the link component.

Taking the code from our example above, let's modify it, we'll start by adding an events property to our link component, and within that events property, our action. Let's do it.

	[
		{
			"name": "nav",
			"component": "Navbar",
			"sections": {
				"links": [
					{
						"component": 'NavLink',
						"props": { "label": 'Home' }
						"events": {
							"onClick": [
								{
									"fn": "alert",
									"args": ["Jaseci"]
								}
							]
						}
					}
				]
			},
			"props": {
				"label": "Jaseci"
			},
		}
	]

After adding the code above, whenever we click on the Home link we are going see the message "Jaseci" printed in an alert dialog.

Actions

Actions allow us to run arbitrary javascript functions and built-in functions provided by the webkit in response to certain events. You can perform multiple actions, one after the other, or compose actions to perform an action after another completes, or fails.

In the example above we created an alert action in response to an onClick event. Let's take a closer look at what we did.

To create an action we need to create an object with the fn property and the args property.

  • fn - the name of the function we want to call
  • args - the values we pass to the function as arguments

Here what our action looked like.

"onClick": [
	{
		"fn": "alert",
		"args": ["Jaseci"]
	}
]

What if we wanted to update a property instead? We can changing our action function to update along provide its two args: the property we want to update and the value of the property.

Here what our action looks like now.

"onClick": [
	{
		"fn": "update",
		"args": ["nav.label", "Jaseci 2.0"]
	}
]

Assuming this onClick event is still attached to our nav link component, whenever we click on this link the label of the navbar will change from Jaseci to Jaseci 2.0.

Action conditions

Consider the following code.

"onClick": [
	{
		"fn": "update",
		"args": ["nav.label", "Jaseci 2.0"],
		"cond": ['var(nav.label)::#neq::Jaseci 2.0'],
	}
]

What have you noticed? It works the same as before, however, there's now a cond property, this property allow us to prevent the execution of some actions unless a certain condition is satisfied. Each condition is a string. In this example, if we click update and the label is not already Jaseci 2.0 the action will run, if is Jaseci 2.0, it remain the same.

We can provide multiple conditions to an action and the action will only run if all conditions are satisfied.

Chaining Actions

We can run until another is executed. We can do so by providing an action as the value to the onCompleted property of another action. Here's an example using the code above:

"onClick": [
	{
		"fn": "update",
		"args": ["nav.label", "Jaseci 2.0"],
		"cond": ['var(nav.label)::#neq::Jaseci 2.0'],
		"onCompleted": {
			"fn": "alert",
			"args": ["Navbar title updated."]
		}
	}
]

Also consider the following code.

"onClick": [
	{
		"fn": "update",
		"args": ["nav.label", "Jaseci 2.0"],
		"cond": ['var(nav.label)::#neq::Jaseci 2.0']
	},
	{
		"fn": "alert",
		"args": ["Navbar title updated."]
	}
]

In the first code above, we ask for an alert dialog to display after the update action is finished. But what about the second code, doesn't it work the same? In this case, yes, but also no. Taking a closer lock at the update action of the first code, it has a cond, this condition will also prevent the onCompleted action from running if the update function did not run. The second code will give us an alert dialog even if the update function did not run.

Operations

Operations are "custom actions" that we can create that will allow us to reuse a particular sequence of actions across components. Each operation has a name and can optionally accept a number of arguments.

Take a look at the following example:

[
    {
        "component": "Container",
        "name": "container1",
        "operations": {
            "sayHi": {
                "args": [
                    "message"
                ],
                "run": [
                    {
                        "fn": "alert",
                        "args": [
                            "arg(message) cool!"
                        ]
                    }
                ]
            }
        }
    },
    {
        "component": "Button",
        "events": {
            "onClick": [
                {
                    "fn": "runOperation",
                    "operation": "container1.sayHi",
                    "args": [
                        "Hello world!"
                    ]
                }
            ]
        },
        "props": {
            "name": "btn1",
            "label": "Say Hello"
        }
    }
]

In the example above, we defined an operation within the container1 component then we called this operation in the btn1 component. Each operation is called using the callOperation action. The callOperation action requires the operation property to be defined with a valid operation as its value. An operation is referenced using the format [Component Name].[Operation Name] which translates to container1.sayHi in this case.

Operation Args

When defining an operation we can set the args property to a list of strings. This can be used to accept args to your operation. Each arg can be used in the actions within an operation and will be replaced by the values passed in the args of the callOperation action. In the example above, arg(message) is replaced with 'Hello world!' and the message alerted is 'Hello world! cool'.

Property References

Property references allow us to get the value of the property of a component. This allows us to move across components. For example, let's say you have an Inputbox component and you want to alert the value of the input box whenever the user presses the Enter button. How would we do that? Let's explore.

Let's start by taking a look at the following code:

[
		{
			"name": "inputbox1",
			"component": "Inputbox",
			"sections": []
			"events": {
				"onEnter": [
					{
						fn: "alert",
						args: "Hello, var(inputbox1.value)!"
					}
				]
			},
			"props": {
				"placeholder": "Enter your name..."
			},
		}
]

The above code will render an Inputbox component with a placeholder of Enter your name.... If we enter our name in the box and press the Enter button on the keyboard an alert dialog is shown and var(inputbox1.value) is replaced with the value of the input box.