Tutorial

This tutorial is intended to explain the concepts of Abell while building a simple portfolio.

If you are someone who prefers to stare at the code until it explains itself, you can get started with Abell Starters



What is Abell?

Abell is a JavaScript-based Static-Site-Generator to help you build fast and scalable static websites. It is similar to tools like Jekyll, Eleventy, Hugo, and Hexo.

Abell files are like HTML files except you can write JavaScript in double curly brackets which is executed to build you a static page.

Input - index.abell

{{
  const a = 'Hello';
  const b = ', World 🌻';
}}

<html>
  <body>
    I can render JavaScript! Look: {{ a + b.toUpperCase() }}
  </body>
</html>

Output - index.html

<!-- Output - index.html -->
<html>
  <body>
    I can render JavaScript! Look: Hello, WORLD 🌻
  </body>
</html>

Installation and Setup

If you want to play around online, you can skip this installation section and Start on CodeSandbox with an Example Project

Make sure you have Node.js v10+ Installed

Let's start by creating a new Abell project. We're naming it my-cool-portfolio, but you can use any name you'd like.

  1. Run this command in your terminal-
npx create-abell-app my-cool-portfolio --template minimal --installer npm
  1. Change the directory and run the dev server
cd my-cool-portfolio
npm run dev
  1. Tadaaaa 🎉
    Visit http://localhost:5000 in your browser and you should be able to see 'Hello, Abell Example!' on screen.

Setting Up Our Portfolio

We have two important files that we need to care about-

- theme/
  |- index.abell

- abell.config.js

Let's start by changing Abell Example with your name.

module.exports = {
  // ..
  globalMeta: {
    siteTitle: 'Saurabh', // replace with your name
  },
  //...
};
{{
  const { globalMeta } = Abell;
}}

<body>
  <h1>Hello, {{ globalMeta.siteTitle }}</h1>
</body>

This will change the text on our website to Hello, Saurabh! 🥳

All values defined inside globalMeta in ./abell.config.js are accessible from any .abell file with Abell.globalMeta.<key>.

Thus the siteTitle variable defined in the abell.config.js file can be used in .abell files with {{ Abell.globalMeta.siteTitle }}

Adding CSS to Our Project

Fun fact! It's literally same process you'd do with HTML.

  1. Create theme/index.css
  2. Add <link> to our theme/index.abell file.
<head>
  <link rel="stylesheet" href="./index.css" />
</head>

Though for this case it will work perfectly, It is recommended to use Abell.$root to make sure the path always points from root. 3. Add {{ Abell.$root }} to href.

<head>
  <link rel="stylesheet" href="{{ Abell.$root }}/index.css" />
</head>

Abell.$root always points to the theme directory irrespective of which abell file it is built from. This is to avoid confusion of linking files from components or dynamic paths (we will learn these things later in this tutorial)

Require JSON to Abell Files

Next we will add Projects to our Portfolio.

Normally in a Plain HTML Website, we have to edit the whole HTML to add/remove/edit the project.

With Abell, we can create a projects.json and define our projects there and render it in our website.

  1. Create theme/data/projects.json
  2. Require this JSON file in theme/index.abell with require('./data/projects.json')
  3. We can use JavaScript's .map method to loop over our projects
{{
  // ...
  const projects = require('./data/projects.json');
}}

<!-- ... -->
<section class="projects">
{{
  projects.map(project => /* html */ `
    <div class="project">
      <h3><a href="${project.url}">${project.title}</a></h3>
      <p>${project.description}</p>
    </div>
  `)
}}
</section>
[
  {
    "title": "My cool project",
    "description": "Here's my sample project that is built with Abell",
    "url": "https://abelljs.org"
  },
  {
    "title": "My Random Library",
    "description": "I made this NPM library to learn Mewtwo Programming Language",
    "url": "https://github.com/abelljs/abell"
  }
]




Fun fact: We can use require to require NPM packages, Node Modules and everything that can be required from Node.js!

Building Your First Abell Component!



Components allow you to group markup, styles and scripts that can be reused across pages (even across projects).

This is how a syntax of Abell Components looks-

<AbellComponent>
<template>
  <!-- HTML/Abell -->
</template>

<style>
  /* CSS (Scoped for Component By Default) */
</style>

<script>
  // FrontEnd JavaScript
</script>
</AbellComponent>

Let's build a component for our portfolio

This will be a simple Footer with year and your name on it.

  1. Create theme/components/Footer.abell. This will hold our Footer Component.
  2. We will add HTML and CSS for our component.
  3. Require Footer in our index.abell and use the Footer element.

Wohooo we will now see © Saurabh Daware at the bottom of our page.

The styles we wrote in Footer.abell are scoped for the component so they won't be applied to the elements outside our Footer component.

Passing Props to Components

Currently we have our name statically mentioned inside the Footer component. Let's pass it from theme/index.abell instead.

Note that props is the only possible attribute for Components. Any value that you want to pass, has to be inside the props attribute object.

Add JavaScript to Components

Now we will add year to our Footer!

<AbellComponent>
<template>
  <footer>&#169; {{ new Date().getFullYear() }} {{ props.footerText }}.</footer>
</template>

<!-- ... -->
</AbellComponent>

Tadaaaa 🎉 Easy Right? NO!

The JavaScript that we write inside curly bracket is executed on build. Which means every year, we will have to trigger a build to reload the footer year.

Orrr.. we can write client-side JavaScript to always get the latest year.

Adding Client-Side JavaScript-

<AbellComponent>
<template>
  <footer>&#169; <span class="year"></span> {{ props.footerText }}.</footer>
</template>

<!-- css... -->

<script>
  document.querySelector('.year').innerHTML = new Date().getFullYear();
</script>
</AbellComponent>

We can write our Frontend JavaScript inside <script> tag in our Component. This will now be executed on Client-side and it will make sure that it always fetches the latest year irrespective of build.

All problems solved right? Nope...

Here we are using document.querySelector('.year') which will look for element with class="year" in entire website (even outside our component) and return the top-most element.

Though this will work in this case, but we need a way to make sure we don't end up selecting elements outside Component to maintain the scalability.

Thus in Abell Component scripts, it is recommended to use scopedSelector() and scopedSelectorAll() instead of document.querySelector() or document.querySelectorAll()

<AbellComponent>
<template>
  <footer>&#169; <span class="year"></span> {{ props.footerText }}.</footer>
</template>

<!-- css... -->

<script>
  scopedSelector('.year').innerHTML = new Date().getFullYear();
</script>
</AbellComponent>

This maintains the scope of the Component which will help you in maintaining scalability.

Dynamic Page Generation

Next up, we will be adding blogs to our portfolio.

For blogs in our portfolio, we can't go on creating an Abell Page for every blog. So what we do instead is create one Abell Page as a layout to hold our blog content, and pass it in the Markdown content of blog.

We will be writing blogs in Markdown and that content should reflect in our website.

  1. Create ./content/hello-world/index.md and add sample Blog Content
  2. Create ./theme/[path]/index.abell file to hold layout for all our blogs.
  3. Add list of blogs in ./theme/index.abell
{{
  const { importContent, meta } = Abell;
}}

<html>
<body>
  <div class="content">
    {{ importContent(`${meta.$path}/index.md`) }}
  </div>
</body>
</html>
# Hello World!

This text is coming from `content/hello-world/index.md`
{{
  const { contentArray } = Abell;
}}

<html>
<body>
<section>
  <h2>Blogs</h2>
  <div class="blogs">
  {{
    contentArray.map(meta => /* html */ `
      <div>
        <h3>${meta.title}</h3>
        <p>${meta.description}</p>
      </div>
    `)
  }}
  </div>
</section>
</body>
</html>

This makes Abell loop over the content directory to generate pages with names of the folders.

For examples in this case, A dist/hello-world/index.html will be generated with the layout of theme/[path]/index.abell.

You can now go on creating markdown files with format content/<blog-slug>/index.md and the pages will be generated out of the content.

So our directory structure will look like-

- content/
  - hello-world/
    - index.md
  - my-second-blog/
    - index.md

- theme/
  - [path]/
    - index.abell
  - index.abell

You can also access the information of all the content in Abell.contentArray variable from all .abell files. In the above code, we're using this variable data to create a list of blogs on index page.

You can modify the default meta values (title, date, etc) from meta.json file in the blog folder.

  1. Create content/hello-world/meta.json file
  2. Add the following content
{
  "title": "Hello, World!",
  "description": "This is the first blog.",
  "$createdAt": "18 June 2020"
}

You can access these values in Abell.meta variable in theme/[path]/index.abell.

Read more about it in the Abell.meta API Reference

Warning: We do feel there is a lot of abstraction here that makes things confusing. We're looking for new ways to handle dynamic routing in final Abell version. Have an idea? do drop it in the GitHub Discussions of "Issues with Dynamic Routing in Abell"