aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--files/ru/learn/tools_and_testing/client-side_javascript_frameworks/angular_item_component/index.md456
1 files changed, 456 insertions, 0 deletions
diff --git a/files/ru/learn/tools_and_testing/client-side_javascript_frameworks/angular_item_component/index.md b/files/ru/learn/tools_and_testing/client-side_javascript_frameworks/angular_item_component/index.md
new file mode 100644
index 0000000000..ceead04b0d
--- /dev/null
+++ b/files/ru/learn/tools_and_testing/client-side_javascript_frameworks/angular_item_component/index.md
@@ -0,0 +1,456 @@
+---
+title: Создание компонента элемента (item component)
+slug: >-
+ Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_item_component
+original_slug: >-
+ Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_item_component
+translation_of: >-
+ Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_item_component
+tags:
+ - Новички
+ - Фреймворки
+ - JavaScript
+ - Изучение
+ - client-side
+ - Angular
+ - Компоненты
+ - События
+ - Данные
+---
+{{LearnSidebar}}{{PreviousMenuNext("Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_styling","Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_filtering", "Learn/Tools_and_testing/Client-side_JavaScript_frameworks")}}
+
+Компоненты представляют способ для организации вашего приложения. В этой статье описывается создание компонента для обработки отдельных элементов в списке и добавление функций проверки, редактирования и удаления.
+
+<table>
+ <tbody>
+ <tr>
+ <th scope="row">Необходимые условия:</th>
+ <td>
+ Понимание основ <a href="/en-US/docs/Learn/HTML">HTML</a>, <a href="/en-US/docs/Learn/CSS">CSS</a> и <a href="/en-US/docs/Learn/JavaScript">JavaScript</a> знание <a href="/en-US/docs/Learn/Tools_and_testing/Understanding_client-side_tools/Command_line">терминала/командной строки
+ </td>
+ </tr>
+ <tr>
+ <th scope="row">Цель:</th>
+ <td>
+ Изучить больше информации о компонентах, включая обработку событий, чтобы реализовать функции проверки, редактирования и удаления.</td>
+ </tr>
+ </tbody>
+</table>
+
+## Создание нового компонента
+
+Создайте компонент с именем `item` с помощью следующих команд в командной строке:
+
+```bash
+ng generate component item
+```
+
+Команда `ng generate component` создаст компонент и папку с указаным именем.
+Имя папки и компонента будет `item`.
+Вы можете найти дирректорию `item` внутри папки `app`.
+
+Как и в случае с `AppComponent`, `ItemComponent` состоит из следующих файлов:
+
+- `item.component.html` для HTML
+- `item.component.ts` для логики
+- `item.component.css` для стилей
+
+Можно увидеть ссылку на файлы HTML и CSS в метаданных декоратора `@Component()` в `item.component.ts`.
+
+```js
+@Component({
+ selector: 'app-item',
+ templateUrl: './item.component.html',
+ styleUrls: ['./item.component.css'],
+})
+```
+
+## Добавьте HTML для ItemComponent
+
+С помощью компонента `ItemComponent` можно будет отмечать элементы списка как выполненные, редактировать или удалять их.
+
+Добавьте разметку для управления элементами, заменив содержимое `item.component.html` следующим кодом:
+
+```html
+<div class="item">
+
+ <input [id]="item.description" type="checkbox" (change)="item.done = !item.done" [checked]="item.done" />
+ <label [for]="item.description">\{{item.description}}</label>
+
+ <div class="btn-wrapper" *ngIf="!editable">
+ <button class="btn" (click)="editable = !editable">Edit</button>
+ <button class="btn btn-warn" (click)="remove.emit()">Delete</button>
+ </div>
+
+ <!-- Этот блок отображается, если пользователь кликнул на кнопку Edit -->
+ <div *ngIf="editable">
+ <input class="sm-text-input" placeholder="edit item" [value]="item.description" #editedItem (keyup.enter)="saveItem(editedItem.value)">
+
+ <div class="btn-wrapper">
+ <button class="btn" (click)="editable = !editable">Cancel</button>
+ <button class="btn btn-save" (click)="saveItem(editedItem.value)">Save</button>
+ </div>
+ </div>
+
+</div>
+```
+
+Чекбокс позволяет пользователям отмечать элементы как выполненные.
+Двойные фигурные скобки, `\{{}}`, в `<input>` и `<label>` означают Angular-интерполяцию.
+Angular использует `\{{item.description}}` для получения описания текущего `item` из массива `items`.
+В следующем разделе подробно объясняется, как компоненты обмениваются данными.
+
+Следующие две кнопки для редактирования и удаления текущего элемента находятся внутри `<div>`.
+К этому `<div>` применена `*ngIf` — это встроенная Angular-директива, которую вы можете использовать для динамического изменения структуры DOM.
+
+В данном случае `*ngIf` означает, что если `editable` равен `false`, то `<div>` будет присутствовать в DOM. Если `editable` равен `true`, Angular удалит этот `<div>` из DOM.
+
+```html
+<div class="btn-wrapper" *ngIf="!editable">
+ <button class="btn" (click)="editable = !editable">Edit</button>
+ <button class="btn btn-warn" (click)="remove.emit()">Delete</button>
+</div>
+```
+
+Когда пользователь кликает на кнопку **Edit**, `editable` становится true, что удаляет `<div>` и его дочерние элементы из DOM.
+Если вместо клика по кнопке **Edit**, пользователь кликает по кнопке **Delete**, `ItemComponent` вызывает событие, которое уведомляет `AppComponent` об удалении.
+
+Директива `*ngIf` так же применяется к следующему `<div>`.
+В этом случае, если `editable` равен `true`, Angular вставляет `<div>` и его дочерние элементы `<input>` и `<button>` в DOM.
+
+```html
+<!-- Этот блок отображается, если пользователь кликнул на кнопку Edit -->
+<div *ngIf="editable">
+ <input class="sm-text-input" placeholder="edit item" [value]="item.description" #editedItem (keyup.enter)="saveItem(editedItem.value)">
+
+ <div class="btn-wrapper">
+ <button class="btn" (click)="editable = !editable">Cancel</button>
+ <button class="btn btn-save" (click)="saveItem(editedItem.value)">Save</button>
+ </div>
+</div>
+```
+
+С помощью `[value]="item.description"`, значение `<input>` будет привязано к `description` текущего элемента.
+Эта привязка делает значение `description` значением `<input>`.
+Поэтому если в `description` находится текст `eat`, то он будет автоматически отображён в `<input>`.
+Таким образом, когда пользователь начнёт редактировать элемент, в `<input>` уже будет находится текст `eat`.
+
+Переменная шаблона `#editedItem` в `<input>` означает, что Angular хранит все, что пользователь печатает в этом `<input>` в переменной `editedItem`.
+Событие `keyup` вызывает метод `saveItem()` и передаёт туда значение из `editedItem`, если пользователь нажмёт Enter вместо кнопки **Save**.
+
+Когда пользователь нажимает кнопку **Cancel**, `editable` переключается на `false`, что удаляет поле для ввода и кнопки для редактирования из DOM.
+Когда в `editable` находится значение `false`, Angular вставляет `<div>` вместе с кнопками **Edit** и **Delete** обратно в DOM.
+
+Нажатие на кнопку **Save** вызывает метод `saveItem()`.
+Метод `saveItem()` берёт значение из переменной `#editedItem` `<input>` и изменяет описание элемента `description` на строку `editedItem.value`.
+
+## Подготовка AppComponent
+
+В следующем разделе вы добавите код, который свяжет `AppComponent` и `ItemComponent`.
+Сначала настройте AppComponent, добавив в `app.component.ts`:
+
+```js
+remove(item) {
+ this.allItems.splice(this.allItems.indexOf(item), 1);
+}
+```
+
+Метод `remove()` использует JavaScript-метод `Array.splice()` для удаления одного элемента, начиная с индекса, который возвращает метод `indexOf`.
+Чтобы узнать больше о методе `splice()`, посмотрите статью на MDN [`Array.prototype.splice()`](/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/splice).
+
+## Добавление логики в ItemComponent
+
+Чтобы использовать `ItemComponent` вы должны добавить к нему логику и способы ввода/вывода данных.
+
+В `item.component.ts` отредактируйте JavaScript-импорты так:
+
+```js
+import { Component, Input, Output, EventEmitter } from '@angular/core';
+import { Item } from "../item";
+```
+
+`Input`, `Output`, и `EventEmitter` позволяют `ItemComponent` использовать данные совместно с `AppComponent`.
+Импортируя `Item`, `ItemComponent` может понять, что такое `item`.
+
+В `item.component.ts`, замените сгенерированный класс `ItemComponent` следующим:
+
+```js
+export class ItemComponent {
+
+ editable = false;
+
+ @Input() item: Item;
+ @Input() newItem: string;
+ @Output() remove = new EventEmitter<Item>();
+
+ saveItem(description) {
+ if (!description) return;
+ this.editable = false;
+ this.item.description = description;
+ }
+}
+```
+
+Свойство `editable` помогает переключать раздел шаблона, где пользователь может редактировать элемент.
+`editable` это свойство, которое находится в директиве `*ngIf`: `*ngIf="editable"`.
+Когда вы используете свойство в шаблоне, вы так же должны объявить его в классе.
+
+`@Input()`, `@Output()`, и `EventEmitter` облегчают связь между вашими двумя компонентами.
+`@Input()` служит дверью, через которую данные входят в компонент, а `@Output()` служит дверью для выхода данных.
+`@Output()` должен быть типа `EventEmitter`, чтобы компонент мог вызвать событие, когда данные готовы для отправки в другой компонент.
+
+Используйте `@Input()`, чтобы указать, что значение свойства может поступать извне.
+Используйте `@Output()` в сочетании с `EventEmitter`, чтобы указать, что свойство может отправить значение в другой компонент.
+
+Метод `saveItem()` принимает в качестве аргумента `description`.
+`description` это текст, который пользователь вводит в HTML-тэг `<input>` во время редактирования элемента списка.
+`description` это та же строка из `<input>`, полученная через переменную шаблона `#editedItem`.
+
+Если пользователь не вводит значение, но нажимает **Save**, `saveItem()` ничего не возвращает и не обновляет `description`.
+Если бы выражения `if` не было, пользователь бы нажимал **Save** с пустым `<input>`, и `description` становился бы пустой строкой.
+
+Если пользователь вводит текст и нажимает сохранить, `saveItem()` устанавливает `editable` в false, что заставляет `*ngIf` в шаблоне удалить форму редактирования и отобразить кнопки **Edit** и **Delete** снова.
+
+Несмотря на то что приложение уже должно компилироваться, вам нужно использовать `ItemComponent` в `AppComponent`, чтобы вы могли увидеть новые функции в браузере.
+
+## Использование ItemComponent в AppComponent
+
+Включение одного компонента в другой в контексте родительско-дочерних отношений даёт вам гибкость в использовании компонентов.
+
+`AppComponent` служит оболочкой для включения других компонентов.
+
+Чтобы использовать `ItemComponent` в `AppComponent`, вставьте селектор `ItemComponent` в шаблон `AppComponent`.
+В Angular селектор компонента указывается в метаданных декоратора `@Component()`.
+В этом примере селектор это `app-item`:
+
+```js
+@Component({
+ selector: 'app-item',
+ templateUrl: './item.component.html',
+ styleUrls: ['./item.component.css']
+})
+```
+
+Чтобы использовать селектор `ItemComponent` в `AppComponent`, добавьте элемент `<app-item>`, что соответствует селектору, добавленному в класс компонента, в `app.component.html`.
+Замените текущий неупорядоченный список в `app.component.html` обновлённой версией:
+
+```html
+<h2>\{{items.length}} <span *ngIf="items.length === 1; else elseBlock">item</span>
+<ng-template #elseBlock>items</ng-template></h2>
+
+<ul>
+ <li *ngFor="let item of items">
+ <app-item (remove)="remove(item)" [item]="item"></app-item>
+ </li>
+</ul>
+```
+
+Двойные фигурные скобки, `\{{}}`, в `<h2>` интерполируют длину массива `items` и отображает её.
+
+`<span>` в `<h2>` использует `*ngIf` и `else` чтобы определить, должен ли `<h2>` отображать строку "item" или "items".
+Если в списке всего один элемент, `<span>` будет содержать строку "item".
+В противном случае, если длина массива `items` будет отличаться от `1`, то вместо `<span>` отобразится шаблон `<ng-template>`, который мы назвали `elseBlock`с помощью синтаксиса `#elseBlock`.
+Angular позволяет использовать `<ng-template>`, если вы не хотите, чтобы контент отображался по умолчанию.
+В нашем случае, когда длина массива `items` не `1`, `*ngIf` отображает `elseBlock` а не `<span>`.
+
+`<li>` использует Angular-директиву для повторения,`*ngFor`, чтобы перебрать все элементы массива `items`.
+В Angular `*ngFor` это что-то вроде `*ngIf` - ещё одна директива, которая помогает изменять структуру DOM с помощью меньшего количества кода.
+Для каждого элемента `item`, Angular повторяет `<li>` и всё, что в нём находится, включая `<app-item>`.
+Это означает, что для каждого элемента в массиве Angular создаёт новый экземпляр `<app-item>`.
+Для любого количества элементов в массиве Angular создаёт множество элементов `<li>`.
+
+Вы можете использовать `*ngFor` для других элементов, таких как `<div>`, `<span>` или `<p>` и д.р.
+
+В `AppComponent` есть метод `remove()` для удаления элемента, связанного со свойством `remove` в `ItemComponent`.
+Свойство `item` в квадратных скобках `[]` связывает значение `item` между `AppComponent` и `ItemComponent`.
+
+Теперь вы можете редактировать и удалять элементы списка.
+Когда вы добавляете или удаляете элементы, количество элементов так же должно изменятся.
+Чтобы сделать список более удобным, добавьте немного стилей в `ItemComponent`.
+
+## Добавление стилей в ItemComponent
+
+Вы можете использовать стили, чтобы изменить внешний вид компонента.
+Следующий CSS добавляет базовые стили, flexbox для кнопок и стилизованные чекбоксы.
+
+Вставьте следующе стили в `item.component.css`.
+
+```css
+.item {
+ padding: .5rem 0 .75rem 0;
+ text-align: left;
+ font-size: 1.2rem;
+}
+
+.btn-wrapper {
+ margin-top: 1rem;
+ margin-bottom: .5rem;
+}
+
+.btn {
+ /* flexbox стили для кнопок меню */
+ flex-basis: 49%;
+}
+
+.btn-save {
+ background-color: #000;
+ color: #fff;
+ border-color: #000;
+
+}
+
+.btn-save:hover {
+ background-color: #444242;
+}
+
+.btn-save:focus {
+ background-color: #fff;
+ color: #000;
+}
+
+.checkbox-wrapper {
+ margin: .5rem 0;
+}
+
+.btn-warn {
+ background-color: #b90000;
+ color: #fff;
+ border-color: #9a0000;
+}
+
+.btn-warn:hover {
+ background-color: #9a0000;
+}
+
+.btn-warn:active {
+ background-color: #e30000;
+ border-color: #000;
+}
+
+.sm-text-input {
+ width: 100%;
+ padding: .5rem;
+ border: 2px solid #555;
+ display: block;
+ box-sizing: border-box;
+ font-size: 1rem;
+ margin: 1rem 0;
+}
+
+/* Пользовательские чекбоксы
+Адаптировано из https://css-tricks.com/the-checkbox-hack/#custom-designed-radio-buttons-and-checkboxes */
+
+/* Основа для стилизации лэйбла */
+[type="checkbox"]:not(:checked),
+[type="checkbox"]:checked {
+ position: absolute;
+ left: -9999px;
+}
+[type="checkbox"]:not(:checked) + label,
+[type="checkbox"]:checked + label {
+ position: relative;
+ padding-left: 1.95em;
+ cursor: pointer;
+}
+
+/* чекбокс */
+[type="checkbox"]:not(:checked) + label:before,
+[type="checkbox"]:checked + label:before {
+ content: '';
+ position: absolute;
+ left: 0; top: 0;
+ width: 1.25em; height: 1.25em;
+ border: 2px solid #ccc;
+ background: #fff;
+}
+
+/* галочка для чекбокса */
+[type="checkbox"]:not(:checked) + label:after,
+[type="checkbox"]:checked + label:after {
+ content: '\2713\0020';
+ position: absolute;
+ top: .15em; left: .22em;
+ font-size: 1.3em;
+ line-height: 0.8;
+ color: #0d8dee;
+ transition: all .2s;
+ font-family: 'Lucida Sans Unicode', 'Arial Unicode MS', Arial;
+}
+/* изменение галочки чекбокса */
+[type="checkbox"]:not(:checked) + label:after {
+ opacity: 0;
+ transform: scale(0);
+}
+[type="checkbox"]:checked + label:after {
+ opacity: 1;
+ transform: scale(1);
+}
+
+/* доступность */
+[type="checkbox"]:checked:focus + label:before,
+[type="checkbox"]:not(:checked):focus + label:before {
+ border: 2px dotted blue;
+}
+```
+
+## Резюме
+
+Теперь у вас должен быть стилизованный Angular-компонент приложения списка дел, позволяющий добавлять, редактировать и удалять элементы.
+Следующий шаг — добавление фильтрации, чтобы можно было просматривать элементы соответствующие определённым критериям.
+
+{{PreviousMenuNext("Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_styling","Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_filtering", "Learn/Tools_and_testing/Client-side_JavaScript_frameworks")}}
+
+## В это модуле
+
+- [Введение в клиентские фреймворки](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Introduction)
+- [Основные функции фреймворков](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Main_features)
+- React
+
+ - [Начало работы с React](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/React_getting_started)
+ - [Начало создания приложения списка дел с React](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/React_todo_list_beginning)
+ - [Компоненты React-приложения](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/React_components)
+ - [Интерактивность React: события и состояние](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/React_interactivity_events_state)
+ - [Интерактивность React: редактирование, фильтрация, условная отрисовка](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/React_interactivity_filtering_conditional_rendering)
+ - [Доступность в React](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/React_accessibility)
+ - [Ресурсы по React](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/React_resources)
+
+- Ember
+
+ - [Начало работы с Ember](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Ember_getting_started)
+ - [Структура Ember-приложения и компоненты](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Ember_structure_componentization)
+ - [Интерактивность Ember: события, классы и состояние](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Ember_interactivity_events_state)
+ - [Интерактивность Ember: функциональность подвала, условная отрисовка](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Ember_conditional_footer)
+ - [Маршрутизация в Ember](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Ember_routing)
+ - [Ресурсы по Ember и устранение неполадок](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Ember_resources)
+
+- Vue
+
+ - [Начало работы с Vue](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Vue_getting_started)
+ - [Создание вашего первого компонента Vue](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Vue_first_component)
+ - [Отрисовка списка Vue-компонентов](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Vue_rendering_lists)
+ - [Добавление новой формы во Vue: события, методы, и модели](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Vue_methods_events_models)
+ - [Стилизация Vue-компонентов с CSS](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Vue_styling)
+ - [Использование вычисляемых свойств во Vue](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Vue_computed_properties)
+ - [Условная отрисовка во Vue: Редактирование существующих дел](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Vue_conditional_rendering)
+ - [Управление фокусом во Vue с помощью refs](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Vue_refs_focus_management)
+ - [Ресурсы по Vue](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Vue_resources)
+
+- Svelte
+
+ - [Начало работы со Svelte](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Svelte_getting_started)
+ - [Начинаем приложение списка дел c использованием Svelte](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Svelte_Todo_list_beginning)
+ - [Динамическое поведение в Svelte: работа с переменными и пропсами](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Svelte_variables_props)
+ - [Компоненты Svelte-приложения](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Svelte_components)
+ - [Продвинутый Svelte: реактивность, жизненный цикл, доступность](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Svelte_reactivity_lifecycle_accessibility)
+ - [Работа хранилищами в Svelte](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Svelte_stores)
+ - [Поддержка TypeScript в Svelte](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Svelte_TypeScript)
+ - [Развёртывание и следующие шаги](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Svelte_deployment_next)
+
+- Angular
+
+ - [Начало работы с Angular](/ru/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_getting_started)
+ - [Начинаем приложение списка дел с использованием Angular](/ru/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_todo_list_beginning)
+ - [Стилизация Angular-приложения](/ru/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_styling)
+ - [Создание компонента элемента (item component)](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_item_component)
+ - [Фильтрация списка дел](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_filtering)
+ - [Сборка Angular-приложений и другие ресурсы](/en-US/docs/Learn/Tools_and_testing/Client-side_JavaScript_frameworks/Angular_building)