React, Drag&Drop и performance

в 3:28, , рубрики: Drag&Drop, javascript, JS, React, ReactJS, Блог компании Macte, интерфейсы

React, Drag&Drop и performance - 1

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

В одном из последних проектов нам предстояло реализовать систему для управления учебным процессом образовательного учреждения. То, что у нас получилось в итоге, можно посмотреть здесь — https://habrahabr.ru/company/macte/blog/341726/

К интерфейсу редактирования расписания были предъявлены следующие требования:

  1. возможность создания, редактирования и удаление занятий;
  2. в рамках одной пары занятие может проводиться сразу у двух групп;
  3. возможность переноса занятия в сетке расписания.

С первыми двумя пунктами никаких проблем не возникло, а вот с третьим пришлось повозиться. На нем и остановимся поподробнее.

React и Drag&Drop

Для начала нам необходимо выбрать Drag&Drop библиотеку. На просторе интернета их великое множество: DraggableJS, dragula, interactjs.io и пр.  А библиотек, заточенных для использования вместе с React, всего две: React-DnD и react-beautiful-dnd.

Библиотека react-beautiful-dnd отлично выглядит на демках, но, к сожалению, вышла уже после реализации проекта. Поэтому мы использовали React-DnD.

Про react-beautiful-dnd Alex Reardon написал статью —  «Rethinking drag and drop», которую можно почитать в переводе на хабре — https://habrahabr.ru/company/edison/blog/339086/

React DnD

Данная библиотека предоставляет нам набор из компонентов высшего порядка (HOC). Если говорить простым языком то:

  • DragSource — делает компонент перетаскиваемым;
  • DropTarget — добавляет компоненту возможность взаимодействовать с перетаскиваемыми компонентами;
  • DragLayer — позволяет реализовать собственное превью для перетаскиваемого элемента;
  • DragDropContext — предназначен для инициализации библиотеки.

Еще одна важная составляющая без которой React DnD не заработает — это drag&drop backend. Библиотека для обеспечения кроссбраузерности, абстракция над стандартным браузерным API.

Авторы React DnD советуют использовать HTML5-Backend, хотя совсем и не обязательно. Можно выбрать любой другой или написать свой.

Реализация

Для начала разобьем верстку сетки расписания на четыре основных компонента:

  1. Сетка расписания — ScheduleGrid (в этом компоненте мы будем инициализировать библиотеку React-DnD)
  2. Блок с расписанием на день — ScheduleColumn
  3. Секция с парами — SubjectSeciton (в нашем случае это будет DropTarget)
  4. Отдельное занятие — SubjectItem (в нашем случае это будет DragSource)

React, Drag&Drop и performance - 2

визуально разметили наши компоненты

Компонент App

Реализуем базовый компонент-контейнер App, который будет хранить информацию о недельном расписании и рендерить описанные выше компоненты.

Листинг компонента App

import React, { Component } from 'react';
import ScrollArea from 'react-scrollbar';

import ScheduleGrid from './ScheduleGrid';

import subjectsArray from './schedule-data';

class App extends Component {

  state = {
    subjectsArray: []
  }

  moveSubject = (movedSubjectId, newPosition) => {
    this.setState({
      subjectsArray: this.state.subjectsArray.map(subject => {
        if (subject.ID == movedSubjectId) {
          return {
            ...subject,
            DAY_OF_WEEK: newPosition.day,
            PERIOD: newPosition.period,
          }
        }
        
        return subject;
      }),
    });
  }

  render() {
    return (
      <div className="app-container">
        <div className="app-container-header">
          <h3>Редактирование расписания</h3>
          <h4>Версия React: {React.version}</h4>
        </div>
        <div className="posit-unit--middle">
          <ScrollArea
            speed={0.8}
            className="area"
            smoothScrolling={true}
            contentClassName="schedule-grid"
            horizontal={true}
            vertical={false}
          >
            <ScheduleGrid
              subjectsArray={this.state.subjectsArray}
              columns={6}
              itemsInColumn={6}
              moveSubject={this.moveSubject}
            />
          </ScrollArea>
        </div>
      </div>
    );
  }
}

export default App;

Формат ответа сервера

[
  {
    "ID": "2833",
    "NAME": "КР-101 (1 пара)",
    "DAY_OF_WEEK": 5,
    "GROUP": "834",
    "NOTICE": null,
    "SHEDULE_TYPE_ID": "1956",
    "SHEDULE_TYPE_NAME": "Практика",
    "SHEDULE_TYPE_CODE": "practice",
    "SUBJECT_ID": "868",
    "SUBJECT_NAME": "информатика",
    "CLASSROOM_ID": "883",
    "CLASSROOM_NAME": "55а-ПМК",
    "EDUCATION": "1",
    "PERIOD": "1",
    "TEACHER_ID": "1732",
    "TEACHER_FIRST_NAME": "Диана",
    "TEACHER_MIDDLE_NAME": "Юрьевна",
    "TEACHER_LAST_NAME": "Матвеева",
    "TEACHER_SHORT_NAME": "Д. Ю. Матвеева"
  },
  {
    "ID": "2832",
    "NAME": "КР-101 (1 пара)",
    "DAY_OF_WEEK": 5,
    "GROUP": "834",
    "NOTICE": null,
    "SHEDULE_TYPE_ID": "1957",
    "SHEDULE_TYPE_NAME": "Занятие",
    "SHEDULE_TYPE_CODE": "lesson",
    "SUBJECT_ID": "1491",
    "SUBJECT_NAME": "информационные сервисы",
    "CLASSROOM_ID": "883",
    "CLASSROOM_NAME": "55а-ПМК",
    "EDUCATION": "1",
    "PERIOD": "1",
    "TEACHER_ID": "1732",
    "TEACHER_FIRST_NAME": "Диана",
    "TEACHER_MIDDLE_NAME": "Юрьевна",
    "TEACHER_LAST_NAME": "Матвеева",
    "TEACHER_SHORT_NAME": "Д. Ю. Матвеева"
  }
]

Для построения сетки занятий нас интересуют поля:

  • PERIOD — номер занятий
  • DAY_OF_WEEK — день недели
  • TEACHER_SHORT_NAME — ФИО преподавателя
  • CLASSROOM_NAME — номер аудитории
  • SUBJECT_NAME — название предмета

ScheduleGrid

Далее приступим к реализации сетки занятий, генерируем колонки  ScheduleColumn.
В этом компоненте инициализируем React DnD. Для этого оборачиваем наш компонент в DragDropContext и передаем ему HTML5-backend.

Листинг ScheduleGrid

import React, { Component } from 'react';
import propTypes from 'prop-types';

import { DragDropContext } from 'react-dnd';
import HTML5Backend from 'react-dnd-html5-backend';

import ScheduleColumn from './Grid/ScheduleColumn';
import ScrollButton from './Grid/ScrollButton';

import throttle from './utils/throttle.js';

class ScheduleGrid extends Component {

  static contextTypes = {
    scrollArea: propTypes.object,
  };

  constructor(props, context) {
    super(props);
    this.weekDays = ['Пн', 'Вт', 'Ср', 'Чт', 'Пт', 'Сб'];
    this.scrollLeft = throttle(context.scrollArea.scrollLeft, 1500);
    this.scrollRight = throttle(context.scrollArea.scrollRight, 1500);
  } 
  
  printColumns = () => {
    return Array.from({ length: this.props.columns }, (el, index) => (
      <ScheduleColumn 
        itemsInColumn={this.props.itemsInColumn}
        moveSubject={this.props.moveSubject}
        subjectsArray={this.props.subjectsArray}
        xPos={index + 1}
        key={index}
        weekDay={this.weekDays[index]} />
    ));
  }

  render() {
    return (
      <div>
        <div className="table-schedule table-schedule--days clearfix">
          {this.printColumns()}          
        </div>
        <div className="navigation-table-day">
          <ScrollButton type={'prev'} handleClick={this.scrollLeft} />
          <ScrollButton type={'next'} handleClick={this.scrollRight} />
        </div>         
      </div>
    );
  }
}

export default DragDropContext(HTML5Backend)(ScheduleGrid);

ScheduleColumn

Каждая из колонок состоит из нескольких секций. Секции имеет координаты xPos и yPos, которые соответствуют полям PERIOD и DAY_OF_WEEK из API.

Листинг ScheduleColumn

import React, { Component } from 'react';

import SubjectSection from './SubjectSection';

class ScheduleColumn extends Component {

  getSectionData = (x, y) => {
    return this.props.subjectsArray.filter(subject => {
      return subject.DAY_OF_WEEK == x && subject.PERIOD == y;
    });
  }

  generateSection = yPos => {    
    const { xPos, emptyColumnItemClick, onColumnItemClick } = this.props;
    const sectionData = this.getSectionData(xPos, yPos).slice(0, 2);

    return (
      <SubjectSection        
        key={`${yPos}_${xPos}`}
        yPos={yPos}
        xPos={xPos}
        sectionData={sectionData}
        moveSubject={this.props.moveSubject}
      />
    );
  }

  render() {
    const { weekDay, itemsInColumn } = this.props;

    return (
      <div className="table-schedule--column">
        <div className="day">{weekDay}</div>
        <div className="technical-data">
          <div className="items">     
            <div className="item subject">Предмет:</div>
            <div className="item teacher">Преподаватель:</div>
            <div className="item lecture">Аудитория:</div>
          </div>
        </div>
        <div className="couples-description">
          <div className="sections">
            {Array.from({ length: this.props.itemsInColumn }, (el, index) => this.generateSection(index + 1))}
          </div>
        </div>
      </div>
    ); 
  }  
}

export default ScheduleColumn;

SubjectSection

В терминологии React DnD, данный компонент является DropTarget, т.е. предназначен для взаимодействия с другими перетаскиваемыми компонентами.

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

Необходимо также задать ему тип. В нашем случае это — subjectItem. Теперь в него можно перетаскивать компоненты DragSource с аналогичным типом.

Листинг SubjectSection

import React, { Component } from 'react';
import { DropTarget } from 'react-dnd';
import className from 'classnames';

import SubjectItem from './SubjectItem';

const SubjectSectionTarget = {
  drop(props) {
    return props;
  },
  canDrop(props) {
    return props.sectionData.length <= 1;
  },
};

const collect = (connect, monitor) => {
  return {
    connectDropTarget: connect.dropTarget(),
    isOver: monitor.isOver(),
  };
}

class SubjectSection extends Component {

  itemTemplate(xPos, yPos, styles, isOver) {
    const itemClass = className({
      'section': true,
      'section--2-elements': this.props.sectionData.length > 1 || isOver, 
      'section--drag-here': this.props.sectionData.length <= 1 && isOver,
    });

    return (
      <div
        key={`${xPos}_${yPos}`}
        style={styles}
        className={itemClass}       
      >
        {this.props.sectionData.map((data, index, sectionData) => {
          return (
            <SubjectItem
              key={data.ID}
              moveSubject={this.props.moveSubject}
              data={data}
              index={index}
              sectionData={sectionData}
            />
          );
        })}
      </div>
    );
  }

  emptyItemTemplate(xPos, yPos, styles) {
    return (
      <div
        key={`${xPos}_${yPos}`}
        style={styles}
        className="section section--is-dragging"
      >
        <div className="technical-data" />
      </div>
    );
  }

  render() {
    const { connectDropTarget, isOver, sectionData, xPos, yPos } = this.props;
    
    const styles = isOver ? { opacity: 0.7 } : null;
    const sectionIsEmpty = sectionData.length === 0;

    const subjectSection =
      sectionIsEmpty
        ? this.emptyItemTemplate(xPos, yPos, styles)
        : this.itemTemplate(xPos, yPos, styles, isOver);

    return connectDropTarget(subjectSection);
  }
}

export default DropTarget('subjectItem', SubjectSectionTarget, collect)(SubjectSection);

Почти закончили, осталось только реализовать компонент для отображения информации о занятии SubjectItem

SubjectItem

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

Листинг SubjectItem

import React, { Component } from 'react';
import { DragSource } from 'react-dnd';
import classNames from 'classnames';
import equals from 'shallow-equals';

import SubjectContent from './SubjectContent';

const subjectSource = {
  beginDrag(props, monitor, component) {
    return props;
  },
  endDrag(props, monitor, component) {
    
    if (!monitor.didDrop()) {
      return;
    }
    const item = monitor.getItem();
    const dropResult = monitor.getDropResult();

    props.moveSubject(item.data.ID, {
      day: dropResult.xPos,
      period: dropResult.yPos,
    });
  },
};

function collect(connect, monitor) {
  return {
    connectDragSource: connect.dragSource(),
    isDragging: monitor.isDragging(),
    connectDragPreview: connect.dragPreview(),
  };
}

class SubjectItem extends Component {
 
  onTooltipClick(event) {
    event.preventDefault();
    return false;
  }    

  render() {
    const { connectDragSource, isDragging, index, sectionData, data } = this.props;

    const itemClass = classNames({
      'technical-data': true,
      'l-separation': index === 0 && sectionData.length > 1,
    });

    return connectDragSource(
      <a
        href="#"
        style={{ opacity: isDragging ? 0 : 1 }}
        onClick={this.onTooltipClick}
        className={itemClass}
      >
        <div style={{ height: '100%' }}>
          <SubjectContent 
            data={data}
            isDragging={isDragging}
          />          
        </div>
      </a>
    );
  }
}

export default DragSource('subjectItem', subjectSource, collect)(SubjectItem);

Листинг SubjectContent

import React from 'react';
import ReactTooltip from 'react-tooltip';

const printSubjectType = ({ SHEDULE_TYPE_CODE, ID, SHEDULE_TYPE_NAME, NOTICE }) => (
  <span className="types-classes types-classes--vertical">
    <ReactTooltip
      delayShow={250}
      id={`tolltip${ID}`}
      place="bottom"
      class="customeTheme"
      effect="solid"
    />
    <span className="types-classes__items">
      {
        SHEDULE_TYPE_CODE != 'lesson' ? (
          <span
            className="types-classes__item"
            data-tip={SHEDULE_TYPE_NAME}
            data-for={`tolltip${ID}`}
          >
            <span
              className={
                'type-occupation type-occupation--' +
                SHEDULE_TYPE_CODE +
                ' icon'
              }
            />
          </span>
      ) : null }
      {NOTICE ? (
        <span
          className="types-classes__item"
          data-tip={NOTICE}
          data-for={`tolltip${ID}`}
        >
          <span className="type-occupation type-occupation--note icon" />
        </span>
      ) : null}
    </span>
  </span>
);


const SubjectContent = props => {
  const { isDragging } = props;
  const { SUBJECT_NAME = '(нет)', TEACHER_SHORT_NAME = '(нет)', CLASSROOM_NAME = '(нет)' } = props.data;

  return (
    <span className="items">
      <span className="item subject">
        <span className="imit-table">
          <span className="imit-table--column">
            {SUBJECT_NAME}
          </span>
        </span>
      </span>
      <span className="item teacher">
        <span className="imit-table">
          <span className="imit-table--column">
            {TEACHER_SHORT_NAME}
          </span>
        </span>
      </span>
      <span className="item lecture">
        <span className="imit-table">
          <span className="imit-table--column">
            {CLASSROOM_NAME}
          </span>
          {printSubjectType(props.data)}
        </span>
      </span>
    </span> 
  )
};

export default SubjectContent;

Ну что, вроде бы все готово. Можно приступать к тестам.

Тестируем

Запускаем наш чудо-интерфейс и пробуем переместить предмет.

React, Drag&Drop и performance - 3

«Вот, блин!» — сказал мне Google Chrome 62, а в Firefox 57 все отработало нормально.

Позже выяснилось, что React-DnD конфликтует с некоторыми библиотеками, например ReactTooltip. Есть даже открытый issue на github.

Ну что, tooltip нам нужен. Придется как-то фиксить. Попробуем задать свое превью изображение, для этого добавим буквально пару строк.

Фиксим вылет браузера

function collect(connect, monitor) {
  return {
    connectDragSource: connect.dragSource(),
    isDragging: monitor.isDragging(),
    connectDragPreview: connect.dragPreview(),
  };
}

class SubjectItem extends Component {

  componentDidMount() {
    const img = new Image();
    img.src = '';
    img.onload = () => this.props.connectDragPreview(img);
  }

Обновляем страницу и проверяем.

React, Drag&Drop и performance - 4

Так, теперь все работает. Но коня, конечно, необходимо убрать. Заменим его на прозрачный пиксель.

Убираем коня

componentDidMount() {
    const img = new Image();
    img.src = '';
    img.onload = () => this.props.connectDragPreview(img);
  }

Интересное замечание. Если в Windows выбрать упрощенную цветовую схему «Windows Classic», то в браузере не будет отображаться тень (preview) при перемещении от drag&drop.

React, Drag&Drop и performance - 5

после установки прозрачного превью

Уже лучше, однако, все равно похоже на дешевую подделку. Что ж, будем реализовывать свое превью, чтобы не зависеть от особенностей браузеров и ОС.

В React-DnD для этого предусмотрен DragLayer — компонент, который будет отображаться при перемещении DragSource.

Листинг GridDragLayer

import React, { Component } from 'react';
import { DragLayer } from 'react-dnd';
import SubjectContent from './SubjectContent';

function collect(monitor) {
  return {
    item: monitor.getItem(),
    currentOffset: monitor.getSourceClientOffset(),
    isDragging: monitor.isDragging(),
  };
}

function getItemTransform(props) {
  const { currentOffset } = props;
  if (!currentOffset) {
    return {
      display: 'none',
    };
  }

  const { x, y } = currentOffset;
  const transform = `translate(${x}px, ${y}px) rotate(3deg)`;
  return {  
    position: 'fixed', 
    display: 'block',
    zIndex: 10000,
    transform: transform,
    WebkitTransform: transform,
    cursor: 'move',
  };
}

class GridDragLayer extends Component {
  constructor(props) {
    super(props);
    this.lastUpdate = +new Date();
  }

render() {
    const { item, isDragging } = this.props;

    if (!isDragging) {
      return null;
    }

    return (
      <div
        id="drag-placeholder"
        style={getItemTransform(this.props)}
      >
        <SubjectContent
          data={item.data}
          isDragging={isDragging}
        />
      </div>
    );
  }
}

export default DragLayer(collect)(GridDragLayer);

Помещаем наш GridDragLayer в метод ScheduleGrid.render

Листинг ScheduleGrid

render() {
    return (
      <div>
        <GridDragLayer />
        <div className="table-schedule table-schedule--days clearfix">
          {this.printColumns()}          
        </div>
        <div className="navigation-table-day">
          <ScrollButton type={'prev'} handleClick={this.scrollLeft} />
          <ScrollButton type={'next'} handleClick={this.scrollRight} />
        </div>         
      </div>
    );
  }

В очередной раз проверяем.

React, Drag&Drop и performance - 6

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

В Chrome DevTools переходим на вкладку Performance и включаем CPU 4x slowdown. Видим примерную картину того как будет работать у обычного пользователя.

React, Drag&Drop и performance - 7

Решаем проблему производительности

shouldComponentUpdate

Основная проблема заключается в том, что на каждое событие drag (а триггерится оно очень часто) React перерисовывает компонент GridDragLayer. Выполняется большое число ненужных операций.

Чтобы избавится от лишних перерисовок реализуем метод shouldComponentUpdate в GridDragLayer. Для плавности нам нужно обеспечить 60fps, т.е. одна перерисовка на 16 мс.

Листинг shouldComponentUpdate

constructor(props) {
    super(props);
    this.lastUpdate = +new Date();
    this.updateTimer = null;
  }

  shouldComponentUpdate(nextProps, nextState) {
    if (+new Date() - this.lastUpdate > 16) {
      this.lastUpdate = +new Date();
      clearTimeout(this.updateTimer);
      return true;
    } else {
      this.updateTimer = setTimeout(() => {
        this.forceUpdate();
      }, 100);
    }
    return false;
  }

Возможные «залипания», когда компонент изменил свое состояние в интервале 16 мс, но не был перерисован, устраняются таймером и forceUpdate.

Проверяем при CPU 4x slowdown. Стало намного шустрее, но все равно недостаточно.

React, Drag&Drop и performance - 8

Реализуем drag placeholder на vanilla js

Для этого немного допишем наш SubjectItem. Реализуем функцию generatePlaceholder, которая возвращает нам разметку элемента. Также напишем обработчик createMouseMoveHandler, который будет изменять положение placeholder. В объект subjectSource добавим event listeners, которые будут реагировать на событие dragover.

Листинг SubjectItem

import React, { Component } from 'react';
import { DragSource } from 'react-dnd';
import classNames from 'classnames';
import equals from 'shallow-equals';
import SubjectContent from './SubjectContent';

import throttle from '../utils/throttle.js';

function generatePlaceholder(item) {
  const placeholder = document.createElement('div');
  placeholder.id = 'drag-placeholder';
  placeholder.style.cssText =
    'display:none;position:fixed;z-index:100000;pointer-events:none;';
  
  placeholder.innerHTML = `<span class="items">
					                    <span class="item subject">			                        
					                        ${item.data.SUBJECT_NAME || '(нет)'}
					                    </span>
					                    <span class="item teacher">
					                        ${item.data.TEACHER_SHORT_NAME || '(нет)'}
					                    </span>
					                    <span class="item lecture">
					                        ${item.data.CLASSROOM_NAME || '(нет)'}
					                    </span>
					                </span>`;
  return placeholder;
}

function createMouseMoveHandler() {
  let currentX = -1;
  let currentY = -1;

  return function(event) {
    let newX = event.clientX - 8;
    let newY = event.clientY - 2;

    if (currentX === newX && currentY === newY) {
      return;
    }

    const dragPlaceholder = document.getElementById('drag-placeholder');
    const transform = 'translate(' + newX + 'px, ' + newY + 'px) rotate(3deg)';

    dragPlaceholder.style.transform = transform;
    dragPlaceholder.style.display = 'block';
  };
}

const mouseMoveHandler = createMouseMoveHandler();

const throttledMoveHandler = throttle(createMouseMoveHandler(), 16);

const subjectSource = {
  beginDrag(props, monitor, component) {
    document.addEventListener('dragover', throttledMoveHandler);
    document.body.insertBefore(
      generatePlaceholder(props),
      document.body.firstChild
    );
    return props;
  },
  endDrag(props, monitor, component) {
    document.removeEventListener('dragover', throttledMoveHandler);
    let child = document.getElementById('drag-placeholder');
    child.parentNode.removeChild(child);

    if (!monitor.didDrop()) {
      return;
    }
    const item = monitor.getItem();
    const dropResult = monitor.getDropResult();

    props.moveSubject(item.data.ID, {
      day: dropResult.xPos,
      period: dropResult.yPos,
    });
  },
};

function collect(connect, monitor) {
  return {
    connectDragSource: connect.dragSource(),
    isDragging: monitor.isDragging(),
    connectDragPreview: connect.dragPreview(),
  };
}

class SubjectItem extends Component {

  componentDidMount() {
    const img = new Image();
    img.src = '';
    img.onload = () => this.props.connectDragPreview(img);
  }
  
  onTooltipClick(event) {
    event.preventDefault();
    return false;
  }    

  render() {
    const { connectDragSource, isDragging, index, sectionData, data } = this.props;

    const itemClass = classNames({
      'technical-data': true,
      'l-separation': index === 0 && sectionData.length > 1,
    });

    return connectDragSource(
      <a
        href="#"
        style={{ opacity: isDragging ? 0 : 1 }}
        onClick={this.onTooltipClick}
        className={itemClass}
      >
        <div style={{ height: '100%' }}>
          <SubjectContent 
            data={data}
            isDragging={isDragging}
          />          
        </div>
      </a>
    );
  }
}

export default DragSource('subjectItem', subjectSource, collect)(SubjectItem);

Также проверяем все при CPU 4x slowdown.

React, Drag&Drop и performance - 9

Теперь все работает как надо!

Конечно, это отступление от good practices, ведь теперь мы полностью продублировали код нашего превью в функции generatePlaceholder. Зато интерфейсом стало пользоваться намного удобней и приятней.

Заключение

После обновления React до 16 версии наш интерфейс не стал работать быстрее. Поэтому мы остановились на варианте с placeholder на vanilla js, поскольку он заметно выигрывает по производительности.

Вот так все работает на production:

React, Drag&Drop и performance - 10

Надеемся, что наш опыт разработки интерфейсов с использованием drag&drop будет полезен и другим разработчикам.

Будем благодарны за ваши комментарии и участие в мини-опросе!

Автор: Владислав Петроченко

Источник

Поделиться