Skip to content

Commit

Permalink
Merge branch 'update-docs'
Browse files Browse the repository at this point in the history
  • Loading branch information
prevwong committed Jan 11, 2020
2 parents eb350aa + f731411 commit 30cf195
Show file tree
Hide file tree
Showing 4 changed files with 299 additions and 2 deletions.
34 changes: 34 additions & 0 deletions CONTRIBUTING.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
# Contributing
Thank you for considering to contribute to Craft.js!

From proofreading, translating, writing tutorials or blog posts, improving the documentation, submitting bug reports and feature requests, or writing code, there are many ways to contribute. Upon contribution as per the [all-contributors](https://allcontributors.org/) specification, your profile will be recognised in our README's contributors section.

To get started right away, have a look at our [project tracker](https://github.com/prevwong/craft.js/projects) to check out a list of things that we'd like to work on right now.

If you are interested in proposing a new feature or have found a bug that you'd like to fix, please file a new [issue](https://github.com/prevwong/craft.js/issues).


# Setup
1. Fork this repository and create your branch from `master`
```bash
git clone https://github.com/your-name/craft.js
cd craft.js
```

2. Install the dependencies and start the development server
```bash
> yarn install
> yarn dev
```

3. Here are some additional npm scripts that might be useful
```bash
> yarn clean # clean all build files from all packages in the monorepo
> yarn build # create production build for all craftjs packages
> yarn lint # run tests across the monorepo
```
4. Submit a [pull request](https://github.com/prevwong/craft.js/compare) to merge the changes from your fork :heart:

# License
By contributing, you agree that your contributions will be licensed under MIT License.

2 changes: 1 addition & 1 deletion packages/core/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
</div>

<div align="center" style={{d}}>
<img src="https://user-images.githubusercontent.com/16416929/71734437-f2aada00-2e86-11ea-95d6-f63236b021f4.gif"/>
<img src="https://user-images.githubusercontent.com/16416929/72202590-4d05f500-349c-11ea-9e43-1da1cb0c30e9.gif"/>
</div>

<p align="center">
Expand Down
263 changes: 263 additions & 0 deletions packages/docs/docs/dev.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,263 @@
---
id: dev
title: Craft.js - Build any page editor with React
---



You know a web application is fancy when it has a page editor. Without a doubt, a page editor helps boost user experience significantly. If you're a frontend developer and have been tasked to be build one before, then you know precisely the difficulties and challenges of building one.

Existing libraries such as Grape.js or react-page are great for a working out-of-the-box page editor solution. However, as soon as you need to customise the look and feel of the page editor itself, you will find yourself hacking in the library's code.


## Introducing Craft.js
Craft.js is a React framework to build any type of page editor. Instead of providing a working page editor implementation with a user interface, Craft.js provides an abstraction for you to implement your own page editor upon. It comes backed-in with an extensible drag-n-drop system which handles the way React elements should be rendered/updated, and a cohesive API to interact with the editor which you can additionally implement your own features on top of.

### TL;DR
- Design your own user interface for your page editor
- Write React components that end-user could edit
- Govern drag-and-drop conditions for your components
- Control how your components should be edited. From simple text fields to content editables and drag to resize; if you can do it in React, then you can do it with Craft.js.


## Editable React Components
Let's start with a simple Card component like this:

```jsx
const Card = ({title}) => {
return (
<div>
<h2>{title}</h2>
<div id="p-only">
<p>Hi</p>
</div>
</div>
)
}
```

First, to integrate it with with Craft.js' drag-and-drop system, we just need to do the following modifications:

```jsx
import {useNode} from "@craftjs/core";

const Card = ({title}) => {
const { connectors: { connect, drag } } = useNode();
return (
<div ref={dom => connect(drag(dom))}>
<h2>{title}</h2>
<div id="p-only">
<p>Hi</p>
</div>
</div>
)
}
```

What's happening here?
- We passed the `connect` connector to the root element of our component; this tells Craft.js that this element represents the `Card` component. Hence, the dimensions of the specified element will be taken into consideration during drag and drop events.
- Then, we also passed `drag` connector to the same root element; this adds the drag handlers to the DOM. If the component is rendered as the child of a `<Canvas />`, the user will be able to drag this element and it will move the entire Text component.


Next, we might want to be able to control the drag-n-drop rules of our Card component. For example, let's say we only want our Card component to be draggable as long as its `title` prop is not "No Drag". We can achieve this easily as such:
```jsx
const Card = () => {...}

Card.craft = {
rules: {
canDrag: (node) => node.data.props.title != "No Drag"
}
}
```

#### Droppable regions
Next, let's take a look at the `#p-only` element we specified in our Card component. What if we want this area to be droppable where only `p` can be dropped?

This is where the`<Canvas />` component provided by Craft.js becomes useful. It defines a droppable region where each of its immediate child is draggable.


```jsx
import {useNode, Canvas} from "@craftjs/core";

const Card = ({title}) => {
const { connectors: { connect, drag } } = useNode();
return (
<div ref={dom => connect(drag(dom))}>
<h2>{title}</h2>
<Canvas id="p-only">
<p>Hi</p>
</Canvas>
</div>
)
}
```
Your next question might be about how we control our newly created droppable region. The `<Canvas />` component accepts an `is` prop which can be either a HTML element or a React component (by default, it's a `div`). So, if we supply a React component, we can essentially achieve the same design flexibility as we have with our Card component:

```jsx
import {useNode, Canvas} from "@craftjs/core";

const Container = ({children}) => {
const { connectors: {connect} } = useNode();
return (
<div ref={dom => connect(dom)}>
{children}
</div>
)
}

Container.craft = {
rules: {
canMoveIn: (incomingNode) => incomingNode.data.type == "p"
}
}

const Card = ({title}) => {
const { connectors: { connect, drag } } = useNode();
return (
<div ref={dom => connect(drag(dom))}>
<h2>{title}</h2>
<Canvas id="p-only" is={Container}>
<p>Hi</p>
</Canvas>
</div>
)
}
```

Let's break this down a bit. Our `<Container />` component is being rendered as a droppable region. This means, all dropped elements will be rendered in the component's `children` prop. Next, we also specified the `connectors` just as we did previously. This time, we specified a `canMoveIn` rule where only elements with the `p` tag will be accepted into the component.

## Control editing behaviours
Of course, any page editor must allow the end-user to edit the elements that are rendered. With Craft.js, you are in control of this as well.

Let's make our `h2` element content editable so our users could visually edit the `title` prop:

```jsx
const Card = ({title}) => {
const { connectors: { connect, drag }, setProp } = useNode();
return (
<div ref={dom => connect(drag(dom))}>
<h2 contentEditable onKeyUp={e => setProp(e.target.innerText)}>{title}</h2>
...
</div>
)
}
```
> In a real application, you may want to consider using react-contenteditable
Obviously, we do not want the element to be content editable all the time; perhaps only when our Card component is actually clicked:

```jsx
const Card = ({title}) => {
const { connectors: { connect, drag }, setProp, isSelected } = useNode((node) => ({
isSelected: node.events.selected
}));

return (
<div ref={dom => connect(drag(dom))}>
<h2 contentEditable={isSelected} onKeyUp={e => setProp(e.target.innerText)}>{title}</h2>
...
</div>
)
}
```
What we are doing is essentially accessing Craft's internal state and retrieving information about the current instance of our Card component. In this case, we are retrieving the `selected` event state which returns `true` when the user clicks on the DOM specified by the `connect` connector of the Component.

## Your page editor, your user interface

Essentially, Craft.js exposes these few React components:
- `Editor` creates the Editor context and sets everything up
- `Frame` defines the editable area

As you may have noticed, none of these are UI components. Instead, they are intended to be plugged into any user interface you would like to implement for your page editor.

Let's take a look at a super simple interface that we could design:

```jsx
// pages/index.js
import React from 'react';

export default function App() {
return (
<>
<h2>My page editor</h2>
<Editor> {/* sets everything up */ }
<div>
<Frame> {/* This part is now editable */ }
<Canvas is="div" style={{background: "#eee"}}> { /* Creates a gray droppable div */ }
<h2>Drag me around</h2>
<p>I'm draggable too</h2>
<Canvas is="div"> {/* a div that is both droppable and draggable */}
<h2>Hi</h2> {/* a draggable h2 */}
</Canvas>
</Canvas>
</Frame>
</div>
</Editor>
</>
);
}
```
### Interacting with the editor
Your user interface will most likely need to display some information or perform certain editor-related actions.
Let's say we want to design a Toolbar component that does the following things:
- Tells us the the type of the currently selected element
- Has a button to save the current editor state
- Has another button to load the last saved editor state

```jsx
import React, {useState} from 'react';
export default function App() {
return (
<>
<h2>My page editor</h2>
<Editor>
<Toolbar /> {/* Add this */}
<div>
<Frame>
...
</Frame>
</div>
</Editor>
</>
);
}
const Toolbar = () => {
const [savedState, setSavedState] = useState();
const { selectedType, actions, query } = useEditor(state => {
const selectedId = state.events.selected;
return {
selectedType: selectedId && state.nodes[selectedId].data.type
}
});
return (
<div>
<h2>Currently selected: {selectedType} </h2>
<a onClick={() => {
const editorState = query.serialize();
setSavedState(editorState);
}}>Save checkpoint</a>
{
savedState ? (
<a onClick={() =>
actions.deserialize(editorState)
}>Load from checkpoint</a>
) : null
}
</div>
)
}
```

Let's break this down:
- First, we access the editor's state to retrieve the type of the currently selected element
- Then, when the "Save Checkpoint" button is clicked, we use the `serialize` query which tells the editor to return its state in a serialised JSON form. We then save the JSON output in our component's state
- Once we have the JSON, we display the "Load from checkpoint" button. When this button is clicked, we simply call the `deserialize` editor action which essentially returns the editor to the state stored in the JSON output.
## Closing words
This has been a high-level overview of Craft.js and we've only covered some very basic examples. We've seen how we could easily control almost every aspect of the page editor experience. Hopefully, this article has given you an idea on the possibilities of what you can do with Craft.js.
2 changes: 1 addition & 1 deletion packages/docs/docusaurus.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@ module.exports = {
},
{
label: 'Tutorial',
to: 'docs/basic-tutorial',
to: 'docs/guides/basic-tutorial',
},
{
label: 'API Reference',
Expand Down

0 comments on commit 30cf195

Please sign in to comment.