Seven days off, five days of rain. I am bored with my visual editor, so I will optimize it later
Introduction: What is the use of H5 visualization platform?
Xiaobai spent half of his working time in low code this year, and had some contact with both B and C terminals. Let’s talk about how I feel about the business and development side of Low Code.
For b-side projects
The main form of management is from the back. The main characteristics of background management projects are generally CRUD + CRUD. Although B-side projects are also built visually, it is more common for developers to write JSON as far as I know. Json is then handed over to a low-code framework to concatenate the data into components mapped to the corresponding technology stack for rendering.
Q: So how does it compare to traditional React + ANTD or vue+ Element?
A: Fast, fucking fast. If you are familiar with a low code framework related to MIS. When you’re done with a project and you’ve been fishing for two hours and you look at a kid’s shoe developed using a traditional technology stack, it’s probably not even halfway through, so to speak.
You might have a question at this point, since this thing is so powerful. Why do you think so few people use it?
In fact, everything has two sides. It’s great at one thing, and it can be terrible at another. It has the following disadvantages for mis-shaped low code
-
Customize the poor
-
Relying heavily on the framework itself, troubleshooting is difficult
It’s easy to find out if you’ve touched something about Low Code. Basically lowcode’s stuff all the components in it are strongly coupled to itself.
Such as
{ type:'dialog', config:{ ... }}Copy the code
For such a shema, the general LowCode framework just maps to the dialog component it specifies. We can not think of the use of traditional technology development, found that the component library is not suitable can immediately change a new, or secondary development or their own are relatively easy.
But this is not so much fun in the Lowcode stack.
For its arrangement mistake, I still have deep experience 😮💨. The first low code framework I came into contact with was The Amis of Baidu. At that time, amis documents were very confusing, and my project at that time had some customization. To troubleshoot, to reduce amis’s limitations in development. I was developing and ripping the source code. In the end, the project involved layer upon layer of agents… True vice vomit.
For c-side projects
For c-side more is the use of some editor platforms to generate H5 pages by drag and drop. Which business scenarios are used more often? I don’t know if you noticed. In fact, most of the landing promotion pages are more similar. And the development of these landing pages although there is no complexity, but also to consume a certain amount of manpower. At the same time, when these landing pages accumulate to a certain number, it is also more painful for some developers who have technical pursuits. As their jobs gradually shifted from being front-end engineers to CV engineers, their enthusiasm was greatly reduced.
So the visual construction came into being, and the users of the visual construction platform are generally non-technical personnel. The liberated CV became the developer of the editor component, and everyone started writing code happily again.
I. Basic functions of the project
First of all, before I introduce myself, I would be a FW 😠without UI. First please brothers ignore my ugly to explode page!!
Although the function is not very much, but for the traditional H5 visual construction project, the core function has been basically complete.
1. Support component drag and drop
2. Support component configuration
3. Support schme download
4. Support project preview
5. Supports schema import
Ii. Basic structure of the project
2.1 Some Theories
For the company, each BU children’s shoes may use a different technology stack. Or React, or Vue, or various applets.
So we should design something in advance before we start
- How can an editor support multi-platform code
- The basic architectural components of the thinking platform
- Think about how each submodule project exists
Q: Why is the editor made to support multiple platforms?
A: Editors do not support multiple platforms, so do we need to get an editor for each component of the technology stack? It’s obviously not reasonable
Q: So how?
A: The simplest implementation is for the editor to recognize only the schema information for all materials (components)
Q: How do you put it? The react or vue code is not allowed to be read by the editor. How do you implement presentation?
A: The first thing to understand is that the editor is absolutely central to a visual architecture platform. And it’s definitely not going to change that the editor will only write on one stack. For the above problem, we can completely separate the editor and preview into two projects. To realize the preview function at any time, we just need to wrap the preview project in ifame and put it in the editor
Q: What are the basic architectural components of the platform at this point?
A: First of all, it has been confirmed that the editor and preview have been split into two projects, so the materials area should be separated. The material acts as the producer of schma and components. The schema needs to be submitted to the editor for consumption, and the editor spits out the processed schema to the preview project. The preview project relies on the components produced by the material and the latest schema produced by the editor to complete the page rendering. Then a more complete process comes out.
2.2 my demo
The technology stack I used for the whole project was React, and the editor was no longer relevant to the material technology stack
Because there are multiple subprojects involved, I used LERNA.
- Super Tempalte materials, using the development mode of component library, rollup packaging
- Super show preview, this is relatively simple to directly use the SHELF created by CRA
- Super-editor, UMI a shuttle
- Super- Server some simple platform back-end code express
Three: basic implementation
3.1 Simply Design a Set of Schema Rules [Easy]
First and foremost, each component should have its own schema. This scheme is intended to consume the editor and preview projects separately. What does the editor and preview projects need?
Look at the editor
The middle area is the iframe, so you can leave it alone. Both sides need to be rendered according to the schema provided by the material
Here are some ideas for my own simple kit
The schema for the Button component and itself
{
"name": "button"."compId": "Button"."description": "Button assembly"."pic": "https://img12.360buyimg.com/ddimg/jfs/t1/206278/28/8822/54487/615539f5E4f4cb5ab/49773bdc89799e5c.png"."config": [{
"name": "bgcColor"."label": "Button color"."type": "string"."format": "color"
},
{
"name": "btnText"."label": "Button copy"."type": "string"."format": "text"},]."defaultConfig": {
"btnText": "This is a button."."bgcColor": "# 333333"}}Copy the code
import React from "react";
import "./index.less";
const Button = ({ defaultConfig }) = > {
const { bgcColor, btnText } = defaultConfig;
return (
<div className="super-btn" style={{ background:` ${bgcColor} `}} >
<button
onClick={()= > {
console.log(111);
}}
>
{btnText}
</button>
</div>
);
};
export default Button;
Copy the code
The name, compId, and description attributes should be ignored for now.
PIC: Thumbnail on the left side of the editor, because the thumbnail on the left side of the editor is for presentation only. So simply put a map on the good 😳
Config: indicates the configurable properties of this component
DefaultConfig: Default values for configured properties supported by this component
3.2 Editor renders the thumbnail display area of components on the left [Easy]
The editor starts by rendering all the component thumbnails (you can optimize the virtual list if there are many components), but there is a schema for each component in the material. Where do I do the integration?
I slipped up and changed all json files to JS files. Then a layer of integration was done when the component was exported… The editor then renders it directly
material
The editor
import { schameMap } from 'super-template/build/bundle'; Loop render...Copy the code
3.5 Design Preview project [Easy]
Q: Why design two main routes?
A: In the editor, we don’t know what components the user selects, the order of the components, and the configuration of the components, all of which are more immediate. That is, when the user completes the drag-and-drop operation in the editor, the preview project should complete the response immediately. The situation is much better online, where all information about the component is already fixed.
Therefore, if two routes are combined together, there will not only be excessive functions, problems and errors for online display, but also certain obstacles. And not as clear as pulling away.
Implement rendering of the schema
The core method, either for the editor or for the actual display of the two main routing pages, is to render the JSON information passed in.
First, let’s take a look at what json looks like in the end from my editor (ignore the clientHeight attribute, which is used to solve the drag container height problem).
{
"currentCacheCopm": [{"name": "button"."compId": "Button"."description": "Button assembly"."pic": "https://img12.360buyimg.com/ddimg/jfs/t1/206278/28/8822/54487/615539f5E4f4cb5ab/49773bdc89799e5c.png"."config": [{"name": "bgcColor"."label": "Button color"."type": "string"."format": "color"
},
{
"name": "btnText"."label": "Button copy"."type": "string"."format": "text"}]."defaultConfig": { "btnText": "This is a button."."bgcColor": "# 333333" },
"clientHeight": 100
},
{
"name": "dialog"."compId": "Dialog"."description": "Popover assembly"."pic": "https://img11.360buyimg.com/ddimg/jfs/t1/97204/11/18195/74905/61553bb8E9ba92a0d/8d59c5db08ccd759.png"."config": [{"name": "dialogText"."label": "Please fill in the popup copy."."type": "string"."format": "text"}]."defaultConfig": { "dialogText": "Default popbox text" },
"clientHeight": 100
},
{
"name": "button"."compId": "Button"."description": "Button assembly"."pic": "https://img12.360buyimg.com/ddimg/jfs/t1/206278/28/8822/54487/615539f5E4f4cb5ab/49773bdc89799e5c.png"."config": [{"name": "bgcColor"."label": "Button color"."type": "string"."format": "color"
},
{
"name": "btnText"."label": "Button copy"."type": "string"."format": "text"}]."defaultConfig": { "btnText": "This is a button."."bgcColor": "# 333333" },
"clientHeight": 100}}]Copy the code
So we can render it like this
import * as template from "super-template/build/bundle";
/** * Parse json for component rendering */
const renderJson = (schema) = > {
const { compId, defaultConfig } = schema;
const Comp = template[compId];
return <Comp defaultConfig={defaultConfig} />;
};
export default renderJson;
Copy the code
- For displaying routing pages online, it is rendered by Scheme
- For those that need to interact with the editor, drag and drop can be mentioned together
3.6 Implement drag and drop communication across IFame
So this is kind of a core point, but the first thing we want to do is drag and drop and we have two options
- Use the native H5 drag and drop API directly
- Find a wheel that fits
Don’t assume that using wheels is any easier than using native. Drag and drop is just a couple of events for native, but the compatibility is a bit disgusting. What if you don’t want to encapsulate it yourself?
Harm, do not know how to ask Google
There’s a nice drag wheel in sight, another gem from the react-DNdredux creator
However, reading the library’s documentation is much more difficult than looking at the native API. So I translated it as I read it, and I’ll probably post it later.
So I’m just going to go ahead and use it, so let’s go back to the page and sort of think about what we’re going to do
Drag starts with two targets: 1. Drag target 2. Place target
So it’s obvious here that the component thumbnail on the left of the editor is our drag target, and the ifame container in the middle is our drop target. Drag and drop is visual, but the real purpose of drag and drop is component communication.
So this looks a lot easier. First of all, drag is stateful. At the beginning of the drag behavior, iframe is given a placeholder label. At the end of the drag behavior, it determines whether the drag target is successfully placed on the placement target.
If yes, replace the placeholder tag with real component information; No, just kill the placeholder tags
There is also a small challenge for new players, how to implement the following features?
As if a sudden feeling train of thought a little blocked, but in fact is often gray simple
We started with the idea that the entire preview iframe area should be a placement target, which would have been a bit confusing. Because you can’t get a variable index.
Index, which index is following the drag behavior is changing and can follow the position?
The answer lies within each component area of the IFrame. Now let’s set each component in the IFrame as the drop target
A map to a
Let’s see if we can easily get index 2 at component 1 after inserting the placeholder. All we need to do at this point is change the component emission position of the entire IFrame region. That is, move the placeholder for index 1 after index 2
For example, the data source for the entire iframe is [1,2,3,4]
Splice (‘ placeholder ‘,1).splice(currentIndex,’ placeholder ‘). CurrentIndex is index 2 at component 1
Kill the placeholder element now, and then fill it back up under the current element.
By this time looks, already had some rudiment. But now I have some very bad news for you. React-dnd does not support cross-iframe drag yet.
That at that time may have irascible elder brother asked, do not support you above said so a pile said 🔨.
I…
I don’t know if anyone noticed what I said above: drag and drop is visual, but the real purpose of drag and drop is component communication. Even iframe is the same, perhaps a bit more cumbersome, with component communication replaced by communication between the parent page and iframe child pages.
For space reasons, this article will stop here and the next article will focus on the implementation of drag across iframe in detail.
3.7 Design and implementation of attribute Configuration area on the right side of the editor [Easy]
Property configuration area: first click on a component of iframe, and the editor will read the Config information in the schema of this component to render the Config form.
The target of the click here is within the IFrame, so there is also a small communication problem that we can ignore for now.
How to render the config form
Looking directly at the code, the following component accepts the selected component schema as props. Then get the Config in the schema and render the corresponding form items according to the format field of config
const EditorConfigForm: FC<EditorConfigFormProps> = ({ compSchema, currentCacheCopm, compActiveIndex, }) = > {
const { config, defaultConfig } = compSchema;
const onFinish = (values) = > {
// Find the index of the current component among all components
// Notify iframe to update component information
};
return (
<Form
name="basic"
wrapperCol={{ span: 16 }}
initialValues={{ remember: true }}
autoComplete="off"
initialValues={defaultConfig}
onFinish={onFinish}
// onFinishFailed={onFinishFailed}
>
{config.map(({ name, format, label }) => {
let SuperFormItem: any;
switch (format) {
case 'text':
SuperFormItem = SuperText;
break;
case 'color':
SuperFormItem = SuperColor;
break;
case 'seleter':
SuperFormItem = SuperSeleter;
break;
case 'upload':
SuperFormItem = SuperUpload;
break;
default:
break;
}
return (
<Form.Item key={name} label={label} name={name}>{/* SuperFormItem({defaultConfig: defaultConfig[name], value,})} */}<SuperFormItem defaultConfig={defaultConfig[name]} />
</Form.Item>
);
})}
<Form.Item wrapperCol={{ span: 12.offset: 6}} >
<Button type="primary" htmlType="submit">save</Button>
</Form.Item>
</Form>
);
};
Copy the code
Four: Write to the end
Because of the length, this article mainly talks about some ideas. But for this kind of project, in fact, the most important thing is the idea of it, and the idea of realizing it is basically no problem. If interested, see the source code: project address
At the same time, the components and materials of such projects generally put into production have a certain coincidence with the business, instead of the form of a button component in my example. So apart from a drag-and-drop platform designed solely for forms, refining components from a page perspective is not easy to control and does not make sense. Regardless of whether the users of our platform are technical or not, easy to use and easy to assemble pages is the key.
If you can help, thank Star
reference
react-dnd.github.io/react-dnd/
Juejin. Cn/user / 380836…