react-grid-layout核心功能实现
2021 M07 22
前言
react-grid-layout是基于react的网格布局系统,支持视图的拖拽和缩放,操作十分灵活。
在线体验。
工作中某个项目模块实现用到了react-grid-layout,就去看了一下核心功能的实现。
实际上,这篇文章也是内部串讲的一部分,有时间会单独分享一下做串讲的经验。
不得不说,作者的思维很巧妙,一阵连环套娃。
今天我们就来看一下这个库的核心功能实现,包括网格布局计算、拖动、缩放。
东西比较多,可选读。
整体结构图和核心功能实现原理如下:
基本使用
可以看到,只需要传递一个带有布局信息的layout数组即可
import React from 'react';
import GridLayout from 'react-grid-layout';
export default class App extends React.PureComponent {
render() {
// layout is an array of objects
// static 表示不可拖动和缩放
// key是必须的
const layout = [
{ i: 'a', x: 0, y: 1, w: 1, h: 1, static: true },
{ i: 'b', x: 1, y: 0, w: 3, h: 2 },
{ i: 'c', x: 4, y: 0, w: 1, h: 2 },
];
return (
<GridLayout layout={layout} width={1200}>
<div key="a">a</div>
<div key="b">b</div>
<div key="c">c</div>
</GridLayout>
);
}
}
网格布局
接下来进入react-grid-layout最为关键的部分,网格布局生成和计算。 简单来说就是根据用户给定的layout,计算出带有px的具体样式,最终展现在页面上。 我们直接看源码中入口组件ReactGridLayout中的render函数:
render() {
const { className, style, isDroppable, innerRef } = this.props;
// 合并类名
const mergedClassName = classNames(layoutClassName, className);
// 合并style
const mergedStyle = {
height: this.containerHeight(),// 计算容器高度
...style,
};
// 绑定drag和drop事件,其中noop是一个空函数
// export const noop = () => {};
return (
<div
ref={innerRef}
className={mergedClassName}
style={mergedStyle}
// 拖拽相关的一些回调,如果业务场景不需要,可以不设置
// 默认isDroppable是false
onDrop={isDroppable ? this.onDrop : noop}
onDragLeave={isDroppable ? this.onDragLeave : noop}
onDragEnter={isDroppable ? this.onDragEnter : noop}
onDragOver={isDroppable ? this.onDragOver : noop}
>
// 渲染节点
{React.Children.map(
this.props.children,
child => this.processGridItem(child)
)}
// 暂且可忽略,默认isDroppable 是false
{isDroppable &&
this.state.droppingDOMNode &&
this.processGridItem(this.state.droppingDOMNode, true)}
// 在拖拽时候展示操纵蒙版
{this.placeholder()}
</div>
);
}
render中做了三件关键的事:
- 合并样式和类名
- 绑定拖拽事件
- 渲染Children
渲染Children
我们先来看渲染Children这部分,函数processGridItem内部用GridItem组件对传入的react元素做了一层包裹后返回。其中GridItem是网格单元的展示组件,它接收布局,拖动,缩放等相关props。关于GridItem更多细节,下面会提到。
processGridItem(
child: ReactElement<any>,
isDroppingItem?: boolean
): ?ReactElement<any> {
// 这里也有一个小细节,如果传入的child没有key,会被return掉,不会在页面上展示。
if (!child || !child.key) return;
// 布局相关
const l = getLayoutItem(this.state.layout, String(child.key));
if (!l) return null;
// xxx...
return (
<GridItem
//... 布局 拖动 缩放 相关props
>
{child}
</GridItem>
);
}
接下来,我们看下布局和相关的东西。上述getLayoutItem函数接收一个来自内部state的参数layout。
state = {
activeDrag: null,
layout: synchronizeLayoutWithChildren(
this.props.layout,// 包含布局信息的数组对象
this.props.children,// react元素
this.props.cols,// 布局列数 默认12
// 控制水平/垂直布局
compactType(this.props)
),
mounted: false,
oldDragItem: null,
oldLayout: null,
oldResizeItem: null,
droppingDOMNode: null,
children: []
};
state中对layout做了一个处理,涉及到了函数synchronizeLayoutWithChildren。
synchronizeLayoutWithChildren
该函数见名知义,用于同步layout和children,为每个child生成一个网格布局单元。对于已有布局(传入的layout中每项的i和child的key匹配上),直接使用。如果没有layout参数,看child上是否有_grid和data-grid属性,有的话就使用,效果和layout参数一致。如果上述提到的布局相关的参数都没有,会创建一个默认布局,添加到已有布局的下方。
function synchronizeLayoutWithChildren(
initialLayout: Layout,
children: ReactChildren,
cols: number,
compactType: CompactType
): Layout {
initialLayout = initialLayout || [];
const layout: LayoutItem[] = [];
React.Children.forEach(children, (child: ReactElement<any>, i: number) => {
// 已有布局直接复用,其实就是一个find操作
const exists = getLayoutItem(initialLayout, String(child.key));
if (exists) {
layout[i] = cloneLayoutItem(exists);
} else {
if (!isProduction && child.props._grid) {
// _grid的废弃警告,建议使用layout或者data-grid传递布局信息
// xxx..
}
const g = child.props["data-grid"] || child.props._grid;
// 如果child有data-grid或者_grid属性直接使用
if (g) {
if (!isProduction) {
validateLayout([g], "ReactGridLayout.children");
}
layout[i] = cloneLayoutItem({ ...g, i: child.key });
} else {
//创建一个默认布局
layout[i] = cloneLayoutItem({
w: 1,
h: 1,
x: 0,
y: bottom(layout),
i: String(child.key)
});
}
}
});
// 边界处理/防堆叠
const correctedLayout = correctBounds(layout, { cols: cols });
// 空间压缩
return compact(correctedLayout, compactType, cols);
}
props传递进来的layout,或者人为拖动/缩放的布局,都有可能发生一些小冲突,比如堆叠,越界。 所以在最后需要对布局进行一些额外处理:如越界修正,防堆叠,压缩额外空间使布局紧凑。
correctBounds
边界控制函数,对于给定的布局,确保每一个都在其边界限制内。 如果是右侧越界,新的x坐标=布局列数-列宽。 如果是左侧越界,新的x坐标为0,列宽= 布局列数。
//cols 网格列数 默认12
function correctBounds(layout: Layout, bounds: { cols: number }): Layout {
// 获取静态item ,static =true
const collidesWith = getStatics(layout);
for (let i = 0, len = layout.length; i < len; i++) {
const l = layout[i];
// 右侧溢出处理
if (l.x + l.w > bounds.cols) {
l.x = bounds.cols - l.w;
}
// 左侧溢出处理
if (l.x < 0) {
l.x = 0;
l.w = bounds.cols;
}
if (!l.static) {
collidesWith.push(l);
} else {
// 如果静态元素碰撞,首项下移,避免堆叠
while (getFirstCollision(collidesWith, l)) {
l.y++;
}
}
}
return layout;
}
function getFirstCollision(
layout: Layout,
layoutItem: LayoutItem
): ?LayoutItem {
for (let i = 0, len = layout.length; i < len; i++) {
if (collides(layout[i], layoutItem)) return layout[i];
}
}
碰撞检测函数
function collides(l1, l2){
if (l1.i === l2.i) return false; // same element
if (l1.x + l1.w <= l2.x) return false; // l1 is left of l2
if (l1.x >= l2.x + l2.w) return false; // l1 is right of l2
if (l1.y + l1.h <= l2.y) return false; // l1 is above l2
if (l1.y >= l2.y + l2.h) return false; // l1 is below l2
return true; // boxes overlap
}
compact
该函数用于对布局空间进行压缩,使布局更紧凑。
function compact(layout, compactType, cols) {
// 获取静态布局 static =true
const compareWith = getStatics(layout);
// 根据传入的压缩方式进行排序
// 水平或者垂直 'horizontal' | 'vertical';
const sorted = sortLayoutItems(layout, compactType);
// 用于放置新布局的数组
const out = Array(layout.length);
for (let i = 0, len = sorted.length; i < len; i++) {
let l = cloneLayoutItem(sorted[i]);
// 不会移动静态元素
if (!l.static) {
// 压缩空间
l = compactItem(compareWith, l, compactType, cols, sorted);
compareWith.push(l);
}
// Add to output array
// to make sure they still come out in the right order.
out[layout.indexOf(sorted[i])] = l;
// Clear moved flag, if it exists.
l.moved = false;
}
return out;
}
// 压缩处理函数
function compactItem(
compareWith: Layout,
l: LayoutItem,
compactType: CompactType,
cols: number,
fullLayout: Layout
): LayoutItem {
const compactV = compactType === "vertical";
const compactH = compactType === "horizontal";
if (compactV) {
// 垂直方向不发生碰撞情况下 压缩y坐标
l.y = Math.min(bottom(compareWith), l.y);
while (l.y > 0 && !getFirstCollision(compareWith, l)) {
l.y--;
}
} else if (compactH) {
// 水平方向不发生碰撞情况下 压缩x坐标
while (l.x > 0 && !getFirstCollision(compareWith, l)) {
l.x--;
}
}
// 发生碰撞就下移或者左移
let collides;
while ((collides = getFirstCollision(compareWith, l))) {
if (compactH) {
resolveCompactionCollision(fullLayout, l, collides.x + collides.w, "x");
} else {
resolveCompactionCollision(fullLayout, l, collides.y + collides.h, "y");
}
// 控制水平方向上的无限增长.
if (compactH && l.x + l.w > cols) {
l.x = cols - l.w;
l.y++;
}
}
// 对上述的y--,x--做容错处理,确保没有负值
l.y = Math.max(l.y, 0);
l.x = Math.max(l.x, 0);
return l;
}
经过correntBounds和compact函数处理,就会生成一个紧凑,无溢出,无堆叠效果的网格布局单元。
容器高度计算
说完了布局生成,再来看一下入口组件render函数中对类名和样式的处理。类名合并上没什么特别的,直接使用classnames进行合并。
// classnames 基本使用
var classNames = require('classnames');
classNames('foo', 'bar'); // => 'foo bar'
// react-grid-layout中使用
const { className, style, isDroppable, innerRef } = this.props;
// 合并类名
const mergedClassName = classNames(layoutClassName, className);
样式合并涉及到了一个用于计算容器高度的函数containerHeight,这里还是有一些值得说的点。一个容器的高度至少要容纳最高占位布局(高度h和位置y),所以需要从给定的布局中找出h+y最大的那一项,作为容器基准高度。如下图所示,为便于观察,每一个布局项高度h都是1,最大的y轴坐标为2,容器基准高度就是3. 但是完整的高度不仅仅是基准高度,还涉及到grid-item之间的margin,容器纵向padding。
containerHeight() {
// 默认autoSize是true
if (!this.props.autoSize) {
return;
}
// 获取底部坐标
// 这里的layout是经过修正的,不同于this.props.layout
const nbRow = bottom(this.state.layout);
const containerPaddingY = this.props.containerPadding
? this.props.containerPadding[1]
: this.props.margin[1];
// 计算成具体的px
// rowHeight默认150 margin默认[10,10]
return `
${nbRow * this.props.rowHeight +
(nbRow - 1) * this.props.margin[1] +
containerPaddingY * 2 }
px`;
}
// 获取布局中y+h的最大值
function bottom(layout: Layout): number {
let max = 0;
let bottomY;
for (let i = 0, len = layout.length; i < len; i++) {
bottomY = layout[i].y + layout[i].h;
if (bottomY > max) {
max = bottomY;
}
}
return max;
}
上述布局计算结果:30(rowHeight)*3(基准高度)+20(2个margin)+20(上下容器padding)=130px。值得注意的是: 计算容器高度的时候,基准高度指的是经过compact函数压缩后的坐标值。来看一个具体的高度计算案例:
export default class App extends React.PureComponent {
render() {
const layout = [
{ i: 'a', x: 0, y: 100, w: 1, h: 1 },
];
return (
<div style={{ width: 600, border: '1px solid #ccc', margin: 10 }}>
<GridLayout layout={layout} width={600}>
<div key="a">a</div>
</GridLayout>
</div>
);
}
}
在containerHeight内部打印一下,会发现y并不是传入的100,而是被compact压缩后的0。如此一来,容器的基准高度就是 h+y=1+0=1。容器高度= 150(rowHeight)*1(基准高度)+0(margin)+20(上下容器padding)=170px。
GridItem
上述是容器布局计算,网格单元的计算是在GridItem组件组件进行的。 该组件接受的props比较多,大致分为布局,拖动,缩放这三类。
processGridItem(child: any, isDroppingItem?: boolean): any {
if (!child || !child.key) {
return;
}
const l = getLayoutItem(this.state.layout, String(child.key));
if (!l) {
return null;
}
const {
width,// 容器宽度
cols, // 布局列数 默认12
margin, // Margin between items [x, y] in px
containerPadding, // Padding inside the container [x, y] in px
rowHeight, // 单个grid-item高度
maxRows,// 最大行数 默认无限 表现为infinite vertical growth
isDraggable, // 是否可拖动 默认true
isResizable, // 是否可缩放 默认true
isBounded, // 控制是否在容器限制内移动 默认false
useCSSTransforms,//默认为true,开启后使用transforms替代left/top,绘制性能提高6倍
transformScale, // 比例系数 默认1 transform: scale(n)
draggableCancel, // 取消拖动手柄 css类名选择器
draggableHandle,// 拖动手柄 css类名选择器
resizeHandles,// 缩放方位 默认se 右下角
resizeHandle, // 缩放手柄
} = this.props;
const { mounted, droppingPosition } = this.state;
// 判断是否可拖动/缩放
const draggable = typeof l.isDraggable === 'boolean' ?
l.isDraggable :
!l.static && isDraggable;
const resizable = typeof l.isResizable === 'boolean' ?
l.isResizable :
!l.static && isResizable;
// 判断缩放方向 默认se
const resizeHandlesOptions = l.resizeHandles || resizeHandles;
// 判断是否限制在容器内移动
const bounded = draggable && isBounded && l.isBounded !== false;
return (
<GridItem
containerWidth={width}
cols={cols}
margin={margin}
containerPadding={containerPadding || margin}
maxRows={maxRows}
rowHeight={rowHeight}
cancel={draggableCancel}
handle={draggableHandle}
onDragStop={this.onDragStop}
onDragStart={this.onDragStart}
onDrag={this.onDrag}
onResizeStart={this.onResizeStart}
onResize={this.onResize}
onResizeStop={this.onResizeStop}
isDraggable={draggable}
isResizable={resizable}
isBounded={bounded}
useCSSTransforms={useCSSTransforms && mounted}
usePercentages={!mounted}
transformScale={transformScale}
w={l.w}
h={l.h}
x={l.x}
y={l.y}
i={l.i}
minH={l.minH}
minW={l.minW}
maxH={l.maxH}
maxW={l.maxW}
static={l.static}
droppingPosition={isDroppingItem ? droppingPosition : undefined}
resizeHandles={resizeHandlesOptions}
resizeHandle={resizeHandle}
>
{child}
</GridItem>
);
}
Render
接下来,我们看一下这个组件的render函数具体做了些什么。
render() {
const {
x, y, w, h,
isDraggable,
isResizable,
droppingPosition,
useCSSTransforms
} = this.props;
// 位置计算,触发拖动和缩放时候也会重新计算
const pos =calcGridItemPosition(
this.getPositionParams(),
x, y, w, h,
this.state
);
// 获取 child
const child= React.Children.only(this.props.children);
// 修改child的类名和样式
let newChild = React.cloneElement(child, {
ref: this.elementRef,
// 修改类名
className: classNames(
'react-grid-item',
child.props.className,
this.props.className, {
static: this.props.static,
resizing: Boolean(this.state.resizing),
'react-draggable': isDraggable,
'react-draggable-dragging': Boolean(this.state.dragging),
dropping: Boolean(droppingPosition),
cssTransforms: useCSSTransforms,
}),
// 修改样式
// 真正将网格单元w,h,x,y换成带有px的具体尺寸
style: {
...this.props.style,
...child.props.style,
...this.createStyle(pos),
},
});
// 添加缩放支持
newChild = this.mixinResizable(newChild, pos, isResizable);
// 添加拖动支持
newChild = this.mixinDraggable(newChild, isDraggable);
return newChild;
}
getPositionParams(props: Props = this.props): PositionParams {
return {
cols: props.cols,
containerPadding: props.containerPadding,
containerWidth: props.containerWidth,
margin: props.margin,
maxRows: props.maxRows,
rowHeight: props.rowHeight
};
}
calcGridItemPosition
该函数接收布局相关参数,经过一系列计算,返回最终的计算结果。 给定参数如下:
{ i: 'a', x: 0, y: 0, w: 2, h: 1, } 容器宽度600,
网格间margin10,
容器paadding10,
列数cols12
计算原理
列宽的计算和之前算高度是类似的,也要考虑网格间的margin和容器的padding(左右)。 列宽colWidth = (containerWidth - margin[0] (cols - 1) - containerPadding[0] 2) / cols 以上述布局为例,计算出来的列宽经过四舍五入后是39,但这个是基于布局单元计算的。 如果gridItem正在缩放,就采用缩放时state记录的宽高(width,height)。 如果gridItem正在拖拽,就采用拖拽时state记录的位置(left,top)。 注意:react-grid-layout里margin存储的是[x,y]形式,与css中margin设置两个值时候效果是相反的。
function calcGridItemPosition(
positionParams,
x,
y,
w,
h,
state
){
const { margin, containerPadding, rowHeight } = positionParams;
// 计算列宽
const colWidth = calcGridColWidth(positionParams);
const out = {};
// 如果gridItem正在缩放,就采用缩放时state记录的宽高(width,height)。
// 通过回调函数获取布局信息
if (state && state.resizing) {
out.width = Math.round(state.resizing.width);
out.height = Math.round(state.resizing.height);
}
// 反之,基于网格单元计算
else {
out.width = calcGridItemWHPx(w, colWidth, margin[0]);
out.height = calcGridItemWHPx(h, rowHeight, margin[1]);
}
// 如果gridItem正在拖拽,就采用拖拽时state记录的位置(left,top)
// 通过回调函数获取布局信息
if (state && state.dragging) {
out.top = Math.round(state.dragging.top);
out.left = Math.round(state.dragging.left);
}
// 反之,基于网格单元计算
else {
out.top = Math.round((rowHeight + margin[1]) * y + containerPadding[1]);
out.left = Math.round((colWidth + margin[0]) * x + containerPadding[0]);
}
return out;
}
// 计算列宽
function calcGridColWidth(positionParams: PositionParams): number {
const { margin, containerPadding, containerWidth, cols } = positionParams;
return (
(containerWidth - margin[0] * (cols - 1) - containerPadding[0] * 2) / cols
);
}
// gridUnits 网格布局基准单元
function calcGridItemWHPx(gridUnits, colOrRowSize, marginPx){
// 0 * Infinity === NaN, which causes problems with resize contraints
if (!Number.isFinite(gridUnits)) return gridUnits;
return Math.round(
colOrRowSize * gridUnits + Math.max(0, gridUnits - 1) * marginPx);
}
createStyle
说完了布局宽高和位置计算,再来看一下对样式的处理。 gridItem样式合并中用到了函数createStyle,可以将计算好的布局转成带px的css样式。
createStyle(pos) {
const { usePercentages, containerWidth, useCSSTransforms } = this.props;
let style;
// 支持 CSS Transforms 默认
// 直接跳过布局和绘制,且不占用主线程资源,比较快
if (useCSSTransforms) {
style = setTransform(pos);
} else {
// 使用 top,left 展示,会比较慢
style = setTopLeft(pos);
// 服务端渲染相关
if (usePercentages) {
style.left = perc(pos.left / containerWidth);
style.width = perc(pos.width / containerWidth);
}
}
return style;
}
// 采用 translate 形式 并添加兼容处理和单位px
function setTransform({ top, left, width, height }) {
const translate = `translate(${left}px,${top}px)`;
return {
transform: translate,
WebkitTransform: translate,
MozTransform: translate,
msTransform: translate,
OTransform: translate,
width: `${width}px`,
height: `${height}px`,
position: "absolute"
};
}
// 采用 left top 形式 并添加单位px
function setTopLeft({ top, left, width, height } {
return {
top: `${top}px`,
left: `${left}px`,
width: `${width}px`,
height: `${height}px`,
position: "absolute"
};
}
拖拽和缩放
mixinDraggable
mixinDraggable函数为child添加拖动支持,实现上依赖react-draggable。
拖拽原理
在react-draggable库的DraggableCore组件内部,触发相应拖拽事件时会生成一些有用的信息,比如坐标,当前节点。这个信息会被封装成对象,作为参数传递给外部对应的回调函数。这样一来,外部回调就可以从这个对象中获取有用信息,重新setState,将dragging的值设置为新的{left,top}。然后这个值会经过函数calcGridItemPosition和createStyle处理,作为css样式附加在child上,从而实现拖拽。
import { DraggableCore } from 'react-draggable';
function mixinDraggable(child, isDraggable) {
// 下面这些拖拽相关的回调函数用于接收额外的位置信息,计算布局
return (
<DraggableCore
disabled={!isDraggable}
onStart={this.onDragStart}
onDrag={this.onDrag}
onStop={this.onDragStop}
handle={this.props.handle}
cancel={`.react-resizable-handle${ this.props.cancel ?
`,${ this.props.cancel}` : ''}`}
scale={this.props.transformScale}
nodeRef={this.elementRef}
>
{child}
</DraggableCore>
);
}
DraggableCore
在react-grid-layout中,不论是mixinDraggable还是mixinResizable都会依赖组件DraggableCore。这是因为拖动和缩放都会涉及相同的鼠标事件(暂不考虑触摸事件),对此,该组件也封装了相应的事件处理函数函数。在这三个函数内部,会调用props中传入的回调函数onStart,onDrag,onStop。
- handleDragStart 拖拽开始:记录拖拽的初始位置
- handleDrag 拖拽中:监听拖拽的距离和方向,并移动真实 dom
- handleDragStop 拖拽结束:取消拖拽中的事件监听
render() {
return React.cloneElement(React.Children.only(this.props.children), {
onMouseDown: this.onMouseDown,
onMouseUp: this.onMouseUp,
// xxx..触摸相关事件
});
}
// dragEventFor 是一个用于标识触发事件类型的全局变量 鼠标 or 触摸
onMouseDown = (e) => {
// 鼠标相关事件
dragEventFor ={
start: 'mousedown',
move: 'mousemove',
stop: 'mouseup'
}
return this.handleDragStart(e);
};
handleDragStart(){
//...
this.props.onStart()
}
handleDrag(){
//...
this.props.onDrag()
}
handleDragStop(){
//...
this.props.onStop()
}
接下来我们来逐一看下这几个事件处理函数的内部操作细节。
handleDragStart
handleDragStart = (e) => {
// 支持鼠标按下的回调函数
this.props.onMouseDown(e);
// Only accept left-clicks.
//xxx...
// 确保获取到document
// https://developer.mozilla.org/zh-CN/docs/Web/API/Node/ownerDocument
const thisNode = this.findDOMNode();
if (!thisNode ||
!thisNode.ownerDocument ||
!thisNode.ownerDocument.body) {
throw new Error('<DraggableCore> not mounted on DragStart!');
}
const {ownerDocument} = thisNode;
if (this.props.disabled ||
(!(e.target instanceof ownerDocument.defaultView.Node)) ||
(this.props.handle &&
!matchesSelectorAndParentsTo(e.target, this.props.handle, thisNode)) ||
(this.props.cancel &&
matchesSelectorAndParentsTo(e.target, this.props.cancel, thisNode))) {
return;
}
/**操作手柄示例
<!--实际上可以没有对应的css样式handle-->
<Draggable handle=".handle">
<div>
<div className="handle">Click me to drag</div>
<div>This is some other content</div>
</div>
</Draggable>*/
// 触摸相关操作 ...
// 非触摸设备,getControlPosition第二个函数为undefined
// 获取鼠标按下时候的坐标
const position = getControlPosition(e, undefined, this);
if (position == null) return;
const {x, y} = position;
// 包含节点自身,坐标和其他信息的对象
const coreEvent = createCoreData(this, x, y);
// 调用props传入的回调 onStart
const shouldUpdate = this.props.onStart(e, coreEvent);
if (shouldUpdate === false || this.mounted === false) return;
// 更新拖拽状态并存储偏移量
this.setState({
dragging: true,
lastX: x,
lastY: y
});
// 将move事件绑定在document上,扩大响应范围
// 这样即使移出当前griditem 依旧能保证事件得到响应。
// 可触摸设备和非可触摸设备结束拖拽时候的响应事件不同,这里需要用两个事件
addEvent(ownerDocument, dragEventFor.move, this.handleDrag);
addEvent(ownerDocument, dragEventFor.stop, this.handleDragStop);
};
handleDrag
在看完handleDragStart函数内部细节后,handleDrag和handleDragStop都会好理解些。 handleDrag主要做的事情是在拖动过程中不断更新位置信息。
handleDrag=(e) => {
// Get the current drag point from the event. This is used as the offset.
const position = getControlPosition(e, null, this);
if (position == null) return;
let {x, y} = position;
const coreEvent = createCoreData(this, x, y);
// Call event handler. If it returns explicit false, trigger end.
const shouldUpdate = this.props.onDrag(e, coreEvent);
if (shouldUpdate === false || this.mounted === false) {
try {
this.handleDragStop(new MouseEvent('mouseup'));
} catch (err) {
// Old browsers
//xxx... 旧浏览器的一些兼容处理
}
return;
}
this.setState({
lastX: x,
lastY: y
});
};
handleDropStop
拖拽结束,重置位置信息,删除绑定的事件处理函数。
handleDragStop= (e) => {
if (!this.state.dragging) return;
const position = getControlPosition(e, this.state.touchIdentifier, this);
if (position == null) return;
const {x, y} = position;
const coreEvent = createCoreData(this, x, y);
// Call event handler
const shouldContinue = this.props.onStop(e, coreEvent);
if (shouldContinue === false || this.mounted === false) return false;
const thisNode = this.findDOMNode();
// Reset the el.
this.setState({
dragging: false,
lastX: NaN,
lastY: NaN
});
if (thisNode) {
// Remove event handlers
removeEvent(thisNode.ownerDocument, dragEventFor.move, this.handleDrag);
removeEvent(thisNode.ownerDocument, dragEventFor.stop, this.handleDragStop);
}
};
mixinResizable
mixinResizable函数为child添加缩放支持,实现上依赖react-resizable。 react-resizable的实现又依赖了react-draggable。
缩放原理
缩放和拖拽底层依赖的是同一个库,这就注定了在功能实现上是类似的思路,都是借助回调函数。DraggableCore组件内部将包含位置信息的事件对象传递给外部回调函数,回调中会重新setState,将resizing的值设置为新的{width,height}。最后,获取到的新的width,height会通过css样式作用在grid-item上,从而实现缩放功能。
function mixinResizable(child,position,isResizable) {
const {
cols,
x,
minW,
minH,
maxW,
maxH,
transformScale,
resizeHandles,
resizeHandle
} = this.props;
const positionParams = this.getPositionParams();
// 最大宽度
const maxWidth = calcGridItemPosition(positionParams, 0, 0, cols - x, 0)
.width;
// 计算最小网格布局和最大网格布局和对应的容器大小
const mins = calcGridItemPosition(positionParams, 0, 0, minW, minH);
const maxes = calcGridItemPosition(positionParams, 0, 0, maxW, maxH);
const minConstraints = [mins.width, mins.height];
const maxConstraints = [
Math.min(maxes.width, maxWidth),
Math.min(maxes.height, Infinity)
];
return (
<Resizable
draggableOpts={{
disabled: !isResizable,
}}
className={isResizable ? undefined : "react-resizable-hide"}
width={position.width}
height={position.height}
minConstraints={minConstraints}
maxConstraints={maxConstraints}
onResizeStop={this.onResizeStop}
onResizeStart={this.onResizeStart}
onResize={this.onResize}
transformScale={transformScale}
resizeHandles={resizeHandles}
handle={resizeHandle}
>
{child}
</Resizable>
);
}
Resizable
resizable组件主要做了3件事:
- 传递resizable内部回调函数给DraggableCore组件,用于获取事件信息对象。
- 在resizable内部回调函数中将获取到的事件信息对象传递给外部回调,用于最终的样式更新,实际上是套了两层
- 渲染操控手柄
render() {
return cloneElement(children, {
...p,
className: `${className ? `${className} ` : ''}react-resizable`,
children: [
...[].concat(children.props.children),
// handleAxis 是一个存储操纵方位的数组
...resizeHandles.map((handleAxis) => {
// 挂载一个node节点 用于操控
const ref = this.handleRefs[handleAxis] ?
this.handleRefs[handleAxis] : React.createRef();
return (
<DraggableCore
{...draggableOpts}
nodeRef={ref}
key={`resizableHandle-${handleAxis}`}
onStop={this.resizeHandler('onResizeStop', handleAxis)}
onStart={this.resizeHandler('onResizeStart', handleAxis)}
onDrag={this.resizeHandler('onResize', handleAxis)}
>
// 渲染不同方位的操控手柄,默认右下角 se
{this.renderResizeHandle(handleAxis, ref)}
</DraggableCore>
);
})
]
});
}
通用事件函数封装
缩放的三个事件处理函数在外部只做简单触发,内部共用一套处理逻辑(onResizeHandler)。
// 停止缩放
onResizeStop: (Event, { node: HTMLElement, size: Position }) => void = (
e,
callbackData
) => {
this.onResizeHandler(e, callbackData, "onResizeStop");
};
// 开始缩放
onResizeStart: (Event, { node: HTMLElement, size: Position }) => void = (
e,
callbackData
) => {
this.onResizeHandler(e, callbackData, "onResizeStart");
};
// 缩放中
onResize: (Event, { node: HTMLElement, size: Position }) => void = (
e,
callbackData
) => {
this.onResizeHandler(e, callbackData, "onResize");
};
onResizeHandler
该函数用于计算缩放后重新生成的网格单元信息,并将变更后的宽和高存储到state的resizing上。
onResizeHandler(
e: Event,
{ node, size }: { node: HTMLElement, size: Position },
handlerName: string
): void {
//根据传入的handler名称获取对应的事件处理函数
const handler = this.props[handlerName];
if (!handler) return;
const { cols, x, y, i, maxH, minH } = this.props;
let { minW, maxW } = this.props;
// 根据宽高计算出网格单元w,h
// 因为缩放是会改变大小的,大小改变对应的网格单元也要变
let { w, h } = calcWH(
this.getPositionParams(),
size.width,
size.height,
x,
y
);
// 最小应该保持一个单元的布局
minW = Math.max(minW, 1);
//最大(cols - x)
maxW = Math.min(maxW, cols - x);
// 限制宽高在min max之间,可以等于min max
w = clamp(w, minW, maxW);
h = clamp(h, minH, maxH);
// 更新reszing 的值,和dragging作用类似,用于最终的样式计算
// 差异是这里边只会存储width/height
// dragging 中会存储left/top
this.setState({ resizing: handlerName === "onResizeStop" ? null : size });
handler.call(this, i, w, h, { e, node, size });
}
// 限制目标值在上下边界之间
function clamp(
num: number,
lowerBound: number,
upperBound: number
): number {
return Math.max(Math.min(num, upperBound), lowerBound);
}
resizeHandler
resizaHandle其实是起到一个中转站的作用,先从DraggableCore中获取节点和位置信息对象。 然后根据获取到的对象信息计算出缩放后的宽高,将其作为触发相应的回调的参数。
resizeHandler(handlerName: 'onResize' | 'onResizeStart' | 'onResizeStop', axis): Function {
return (e, { node, deltaX, deltaY }) => {
// Reset data in case it was left over somehow (should not be possible)
if (handlerName === 'onResizeStart') this.resetData();
// Axis restrictions
const canDragX = (this.props.axis === 'both' || this.props.axis === 'x') && axis !== 'n' && axis !== 's';
const canDragY = (this.props.axis === 'both' || this.props.axis === 'y') && axis !== 'e' && axis !== 'w';
// No dragging possible.
if (!canDragX && !canDragY) return;
// Decompose axis for later use
const axisV = axis[0];
const axisH = axis[axis.length - 1]; // intentionally not axis[1], so that this catches axis === 'w' for example
// Track the element being dragged to account for changes in position.
// If a handle's position is changed between callbacks, we need to factor this in to the next callback.
// Failure to do so will cause the element to "skip" when resized upwards or leftwards.
const handleRect = node.getBoundingClientRect();
if (this.lastHandleRect != null) {
// If the handle has repositioned on either axis since last render,
// we need to increase our callback values by this much.
// Only checking 'n', 'w' since resizing by 's', 'w' won't affect the overall position on page,
if (axisH === 'w') {
const deltaLeftSinceLast = handleRect.left - this.lastHandleRect.left;
deltaX += deltaLeftSinceLast;
}
if (axisV === 'n') {
const deltaTopSinceLast = handleRect.top - this.lastHandleRect.top;
deltaY += deltaTopSinceLast;
}
}
// Storage of last rect so we know how much it has really moved.
this.lastHandleRect = handleRect;
// Reverse delta if using top or left drag handles.
if (axisH === 'w') deltaX = -deltaX;
if (axisV === 'n') deltaY = -deltaY;
// 计算缩放后的宽和高
let width = this.props.width + (canDragX ? deltaX / this.props.transformScale : 0);
let height = this.props.height + (canDragY ? deltaY / this.props.transformScale : 0);
// Run user-provided constraints.
[width, height] = this.runConstraints(width, height);
const dimensionsChanged = width !== this.props.width || height !== this.props.height;
// Call user-supplied callback if present.
const cb = typeof this.props[handlerName] === 'function' ? this.props[handlerName] : null;
// Don't call 'onResize' if dimensions haven't changed.
const shouldSkipCb = handlerName === 'onResize' && !dimensionsChanged;
if (cb && !shouldSkipCb) {
e.persist?.();
cb(e, { node, size: { width, height }, handle: axis });
}
// Reset internal data
if (handlerName === 'onResizeStop') this.resetData();
};
}
再会
情如风雪无常,
却是一动即殇。
感谢你这么好看还来阅读我的文章,
我是冷月心,下期再见。