javascript

Два frontend фреймворка. Два подхода

  • пятница, 3 ноября 2023 г. в 00:00:19
https://habr.com/ru/articles/771586/

# Содержание:

1. [Эволюция Веб приложений в двух словах](#1)  

2. [Два подхода](#2)  

3. [Разработка простой компоненты](#3)  

4. [Разработка компоненты посложнее](#4)  

5. [Повторное использование](#5)  

6. [Данные и методы](#6)  

7. [CSS и стилевое оформление](#7)  

8. [Обработка событий](#8)  

9. [Работа с формами](#9)  

10. [Отладка](#10)  

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

<div id="1">  

### 1. Эволюция Веб приложений

#### 1.1. HTML

Сначала нам было достаточно статических страниц, написанных с помощью языка разметки html.

```html

<div id="app">

    <h1>html - это просто.</h1>

</div>

```

#### 1.2. HTML + JAVASCRIPT

После, к статичным страницам html добавили javascript.

```html

<div id="app">

    <h1>Заголовок</h1>

</div>

<script>

    var h1 = document.querySelector('#app > h1');

    h1.innerText = 'html + js - это гибко.'

</script>

```

#### 1.3. JAVASCRIPT FRAMEWORK

Потом, объединили html и javascript и появились frontend framework. Вот, на примере одного из популярных фреймворков Vue.js.

```html

<div id="app">

    <h1>Фреймворк {{framework}} один из лучших.</h1>

</div>

<script src="https://unpkg.com/vue"></script>

<script>

    const app = new Vue({

        el: '#app',

        data: {

            framework: 'Vue.js'

        }

    })

</script>

```

Удобство современных фреймворков - это реактивность, т.е. связанность данных, когда данные на странице

```html

<h1>Фреймворк {{framework}} один из лучших.</h1>

```

связаны с данными компонента

```javascript

app.framework = 'Vue js';

```

<div id="2">  

### 2. Два подхода

Несмотря на развитие технологий, все равно, на первом месте идет язык разметки - html, а javascript так и остался на подхвате, ведь так изначально применен подход в веб сайтам. Такой подход несомненно очень гибкий, и позволяет создавать немыслимые веб интерфейсы как по красоте, так и по функциональности, но для этого frontend framework предлагает свой, так называемый, декларативный язык. Представляете, вы пишите программу на html, пусть доработанном, но, все же, языке разметки.  

А что, если использовать другой подход? А что если на первое место поставить javascript? Конечно, мы потеряем все то многообразие вариантов красочного и анимированного оформления веб-форм, но приобретем лаконичность и привычную атмосферу разработки десктопного приложения. Этот подход позволит создавать веб приложения на основе готовых библиотек элементов управления. А почему бы и нет, ведь десктопные приложения обходятся же ограниченным набором компонент (это не означает небольшим). И созданные приложения будут не менее красивыми и функциональными, ведь можно использовать множество тем оформления.  

Попытаемся сравнить два подхода.  

Первый подход будет представлять один из популярнейших фреймворков [vue.js](https://ru.vuejs.org/).  

Второй подход будет представлять один новый фреймворк [ui-organizer.ru](https://ui-organizer.ru). Собственно, это прототип библиотеки, но вполне рабочий прототип.

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

<div id="3">  

### 3. Разработка простой компоненты

<div id="3.1">  

#### 3.1. Простейший компонент Vue.js

В Vue.js компоненты необходимо разрабатывать самому, либо использовать сторонние библиотеки. Для тех, кто знает html и javascript, очень просто создать нужный компонент.  

index.html

```html

<div id="app">

    <h1>Фреймворк {{framework}} один из лучших.</h1>

</div>

```  

index.js

```javascript

const App = {

  data() {

    return {

      framework: 'Vue.js'

    }

  }

}

Vue.createApp(App).mount('#app');

```

Как видно из примера, программный код написан на html в файле index.html и на javascript в файле index.js. Здесь опускаются вопросы компиляции и сборки приложения, так как в Vue.js процесс сборки отличается в зависимости используете ли вы typescript или javascript. В примере программный код в index.js написан на javascript, поэтому его не надо компилировать и, вообще, вы можете поместить его прямо на страницу index.html вместе с программным кодом, написанном на html. Да, программный код пишем прямо на html. Это будет заметно на более сложных примерах.  

<div id="3.2">  

#### 3.2 Простейший компонент ui-organizer

ui-organizer не предназначен для создания компонентов. ui-organizer - это библиотека готовых элементов управления (компонентов), с помощью которых вы можете сделать нечто подобное.  

main.ts  

```typescript

var form: IForm = <IForm>{

    type:'IForm',

    name:'form',

    flex: Flex.fixed,

    grouping: Grouping.vertical

    elements: [

        <IDataElement>{

            type: 'IDataElement',

            name: 'dataElement',

            bindingProperty: 'el',

            onAfterSetData: async function(form: IForm, elem: IDataElement, data: any){

                elem.value = `Библиотека ${data} - другой фреймворк.`;

                return true;

            }

        }

    ]

}

AppManager.add([form]);

AppManager.open('form', {"el":"ui-organizer"}, undefined);

```

Как видите здесь html вообще не трогаем. И нам пришлось создать форму и добавить на нее элемент IDataElement. Html будет построен автоматически. Программа построит следующий html:  

```html

<div>Библиотека ui-organizer - другой фреймворк.</div>

```

Обращаем внимание на то, что программный код написан на ```typescript```. Это требует компиляции программы, чтобы получить чистый ```javascript```. Реализовать второй подход, наверное, невозможно без использования ```typescript``` - это снижает риски ошибок, ведь нам важно не написать очень хитрый код, с использованием всех возможностей ```javascript```, а создать надежное, простое в сопровождении приложение.  

<div id="4">  

### 4. Разработка компоненты посложнее

<div id="4.1">  

#### 4.1. Компонента "Список задач" Vue.js

Сделаем список задач с кнопкой.  

Сначала создаем html. Как вы можете заметить, здесь уже язык разметки превратился в полноценную программу с циклами, событиями и др.  

```html

<div id="list-rendering">

  <ol>

    <li v-for="todo in todos">

      {{ todo.text }}

    </li>

  </ol>

  <button v-on:click="reverseMessage" @click="setFlag">Перевернуть сообщение</button>

</div>

```  

В файле ```typescript``` добавляем данные и реализуем обработчики событий ```reverseMessage``` и ```setFlag```. В этом примере мы уже используем typescript.

```typescript

interface ITodo {

    text:string

}

const ListRendering = defineComponent({

  data() {

    return {

      todos: [

        { text: 'Learn JavaScript' },

        { text: 'Learn Vue' },

        { text: 'Build something awesome' }

      ] as Array<ITodo>,

      isReversed: false

    }

  },

  methods: {

    reverseMessage(): void {

      this.todos.forEach(item =>{

          item.split('')

            .reverse()

            .join('')

      })

    },

    setFlag(event):void {

      this.isReversed = !this.isReversed;

    }

  }

})

Vue.createApp(ListRendering).mount('#list-rendering')

```

Безусловно, данные тесно связаны с компонентом и с ними достаточно удобно работать. Еще плюс в том, что это очень гибкий инструмент и мы можем создать любой компонент (элемент управления на форме пользователя). По сути мы только что создали элемент управления, который можем использовать отдельно, а можем встроить в любой другой компонент.  

<div id="4.2">  

#### 4.2 Форма со "Списком задач" ui-organizer

Отличие второго подхода (ui-organizer) в том, что мы не создаем новый элемент управления, а используем существующие, чтобы решить задачу. В данном случае используем два элемента управления IList и IButton, причем элемент управления IList мы доработали добавив свойство ```isReversed``` и метод ```reverseText```. У элемента IButton добавили реализацию метода ```onClick``` .  

На самом деле IList и IButton это интерфейсы, а мы создаем объекты, реализующие эти интерфейсы.  В тексте упростили и написали просто: "элементы управления IList и IButton".  

AppManager открывает форму 'simpleForm' и передает ей массив элементов в формате JSON.

```typescript

interface IMyList extends IList {

    isReversed: boolean;

    reverseText: ()=>void;

}

var form: IForm = <IForm>{

    type: 'IForm',

    name: 'simpleForm',

    flex: Flex.flexible,

    grouping: Grouping.vertical,

    elements: [

        <IMyList>{

            type: 'IList',

            name: 'list',

            isReversed: false, // Добавленное свойство

            itemsElement: <IStr>{

                type: 'IStr',

                name: 'str',

                bindingProperty: 'elem',

            },

            reverseText():void {// Добавленный метод

                this.items.forEach(item => {

                    var str = <string>item.element.value;

                    item.element.value = str.split('')

                        .reverse()

                        .join('')  

                });

                this.isReversed = !this.isReversed;

            }

        },

        <IButton>{

            type: 'IButton',

            name: 'reverse',

            caption: 'Перевернуть текст',

            onClick: async function (form) {

                (<IMyList>form.getElement('list')).reverseText();

                return true;

            }

        }

    ]

}

AppManager.add([form]);

AppManager.open('simpleForm', [

    { elem: "второй подход" },

    { elem: "новый фреймворк" },

    { elem: "еще один компонент" },

], undefined);

```

<div id="5">  

### 5. Компоненты и повторное использование

<div id="5.1">  

#### 5.1. Повторное использование в Vue.js

В Vue компоненты - это части приложения, которые можно использовать повторно. Вот например:  

```javascript

Vue.component('todo-item', {

  props: ['todo'],

  template: '<li>{{ todo.text }}</li>'

})

```

На Html странице укажите, где хотите вставить компонент:  

```html

<div id="app">

  <ol>

    <todo-item

      v-for="item in groceryList"

      v-bind:todo="item"

      v-bind:key="item.id"

    ></todo-item>

  </ol>

</div>

```  

Здесь мы указали, что для каждого item в groceryList выводим на экран элемент todo-item.  

В приложении передаем в данные этот самый groceryList.

```javascript

var app = new Vue({

  el: '#app',

  data: {

    groceryList: [

      { id: 0, text: 'Овощи' },

      { id: 1, text: 'Сыр' },

      { id: 2, text: 'Что там ещё люди едят?' }

    ]

  }

})

```

<div id="5.2">  

#### 5.2. Повторное использование в ui-organizer

Здесь повторное использование - это класс элемента, мы можем создать сколько угодно объектов этого класса. Перепишем предыдущий пример из раздела [4.2](#4.2).  

```typescript

class MyList extends UIList {

    constructor(_name:string, _bindingProperty: string){

      this.name = _name;

      this.isReversed = false;

      this.itemsElement = <IStr>{

        type: 'IStr',

        name: 'str',

        bindingProperty: _bindingProperty

      };

    }

    reverseText():void {// Добавленный метод

        this.items.forEach(item => {

            var str = <string>item.element.value;

            item.element.value = str.split('')

                .reverse()

                .join('')  

        });

        this.isReversed = !this.isReversed;

    }

}

var form: IForm = <IForm>{

    type: 'IForm',

    name: 'simpleForm',

    flex: Flex.flexible,

    grouping: Grouping.vertical,

    elements: [

        <IList>(new MyList('list', 'elem')),

        <IButton>{

            type: 'IButton',

            name: 'reverse',

            caption: 'Перевернуть текст',

            onClick: async function (form) {

                (<MyList>form.getElement('list')).reverseText();

                return true;

            }

        }

    ]

}

AppManager.add([form]);

AppManager.open('simpleForm', [

    { elem: "второй подход" },

    { elem: "новый фреймворк" },

    { elem: "еще один компонент" },

], undefined);

```

Здесь мы создали класс MyList и наследовали его от предопределенного класса UIList, который, в свою очередь, реализует интерфейс IList. При описании формы мы просто создали экземпляр MyList, где в конструктор передали название элемента 'list' и название свойства 'elem', из которого будут подставляться данные.  

Как вы могли заметить, в библиотеке ui-organizer не выделено понятие "компонент", так как, в объекте уже инкапсулированы html, данные и поведение в виде методов.

<div id="6">  

### 6. Данные и методы

<div id="6.1">  

#### 6.1. Данные и методы компонента Vue.js

В экземпляре Vue доступны данные и некоторые предопределенные методы:  

```typescript

var user = { name: 'Иван' };

var vm = new Vue({

  el: '#app',

  data: user

})

vm.name === user.name; // => true

vm.name = 'Дмитрий';

user.name // => Дмитрий

vm.$data === user // => true

vm.$el === document.getElementById('app') // => true

vm.$watch('name', function (newValue, oldValue) {

  // Этот коллбэк будет вызван, когда изменится `vm.name`

})

```

Здесь компоненту ```vm```  мы передали объект user. И теперь имеем доступ к свойствам этого объекта через компонент ```vm.name```. Это, так называемая система, реактивности, - наверно, ключевое преимущество современных фреймворков.  

В примере также показано использование служебных свойств $data и $el и служебного метода $watch. Конечно, служебных свойств и методов гораздо больше.  

<div id="6.2">  

#### 6.2. Данные и методы элемента управления ui-organizer

Здесь основной подход следующий: элементу управления передаются данные, которые каким-то образом могут измениться. Чтобы получить измененные данные обратно, нужно запросить эти данные. Например, при открытии формы, ей передаются первоначальные данные. Пользователь может изменить эти данные. При нажатии кнопки submit, мы можем запросить измененные данные формы и отправить их на сервер.  

```typescript

var user = {

  firstName: 'Иван'

}

var form: IForm = <IForm>{

    type:'IForm',

    name:'form',

    grouping: Grouping.vertical

    elements: [

        <IDataElement>{

            type: 'IDataElement',

            name: 'dataElement',

            bindingProperty: 'firstName',

        }

    ]

}

AppManager.add([form]);

AppManager.open('form', user, undefined);

AppManager.activeForm.getData({}).firstName; //=>'Иван'

var dataElem = AppManager.activeForm.getElement('dataElement');

/*№1*/AppManager.activeForm.setData({firstName :'Дмитрий'}); //=>'Дмитрий'

/*№2*/dataElem.getData({}).firstName == 'Дмитрий'; //=>true

/*№3*/dataElem.setData({firstName:'Михаил'});

/*№4*/AppManager.activeForm.collectData({}).firstName; //=>'Михаил'

AppManager.activeForm.dom; //=> HTMLElement

```

В библиотеке ui-organizer для элемента управления можем установить bindingProperty. В нашем примере bindingProperty элемента управления dataElement установлено в firstName. Это значит, что значение свойства firstName в передаваемых данных будет установлено этому элементу (см. №№1-2). Верно, также обратное, когда мы меняем данные элемента функция collectData формы возвращает новые данные (см.№№3-4).  

Здесь первоначальный объект остается неизменным. Для того, чтобы получить изменения в первоначальном объекте, необходимо функции getData передать этот первоначальный объект:  

```typescript

AppManager.activeForm.getData(user);

user.firstName; //=>'Михаил'

```

Служебное свойство dom возвращает HTMLElement, связанный с этим элементом управления. У каждого элемента управления на форме свой DOMHTMLElement. У каждого элемента управления доступны служебные методы, такие как addClass, getElement и другие.

<div id="7">  

### 7. CSS и стилевое оформление

<div id="7.1">  

#### 7.1. Работа с классами и стилями Vue.js

Для стилевого оформления можем добавлять классы css. Конечно описание этих css классов нужно настроить в соответствующем css файле, который подключен к html.  

Описываем html:  

```html

<div

  class="element"

  v-bind:class="{ readonly: isReadonly, error: hasError }"

></div>

```

```typescript

data: {

  isReadonly: true,

  hasError: false

}

```

В описании компонента задаем данные, от изменения которых будет зависеть добавление классов к HTML элементу. Например, так как isReadonly = true, то класс readonly будет добавлен, а так как hasError = false, то класс error добавлен не будет. Таким образом, имеем:  

```html

<div class="element readonly"></div>

```  

 

То же самое с inline стилями:  

```html

<div v-bind:style="{ position: userPosition, font-size: fontSize + 'em' }"></div>

```

```typescript

data: {

  userPosition: 'relative',

  fontSize: 1.5

}

```

Здесь значение стилей pozition и font-size определяются соответственно переменными userPosition и fontSize, описанными в секции data.  

<div id="7.2">  

#### 7.2. сcs и стилевое оформление ui-organizer

Библиотека включает в поставку два файла: page.css, который определяет компоновку элементов управления и style.css, который определяет стилевое оформление. Вы можете заменить style.css. Кроме этого, можно для элемента IGroup указать файл .css, который будет применен для этой группы элементов. Кроме того каждый элемент имеет два метода addClass и removeClass, которые добавляют и удаляют классы HTML элемента.

```typescript

var form: IForm = <IForm>{

    type:'IForm',

    name:'form',

    grouping: Grouping.vertical,

    stylesheets: 'custom.css',

    elements: [

        <IDataElement>{

            type: 'IDataElement',

            name: 'dataElement',

            bindingProperty: 'firstName',

            isSomeClass: false;

            onClick: async function(){

                this.isSomeClass = !this.isSomeClass;

                if (this.isSomeClass) elem.addClass("someClass");

                else elem.removeClass("someClass");

            }

        }

    ]

}

AppManager.add([form]);

AppManager.open('form', undefined, undefined);

```

<div id="8">  

### 8. Обработка событий

<div id="8.1">  

#### 8.1. Обработка событий vue js

```html

<div id="app">

  <button v-on:click="warn('Что ты наделал???')">Нажми!</button>

</div>

```

```typescript

new Vue({

  el: '#app',

  methods: {

    warn: function (message) {

      alert(message)

    }

  }

})

```

В vue js обработчики событий указываются прямо в html. И есть возможность подписываться на все события, возникающие в HTML элементе. В примере мы подписались на событие click html элемента button.  

Вы также можете генерировать свое событие:  

```typescript

this.$emit('myevent');

```

и подписываться на него  

```html

<my-component v-on:myevent="doSomething"></my-component>

```  

<div id="8.2">  

#### 8.2. Обработка событий ui-organizer

Подход в ui-organizer другой: вы не имеете доступа к событиям html элементов, но можете обрабатывать предопределенные для каждого элемента управления события и генерировать свои.  

Например, ниже написан обработчик события onAfterLoad, в котором задаются обработчики событий show и hide элемента управления IElement:  

```typescript

export var baseElem: IElement = <IElement>{

    type: 'IElement',

    name: 'baseElem',

    caption: 'Базовый элемент',

    onAfterLoad: async function (form: IForm, elem: IElement, data: any) {

        elem.on("show", () => {

            elem.addClass('visible');

        });

        elem.on("hide", () => {

            elem.removeClass('visible');

        })

        return true;

    }

}

```

Можете генерировать свои события и подписываться на них:  

```typescript

export interface ISomeElement<T=ISomeElementEvents> extends IElement<T>  {

    someProperty: string;

    setSomeProperty(val: string): void;

}

export interface ISomeElementEvents {

    someSetted(form: IForm, element: IElement): void,

}

export var elem: ISomeElement = <ISomeElement>{

    type: 'IElement',

    name: 'elem',

    someProperty: undefined,

    setSomeProperty: function (val: string) {

        someProperty = val;

        this.emit("someSetted", this.form, this);

    }

}

```

В последнем примере определены два интерфейса для облегчения написания программы на typescript.

<div id="9">  

### 9. Работа с формами

<div id="9.1">  

#### 9.1. Работа с формами vue js

Для связи компонента vue js c элементом формы input имользуется директива v-model, которая указывается в html. Эта директива связывает данные компонента vue js с элементом ввода input.

```html

<div id="app">

    <input v-model="userText" placeholder="текст">

    <p>Введённое сообщение: {{ userText }}</p>

</div>

```

```typescript

new Vue({

  el: '#app',

  data: {

    userText: 'Текст'

  }

})

```

<div id="9.2">  

#### 9.2. Работа с формами ui-organizer

Здесь для пользователя одинаково, что работать c простым элеметом IDataElement, со списком IList или полем ввода IProperty. Все является элементом формы. Программист не работает напрямую с элементом input или textArea, а использует элементы управления. Которым передаются данные как было показано в разделе [6.2](#6.2).  

Форма с полем ввода и текстом (эхо, дублирующее поле ввода) будет выглядеть следующим образом:  

```typescript

var form: IForm = <IForm>{

    type:'IForm',

    name:'form',

    grouping: Grouping.vertical

    elements: [

        <IProperty>{

            type: 'IProperty',

            name: 'text',

            bindingProperty: 'userText',

            placeHoleder: 'Введите текст'

        },

        <IStr>{

                type: 'IStr',

                name: 'str',

                bindingProperty: 'elem',

        }

    ],

    onAfterLoad: async function(){

      let input: IProperty = this.getElement('text') as IProperty;

      let srt: IStr = this.getElement('str') as IStr;

      input.on('change', (form, elem, val)=>{

        str.value = val;

      })

    }

}

AppManager.add([form]);

AppManager.open('form', {"userText":"Просто текст"}, undefined);

```

<div id="10">  

### 10. Отладка

Как и vue js, так и ui-organizer имеют возможность отладки в VSCode и в браузере.  

Для отладки в VSCode нужно установить соответствующее расширение отладчика, например Debugger Chrome.  

Для отладки в браузере необходимо настроить конфигурацию сборщика, например webpack.  

### Выводы

Конечно же, выводы делать читателям. Авторы заинтересованная сторона и им больше по душе второй подход, т.к. они используют его в своих разработках. Во втором подходе нам нравится простота, однотипность элементов управления, что кнопка, что поле ввода, что список. Одни и те же атрибуты, одни и те же события, методы. А самое главное, - подсказки. Попробуйте создать элемент управления, и вы будете приятно удивлены, как редактор (в нашем случае VSCode) указывает на ошибки и подсказывает в трудной ситуации.