preface
The project is based on the react-Grid-Layout and ANTD grid layout implementation. It may not be suitable for all projects, but the idea can be considered.
Demo Address: User-defined layout
Source address: source code
Realize the function
By visually dragging and dropping blocks, composite blocks generate custom layout code that can be imported directly into antD projects.
The layout code generated after the above visualization:
<Row>
<Col span={10} className="drag-item">
<Col className="drag-subitem" style={{ height: '10vh' }} span={24}></Col>
<Col className="drag-subitem" style={{ height: '95vh' }} span={24}></Col>
</Col>
<Col span={14} className="drag-item">
<Col className="drag-subitem" style={{ height: '25vh' }} span={24}></Col>
<Col className="drag-subitem" style={{ height: '80vh' }} span={24}></Col>
</Col>
</Row>
Copy the code
Code generated pages:
Implementation approach
To generate arbitrary layouts, a grid layout is the most appropriate.
Irregular layouts in special cases are not considered here.
We started by dividing the layout into columns, with several blocks within each column, which basically satisfied most of our business scenarios.
Simple enough to say, but not so simple when combined with visual drag and drop.
We drag and drop the width, height and position of each block to determine where our block should be placed.
We then combine this with ANTD’s raster layout API to generate the layout code we want.
Those of you who have used ANTD for grid layout should know that Row controls Row layout and Col controls column layout. Here’s my idea:
The entire page is a Row, then divided into columns based on the drag-and-drop information, each containing a column that covers the entire width of the column, so that it can be stacked down.
Col controls the width via the SPAN property, and the height is added directly via the style property.
implementation
Implement visual drag and drop
The react-grid-Layout library is very powerful.
Git address: react-grid-layout
We only need to provide simple basic information to generate drag-and-drop scalable blocks. I won’t say much about the API here, but I will implement it.
const dragItem = { x: 14, y: 14, w: 1, h: 1, maxH: 20 }
<GridLayout className="layout"
onResizeStop={onResizeStop}
onDragStop={onDragStop}
preventCollision={true}
margin={[0, 0]}
cols={12}
rowHeight={30}
width={800}
style={{ height: 600 }}
>
{
Object.keys(dragObj).map(item => <div className="drag-item" style={{ backgroundColor: '#' + ('00000' + (Math.random() * 0x1000000 << 0).toString(16)).slice(-6) }} key={item} data-grid={dragObj[item]}></div>)
}
</GridLayout>
Copy the code
Data-driven view, we can add blocks to the view by adding block events to change the dragObj in the code.
When we change the position or size of the block in the view, we need to save the current block information for our subsequent generation of layout code.
The onResizeStop and onDragStop events in the code above help us listen for our drag end and block size changes.
Both events do the same thing, recording the current block state. Here’s the event code:
const onResizeStop = (arr) = > {
let obj = {}, layoutObj = {}
arr.forEach(item= > {
obj[item['i']] = { x: item['x'].y: item['y'].w: item['w'].h: item['h'].maxH: item['maxH']}if (layoutObj[item['x']]) {
layoutObj[item['x']].push(item)
} else {
layoutObj[item['x']] = [item]
}
})
setLayoutObj(layoutObj)
setDragObj(obj)
}
Copy the code
Both events give us an array parameter containing all the block information, which we parse into a data structure we can use and store in State.
Parsing object generation code
After we have created the data structure we want by dragging and dropping, to generate the page we want, we need to know how the location and width and height of the blocks map into our layout.
Here I pass the convention: drag area size is 800 wide, 600 high. Basic division of each block:
Divide the height into 20 equal parts and the width into 12 equal parts.
Like this:
One unit of h corresponds to a height of 0.5vh;
1 unit of W corresponds to 2 units of span (Col span of ANTD is 24 equal)
With the mapping, we simply transform the object that stores the block information into a code string:
const onCodeGenerator = (a)= > {
setCodeStr(
'<Row>\n'
+
Object.keys(layoutObj).map(item= > `
<Col span={${layoutObj[item][0].w * 2}} className="drag-item">
${layoutObj[item].map(subItem => `
<Col className="drag-subitem" style={{height:'${subItem.h * 5}vh'}} span={24}></Col>\n`).join(' ')}
</Col>\n`
).join(' ')
+
'</Row>')}Copy the code
By modifying the code above, we can get the code string we want, copy and paste it into our project to get a proper layout.
The latter
This is a tool. I developed it on the basis of the current business of the company, so it may not be very applicable to everyone. Here are my thoughts for future reference if there are similar needs.