Приветствую всех! В данной статье хотелось бы рассмотреть такой проект как HMPL.js. "Зачем он нужен и как благодаря нему можно сократить размеры javascript файлов с сохранением дизайна сайта?" - на все эти вопросы я постараюсь ответить.

Так как данный проект является активно мной разрабатываемым, то данным блогом я бы хотел начать для себя активную отчётную деятельность, где аудитория, как в роли заказчика, а я, как в роли исполнителя, буду делать небольшие статьи о проделанной работе. Если вам будет интересна данная идея, то будет круто, если вы оставите комментарий по данной теме! Мне будет очень интересно прочитать!

Также, данные статьи будут выпускаться в видеоформате на моём ютуб канале. Также, дополнительный материал буду писать в тг канале.

Пример использования: CodeSandbox

Понятие HMPL

Прежде всего, слово HMPL - это соединение слов HTML и Cample. Слово HTML означает язык гипертекстовой разметки, а слово Cample означает название фреймворка для создания пользовательских интерфейсов (это мой предыдущий проект, который я разрабатывал несколько лет). Работая над синтаксисом, который используется в фреймворке, мне пришла идея вынести те идеи, которые я там реализовал в отдельный язык, который можно было бы использовать в разных проектах не зависимо от того, на чём они сделаны. Пример Cample синтаксиса:

const eachComponent = each(
  "table-rows",
  ({ importedData }) => importedData.rows,
  `<tr key="{{row.id}}" class="{{stack.class}}">
    <td class='col-md-1'>{{row.id}}</td>
    <td class='col-md-4'><a ::click="{{setSelected()}}" class='lbl'>{{row.label}}</a></td>
    <td class='col-md-1'><a ::click="{{importedData.delete(row.id)}}" class='remove'>удалить строку</a></td>
    <td class='col-md-6'></td>
  </tr>`,
  {
    valueName: "row",
    functionName: "updateTable",
    stackName: "stack",
    import: {
      value: ["rows", "delete"],
      exportId: "mainExport",
    },
    functions: {
      setSelected: [
        (setData, event, eachStack) => () => {
          const { setStack, clearStack } = eachStack;
          clearStack();
          setStack(() => {
            return { class: "selected" };
          });
        },
        "updateTable",
      ],
    },
  }
);

Ту же интерполяцию строк, либо же "синтаксическое добавление" слушателя событий к DOM узлу, а также вообще применить тот код, который делает cample одним из самых быстрых фреймворков на реактивности без VDOM в интернете, но в рамках языка шаблонов.

Первым проектом, который стал прототипом HMPL, был cample-html (да, вот так неожиданно). Там я реализовал тот синтаксис, про который я поначалу продумывал, но была ещё проблема в том, зачем же он будет нужен. Лучшей идеей для этого была связка с серверным HTML, который может быть запрошен по роуту api. То-есть, идея в этом, что HTML, который будет видеть пользователь, будет храниться изначально не на клиенте, а на сервере. Таким образом, мы можем сократить размеры файлов, что я ниже и покажу в статье с примерами. Пример синтаксиса cample-html:

const templateFn = CampleHTML.createTemplate(
  `<template data-cample data-src="/api/test" data-method="get"></template>`
);

// (After the response arrives from the server) { element = template (HTMLTemplateElement type), status = 200 }
const elementObj = templateFn({
  credentials: "same-origin",
  get: (prop, value) => {
    if (prop === "element") {
      console.log(value);
    }
  },
});

Это уже было больше похоже на то, что я хотел сделать, но всё ещё не то, потому что для работы с сервером используется тег template , который лень будет постоянно писать.

Что же такое HMPL?

HMPL - это небольшой язык шаблонов для отображения пользовательского интерфейса от сервера к клиенту. Он основан на запросах, отправляемых на сервер с помощью fetch и преобразуемых в готовый HTML.

С предыдущего проекта я взял кодовую базу, которая была и перенёс уже в тот же проект, но под новым названием, потому что старое было слишком большим. После нескольких версий, я понял, что нужно менять template на что-то более подходящее и удобное в использовании. Поэтому, были введены так называемые объекты запроса, которые представляют следующее:

<div>
  { 
    {
      "src": "http://localhost:8000/api/test",
      "indicators": [
           {
             "trigger": "pending",
             "content": "<div>Loading...</div>"
           },
           {
             "trigger": "rejected",
             "content": "<div>Error</div>"
           }
       ] 
    } 
  }
</div>

Это простые javascript объекты, которые благодаря фигурным скобкам, передаются прямо в HTML. Не нужно теперь писать template каждый раз, теперь достаточно просто передать объект, который описывает настройки запроса и всё. Таким образом, код будет выглядеть так:

import { compile } from "hmpl-js";

const templateFn = compile(
  `{ 
     {
       "src": "http://localhost:8000/api/test",
       "indicators": [
           {
             "trigger": "pending",
             "content": "<div>Loading...</div>"
           },
           {
             "trigger": "rejected",
             "content": "<div>Error</div>"
           }
       ] 
     } 
   }`
);

const wrapper = document.getElementById("wrapper");

const elementObj = templateFn({
  credentials: "same-origin",
  get: (prop, value) => {
    if (prop === "response") {
      if (value) {
        wrapper.appendChild(value.content);
      }
    }
  },
});

И это на самом деле очень удобно, потому что такой синтаксис можно спокойно выносить в отдельные файлы, а само название, которое состоит из 4 слов, позволит писать расширение для данных файлов.

Как HMPL сокращает размер файлов

В данном разделе, я бы хотел на практическом примере показать, как же всё таки язык шаблонов может сократить ваш код на проекте.

Уменьшение размера файла javascript позволит страницам быстрее загружаться на клиенте. Если взять современный SPA, то окажется, что размеры файлов, даже с учетом всех минимизаций, все равно довольно велики. Конечно, после того, как вы загрузите страницу один раз, ориентироваться на ней становится проще, но время первой загрузки может составлять от одной секунды до, скажем, нескольких минут при плохом интернет-соединении. Немногие клиенты захотят ждать так долго.

При использовании большинства фреймворков и библиотек для создания пользовательского интерфейса приходится писать много шаблонного кода. Каждый символ занимает много места в памяти. Давайте рассмотрим Vue.js кликер:

createApp({
  setup() {
    const count = ref(0);
    return {
      count,
    };
  },
  template: `<div>
        <button @click="count++">Click!</button>
        <div>Clicks: {{ count }}</div>
    </div>`,
}).mount("#app");

Супер простой кликер, но даже для него требуется изрядное количество строк кода на js, не говоря уже о тех случаях, когда приложение более или менее большое.

1. Размер исходного javascript файла на Vue
1. Размер исходного javascript файла на Vue

Даже без двух запятых там могло бы быть на несколько байт меньше

Это проблема не только с Vue, но и с другими фреймворками и библиотеками, которые работают аналогичным образом. Но дело не только в этом. Есть огромное количество дополнительных модулей, которые идут к ним, и к ним идет такое же количество дополнительных модулей, и так до "бесконечности".

На самом деле, одно из решений этой проблемы было предложено давно и оно до банальности простое - это подготовить пользовательский интерфейс на сервере и просто загрузить его на клиент. Благодаря этому размер файлов приложения может быть значительно уменьшен. Именно эта идея используется в HMPL. И именно с cample-html я хотел подобное создать.

В примере я также попытаюсь создать кликер, но с использованием hmpl.js:

document.querySelector("#app").appendChild(
  hmpl.compile(
    `<div>
        <button>Click!</button>
        <div>Clicks: {{ src: "/api/clicks", after: "click:button" }}</div>
    </div>`
  )().response
);

Как вы можете видеть, пользовательский интерфейс будет таким же, но размер файла будет немного меньше.

2. Размер исходного javascript файла на HMPL
2. Размер исходного javascript файла на HMPL

Даже если вы уменьшите файлы и уберете все лишние пробелы из шаблонов, возможно, файлы будут на одном уровне или что-то большее, но это всего лишь предположение на небольших примерах. Если брать большие приложения, то очевидно, что при таком подходе js будет намного меньше.

Как видно из примера, функциональность вычисления и сохранения состояния приложения может быть при желании перенесена на сервер (но, лучше такое состояние делать на клиенте). Тут, наверное, больше разговор идёт про данные в базе данных, которые как раз лучше идут через hmpl.

Как видно из примера, функциональность вычисления и сохранения состояния приложения может быть при желании перенесена на сервер. Понятно, что при огромном количестве пользователей это просто выведет сервер из строя, но важен тот факт, что пользовательский интерфейс одинаков.

Да, конечно, у этого метода есть не только такой недостаток, но и возможность повторного использования пользовательского интерфейса, как кэширование пользовательского интерфейса, чтобы не загружать все по сто раз, и многое другое. Важна альтернатива, которая при правильной настройке может составить конкуренцию большинству современных решений.

Документация HMPL

Язык шаблонов имеет документацию, которая располагается по адресу https://hmpl-lang.github.io/#/docs. Там можно посмотреть не только те примеры, которые были указаны в статье, но и другие примеры, включая примеры проектов. Также, там рассказано и про свойства объекта запроса, и про установку, и про многое другое.

Интеграция с Webpack

Модуль имеет свой собственный загрузчик для файлов с расширением .hmpl. Вы можете загрузить hmpl-loader и использовать синтаксис языка шаблонов в отдельных файлах:

main.hmpl

<div>
  {
    {
      "src": "/api/test"
    }
  }
</div>

main.js

const templateFn = require("./main.hmpl");

const elementObj = templateFn();

Загрузчик работает с версии 0.0.2 и выше.

Мотивация создания проекта

На самом деле, цикл статей и видеороликов, который будет записываться и писаться, для меня очень важен. За последние 4 года, я старался делать разные проекты, вспомнить ту же 2D игру на canvase, либо же кликер, где вышло больше 10 видеороликов и на самом деле, они были действительно интересны, но в них не хватало той production идеи, которая должна быть в продукте. То-есть, проекты доходили до какого-то более ли менее качественного состояния, а потом просто забрасывались, потому что этим буквально никто не будет пользоваться и в это играть. И, на самом деле это безыдейность меня сильно тревожила и я хотел действительно сделать что-то полезное, что можно использовать в реально больших проектах. Чтобы можно было это подключить и реально взаимодействовать, чекать документацию и просто круто с этим работать!

После того, как я на два года ушёл из ютуба, я изучал много материала по программированию и сделал тот проект, к которому я наверное и стремился это cample.js.

Это фреймворк, который позволяет создавать пользовательские интерфейсы и действительно данный проект был очень интересен для меня. Я на протяжении двух лет каждый день работал и думал о том, как бы его улучшить, сделать новый функционал и добиться той цели, которую я себе поставил. Мне хотелось, чтобы люди могли пользоваться качественным продуктом, но, я также понимал, что сегодняшние фреймворки - это такое количество функционала, что в одиночку сделать его было просто невозможно, поэтому, после определённого момента, главной моей целью была скорость. И, в целом, когда фреймворк стал действительно одним из самых быстрых в интернете, по реализации реактивности без виртуального DOM, я, на самом деле, был счастлив, но в тоже время и опустошён, потому что дальше, смысла реализовывать было всё меньше и меньше, поэтому и этот проект, как и кликер и игра, был в какой-то степени несерьёзным.

Когда ты делаешь проект один, реализуешь крутой там функционал, что-то допиливаешь, ты каждый день сидишь, но на 100% не понимаешь, что вот это нужно, а вот это идея фигня и её реализовывать не стоит, и, вот это понимание оно идёт только от тех, кто пользуется продуктом. И, на самом деле, я понял, что сидеть каждый день в каком-то вакууме, где есть только код и всё - это неправильно, поэтому, для себя, я понял, что надо что-то менять. И, данными видеороликами и статьями, как некими отчётами о проделанном, я постараюсь держать тот уровень ответственности и полнейшей серьёзности, которая должна быть, как если бы я был штатным специалистом, а аудитория неким заказчиком. И, на самом деле, допустим те же новые версии, проекты на языке шаблонов, какие-то созданные смежные модули - всё это действительно должно иметь тот уровень качества, который должен быть. Поэтому, для меня это будет тот проект, который должен быть максимально качественным и полным и он будет крутым, чтобы его использовали люди по всему миру, потому что будущее за серверным HTML и Server-side рендерингом! Проект реально позволит быстрее грузить сайты, потому что файлы будут меньше и это классно!

Видеоверсия статьи

Всем большое спасибо за прочтение статьи!

Социальные сети:

https://habr.com/ru/users/antonmak1

https://github.com/antonmak1

https://dev.to/antonmak1

https://t.me/hmpljsru

Источники:

https://github.com/hmpl-lang/hmpl

https://hmpl-lang.github.io

https://github.com/cample-html/cample-html

https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API

https://developer.mozilla.org/en-US/docs/Glossary/SPA

https://github.com/hmpl-lang/examples

https://hmpl-lang.github.io/blog/blog/2024/08/09/how-to-reduce-javascript-file-size-on-client.html

Комментарии (11)


  1. ressiwage
    16.09.2024 12:52
    +1

    а что у него по быстродействию? не хотелось бы писать сайты, которые бы по 1000 раз перестраивали дом при загрузке


    1. antonmak1 Автор
      16.09.2024 12:52

      HMPL пока не поддерживает кеширование UI (в будущих версиях это 100% будет). То-есть, когда вы будете кликать по кнопке, тот элемент, где располагается объект запроса, будет перерисовываться, даже если HTML один и тот же. То-есть, если с сервера придёт <div></div>два раза, то это будет два разных узла. Родительский элемент не будет перерисовываться заново. По скорости отрисовки - почти моментальная. Там из алгоритма просто цикл по элементам и добавление их в DOM. Это немного другое, если мы берём алгоритм, который используется в циклах по элементам, связанным данными (когда рендерится список и указываются ему ключи), там алгоритм в 100 раз сложнее, но он к языку шаблонов не относится. Его пока не нужно делать, но даже если понадобиться, то можете быть уверены, что это будет одна из самых быстрых реализаций!


    1. antonmak1 Автор
      16.09.2024 12:52

      https://github.com/hmpl-lang/hmpl/issues/1 - добавил в issue проблему кеширования


  1. JerryI
    16.09.2024 12:52

    Интересная вещь для образовательных целей.


    Однако серьезно тянуть это в одиночку, шлифовать, документировать для остальных юзеров при условии конкурентов с большим коммьюнити и инфраструктурой довольно бесполезно будет (100% dead end). Если это решает ваши задачи, хорошая строчка в резюме и на странице пет-проектов - этого уже достаточно. Новых скиллов при доведении этого до идеала вы не сможете получить.


    1. antonmak1 Автор
      16.09.2024 12:52

      Спасибо за отзыв! На мои задачи уже всё равно, строчка в резюме мне ничего не даст, учитывая опыт работы. На скиллы и прочее тоже уже всё равно. Я не рассматриваю уже данный проект как "для образовательных целей", потому что и так много таких проектов сделал. На ютубе их воз и маленькая тележка.

      Однако серьезно тянуть это в одиночку, шлифовать, документировать для остальных юзеров при условии конкурентов с большим коммьюнити и инфраструктурой довольно бесполезно будет

      Да пофиг на конкурентов. Сделаем крутой язык шаблонов и будет красиво. Ща, ишьюзы создам и, если хотите, то можете взять и помочь, проблем в этом вроде как нету. Если что, сам сделаю функционал. Это не фреймворк, где миллион функционала (буквально), так что это вполне возможно.


      1. JerryI
        16.09.2024 12:52
        +1

        Да пофиг на конкурентов. Сделаем крутой язык шаблонов и будет красиво. 

        Если хотите, чтобы другие тоже думали, что он крутой и красивые, тогда надо на заглавную страницу вставлять самые крутейшие метрики. +40% speed, потом про красоту: сразу же вставлять в сравнение React и прочие side by side. Сжать все слова в 4-5 буллетов и написать почему он красив, чем он отличается, почему эта ветка развития стоит того.

        Потом, я пытался найти примеры. Они только на гите. Это не годится по современным стандартам. 90% поленятся клонировать. Надо прям в веб-формате.

        Т.е. все про презентацию, это должен быть фейрверк, разрабы разбалованные сегодня. Иначе никто не узнает про него, а если и узнают, подумаю, что yet another pet project


        1. JerryI
          16.09.2024 12:52
          +1

          И часто вижу сравнения вида "По скорости отрисовки - почти моментальная." и прочее. Не, так не годится. Надо цифры. Бенчмарки и прочее, либо примеры с 100000 tr элементов в таблице.


          1. antonmak1 Автор
            16.09.2024 12:52

            Да, вы правы. Надо поработать над лендингом. Кнопка Demo Sandbox вроде есть на главной для примера, но, этого конечно мало + бенчмарков таких я не делал, но можно конечно сделать, но это всё таки не фреймворк, ну, тоже можно энивей. Side by side делаю тут - https://hmpl-lang.github.io/blog/blog/2024/08/09/how-to-reduce-javascript-file-size-on-client.html, но это инфа внутри блога находится, так что не факт, что это кто-то увидит из потенциальных юзеров


            1. JerryI
              16.09.2024 12:52
              +1

              Тогда отлично. Несите все на главную!


  1. Onni
    16.09.2024 12:52
    +1

    Идейно похоже на HTMX, но со своим видением. Здорово


    1. antonmak1 Автор
      16.09.2024 12:52

      Спасибо! Пошла жара). Если что, разбирал вот тут https://hmpl-lang.github.io/blog/blog/2024/08/10/differences-between-hmpl-and-htmx.html