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

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

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

  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», которую можно почитать в переводе на Хабре

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)



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

Компонент 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;


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

Тестируем


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



«Вот, блин!» — сказал мне 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);
  }


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



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

Убираем коня
componentDidMount() {
    const img = new Image();
    img.src = '';
    img.onload = () => this.props.connectDragPreview(img);
  }


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



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

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

В 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>
    );
  }


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



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

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



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


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. Стало намного шустрее, но все равно недостаточно.



Реализуем 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.



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

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

#update
Odrin предложил использовать «renderToStaticMarkup», чтобы избежать дублирования кода в функции generatePlaceholder. Вариант рабочий, количество кода сократилось и не нужно дублировать разметку превью.

Листинг SubjectItem
import React, { Component } from 'react';
import ReactDOMServer from 'react-dom/server';
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 = ReactDOMServer.renderToStaticMarkup(<SubjectContent { ...item } />);
  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);


Заключение


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

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



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

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

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


  1. Nikulio
    11.12.2017 09:23

    Спасибо! Как раз сейчас делаю «убийцу трелло» и думал над днд


  1. AstarothAst
    11.12.2017 10:15

    Одно время была популярна библиотека Dragula, вроде бы. Она сейчас как? Уже не торт?


  1. Odrin
    11.12.2017 10:58

    ведь теперь мы полностью продублировали код нашего превью в функции generatePlaceholder

    Почему бы не использовать «renderToStaticMarkup»? Сам не пробовал, но вроде как у renderToStaticMarkup нет ограничений в использовании на клиенте.


    1. dr_zorge Автор
      11.12.2017 11:01

      Спасибо за комментарий! Обязательно попробую. Если заработает, то допишу в статье!)


  1. jknight
    11.12.2017 11:21

    Perfomance? Ну нельзя же так, в заголовке-то…


    1. dr_zorge Автор
      11.12.2017 11:25

      Спасибо!) Поправил.


  1. RomanPokrovskij
    11.12.2017 11:49

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


    А что действительно нет возможности не перерисовывать весь GridDragLayer? Я правильно этому ужасаюсь, или «так и должно быть», «иначе не сделаешь»? Оно моргать не будет? А на планшетке? Я когда писал расписания с драгндропами с jquery-ui selectable/draggable/droppable (сотни объектов) никаких проблем с производительностью не имел и вообще не задумвывался о fps. Что-то я как то не готов морально к такому прогрессу. Ободрите, что все тут ok.


    1. dr_zorge Автор
      11.12.2017 11:54

      GridDragLayer — маленькое прямоугольное превью, которое появляется только в момент перетаскивания, а не вся таблица с расписанием, как вы наверное подумали. Не моргает, скринкасты есть в статье.


  1. RomanPokrovskij
    11.12.2017 11:52

    А такие замечательные гифки ловящие курсор и ДнД чем делаются?


    1. dr_zorge Автор
      11.12.2017 11:56

      Можно в ScreenToGif, к примеру.


  1. mayorovp
    11.12.2017 12:26

    А что если получить элемент через ref и при движении менять ему стиль напрямую?

    Кстати, чем не устроили свойства left и top? Неужели трансформации работают быстрее?


    1. dr_zorge Автор
      11.12.2017 12:33

      1. Данный вариант оказался проще в плане реализации (если учесть, что используется React DnD)
      2. Если верить этой статье — www.paulirish.com/2012/why-moving-elements-with-translate-is-better-than-posabs-topleft то да. Сам тесты не проводил.


      1. mayorovp
        11.12.2017 12:55

        Хм, а что сложного? Третий параметр component у beginDrag — это ваш компонент. Пишем что-то вроде <div ref={el => this.div = el}> и можно обращаться к этому элементу как component.div вместо placeholder.


        1. dr_zorge Автор
          11.12.2017 13:10

          А что если в «секции» находится две пары и мы перемещаем одну из них? Плейсхолдер получится в 1/2 от обычной высоты и будет выглядеть не так эстетично)


          1. mayorovp
            11.12.2017 13:16

            Это вы вообще о чем?


            1. dr_zorge Автор
              11.12.2017 13:20




          1. mayorovp
            11.12.2017 13:30

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


            А ведь то же самое можно и с рефом сделать...


            1. dr_zorge Автор
              11.12.2017 13:45

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


  1. mayorovp
    11.12.2017 12:45

    Хм, а в чем преимущества рендера в строку с последующим парсингом перед однократным вызовом ReactDOM.render?


    Не пробовали вместо


      placeholder.innerHTML = ReactDOMServer.renderToStaticMarkup(<SubjectContent { ...item } />);

    написать вот так:


      ReactDOM.render(<SubjectContent { ...item } />, placeholder);


    1. dr_zorge Автор
      11.12.2017 13:07

      Реализовал по совету Odrin, тем самым решив проблему с дублированием кода. Можно попробовать и ваш подход, но не думаю что это даст нам существенный прирост по производительности. Сам метод вызывается всего один раз для создания превью и не является узким местом) В любом случае спасибо!)


  1. gnv_cor
    11.12.2017 13:31

    Недавно работал тоже была необходимость в Drag&Drop. Использовал вот это github.com/danielstocks/react-sortable (простенькая и ничего лишнего)


  1. indestructable
    11.12.2017 13:31

    Не совсем понял из статьи и из комментариев, почему нельзя оптимизировать перерисовку компонента до обновления только свойств top и left (или transform)?


    1. dr_zorge Автор
      11.12.2017 13:36

      В варианте с GridDragLayer так и было, но результат получился не очень. Без троттлинга работало вполне себе, а с CPU 4x slowdown пошли фризы.


      1. indestructable
        11.12.2017 13:41

        Не вижу там shouldComponentUpdate. Мне кажется, можно было бы реализовать его и проверять, изменились ли данные, возвращаемые monitor (а предыдущие сохранять в стейте, например).


        Или, как вариант, возвращать true из shouldComponentUpdate каждые 16мс.
        forceUpdate, как по мне, не выглядит правильным решением.


        1. dr_zorge Автор
          11.12.2017 13:42

          Он в «Листинг shouldComponentUpdate»


  1. vital_pavlenko
    13.12.2017 16:04

    Спасибо! Можно забыть про jQuery UI Draggable и Droppable