Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/tired-fox/mophidian

Static website generator using only python
https://github.com/tired-fox/mophidian

fontawesome html markdown mkdocs python python3 sass static-site static-site-generator tailwindcss

Last synced: 25 days ago
JSON representation

Static website generator using only python

Awesome Lists containing this project

README

        

# Mophidian - A python based SSG
Use markdown to create websites

## Notes
Mophidian has markdown code highlighting with the markdown.codehilite plugin and pygmentize. In the background mophidian runs a command similar to `pygmentize -S -f html -a .highlight > styles/highlight.css` which build the specified theme into a css file. Use `moph highlight ` to generate a theme or `moph highlight` to get a list of available themes.

The core ideas behind this SSG/framework is the same as all the large Javascript frameworks. So in a way, if you used any type of Javascript framework, then this project should be fairly easy to learn.

The documentation includes guides for:
- Basic phml templating. Refer to the [phml](https://tired-fox.github.io/phml/phml.html) documentation for more in depth guides.
- Python's markdown module and how it is integrated with Mophidian
- How the live reload server works and what to expect
- Markdown flavor guide (Specific to the default markdown plugins in this SSG)

### Planned Features:
* site-map
* live reloading server
* components
* templating
* searching
* presets
* tool to build codehilite themes

## Rules and how things work

### File Structure

Mophidian uses a custom language called [`phml`](https://github.com/Tired-Fox/phml). This language was inspired by javascript frameworks similar to Vue.js. The language support python blocks inside `` tags and inline python for conditions and in replacements. With this phml also has a built in component system. This works how you think it would in other javascript languages. You just need to load the component into the compiler then you can use the component anywhere in your phml code.

A simple phml page may look something like this.

```html


Example


Hello World



{{ variables }}


{% if here %}

Right here


{% else %}

Not Here


{% endif %}

    {% for error in errors %}
  • {{ error }}

  • {% endfor %}

```

Phml is similar in some way. Below is how the same example is written in phml.

```python
from phml import PHML

phml = PHML()

phml.load("sample.phml")
phml.render(variables="go", here=True, errors=["er1", "er2"])

# or with shorthand
phml.load("sample.phml").render(variables="go", here=True, errors=["er1", "er2"])
```

```html


{{ variables }}


Right here


Not Here




  • {{ error }}



```

As seen above both are very similar while phml is simplier to write. Phml also has the benifit of using python itself instead of a python like syntax. Also loops retain context so any nested loop using a value from the outer scope won't break.

If you want to know more about phml check out it's [documentation](https://tired-fox.github.io/phml/phml.html)

Mophidian also has built in markdown support. Markdown plugins can be added and modified through Mophidian's configuration file. On top of the plugins Mophidian allows for use of phml components and the use of frontmatter.

Frontmatter is yaml at the start of the file. The parsed values are passed to the markdown pages layout and can be used in rendering a page.

```markdown
---
title: Sample Markdown Page
tags: ['markdown', 'example']
---

# Example Markdown

hello world
```

This above example will use title for the page title. If it isn't present then it will parse the h1 element, `# Example Markdown`, and finally if that is also missing, the markdown file name is used.

You may have noticed that it was mentioned that markdown files use layout pages. These are pages named `layout.phml`. These layouts are expected to be structured like a phml component and it must contain a `` component. Any file that uses this layout will have it's context placed in place of the `` component.

```html

<>

{{ title }}



>
```

Pages inside of mophidian are similar to layouts where they are phml components. This allows for pages to be placed inside of layouts and layout can be placed inside of other layouts. All pages have the name of `page.phml` with an optional layout name seen as `page@{layout}.phml`. See the [File System](#file_system) section to see how this works.

```html

# Any python code goes here. Indentation is based on the first line with text
message = "Hello World"

<>

Hello World


Home>
>

// Any script logic here

/* Any css styles here */

```

At the root of rendering a page in Mophidian there is a base layout which fills in the structure of an html page. ```html



{{ title or '' }}



```

All nested layouts and the page are put into their appropriate slot and then eventually placed in the `` element in the base layout. To add any other element to the `` tag just use the head tag again in a layout or page. Mophidian will take all elements from these duplicate head tags and put them in the base tag. All duplicate tags are ignored. Special tags like title are replace when they are duplicated.

```html

<>

Example

This is a sample page


>
```

### File System

The file system is structured based on javascript libraries like [Astro.js](https://astro.build/). It looks something like this.

```
project
├ moph.yml
├ public/
│ └ */**/*.*
└ src/
├ components/
│ └ */**/.phml
└ pages/
├ */**/*.md
├ */**/page.phml
└ */**/layout.phml
```

As you can see above there are three primary objects in the root project directory. `moph.yml` is used for the Mophidian configuration. `public/` is where static assets are placed. All files and directories are one-to-one to the root of the server. `src/` is where all pages, components, layouts, and additional static files are placed.

Components are placed into their own subdirectory, `src/components/` and each file is a phml component. All components file names are used for the component name. For example, if you have a file named `Callout.phml` the component can be used with ``. Components in subdirectories have their parent directories appended to the name recursively with each name being seperated with a dot. This can look something like this, `header/Nav.phml`, which will expose a component to use that looks like this, ``.

Page, layout, and markdown files are placed into the `src/pages` directory. There can only be one `layout.phml` and `page.phml` per directory. There can be any number of markdown files per directory, with `README.md` being equivelant to a `page.phml` page. Any layout in the same directory as a `page.phml` or markdown file with be used for that file. If there is a layout missing from the current directory, then the next parent directories layout is used. This is recursive up until the base layout.

Mophidian supports groups. These are directories with the syntax of `(group name)`. This is a psuedo directory and is used to name a layout that is in that directory. The layout inside the root of a group inherits it's name. All other files and directories are treated as if they are located in the next parent directory. With the layout inheriting the groups name, all `page.phml` and markdown files can skip it's default assigned layout and use a specific layout. For example if you have the following file structure.

```
...
└ src/
└ pages/
├ (blog)
│ ├ layout.phml
│ └ page.phml
└ about/
[email protected]
```

The `page.phml` file inside the `about/` directory with use the layout from the `(blog)` group. The file structure is treated like it looks like this.
```
...
└ src/
└ pages/
├ layout.phml < has a layout name of blog
├ page.phml
└ about/
[email protected]
```

This file system logic is heavily inspired by [Svelte Kit](https://kit.svelte.dev/docs/advanced-routing#advanced-layouts-group). More about how this all works is in the Mophidian documentation.

### Features

Mophidian comes with many quality of life features with more on the way. Some of notable features include: helper methods, value shorthands, live reloading, and a CLI tool.

#### Helper Methods
- `blank(value)`: Checks if the value is `None`, `False`, or empty. Currently support lists, sets, tuples, and dicts for checking if empty
- `classnames`: Works the same as Vue.js built in [classlist attribute](https://vuejs.org/guide/essentials/class-and-style.html#binding-html-classes). It allows for conditional classes, and listing classes. Example `classnames('red', {'bold': is_bold}, ['underline', {'left-0': lalign}])`
- `filter_sort(collection, filter, key)`: Both filters and sorts a list without calling multiple methods. Uses pythons built in `filter` and `sorted` methods. First it will filter by the passed filter argument, then it will sort the collection by the passed key argument.

#### Value Shorthands
- `@` at the start of `src` and `href` html attributes. The `@` is replaced with the servers root so you don't have to type it out every time.

#### Live Reloading
Mophidian uses a custom library call `watchserver` which is built to automatically handle live reloading pages. It works by determining if it is serving an html page. If it is then it inserts a script that polls the server on a specific endpoint and checks if the current page should be refreshed. The endpoint either response with a truthy response or a falsy one. The server determines if a page should be reloaded based on if it's path fits a file pattern created by a file watch event. I recommend looking at the [watchserver documentation](https://github.com/Tired-Fox/watchserver) for more information on how this works.

#### CLI Tool

Mophidian comes with a cli tool called `moph`. This tool comes with the ability to: create a new project, build a project, build and serve a project, build and preview a project, and generate pygmentize css.

For more information about Mophidian, check out the [documentation](https://tired-fox.github.io/Mophidian/)