1 | <h1 align="center">
|
2 | Markdown Editor
|
3 | </h1>
|
4 |
|
5 | <p align="center">
|
6 |
|
7 | <a href="https://accordproject-markdown-editor.netlify.com/examples/">
|
8 | <img src="https://api.netlify.com/api/v1/badges/952fdc5d-a2bb-4895-a375-25ea1c6f30d8/deploy-status" alt="Netlify Status" />
|
9 | </a>
|
10 | <a href="https://travis-ci.org/accordproject/markdown-editor">
|
11 | <img src="https://travis-ci.org/accordproject/markdown-editor.svg?branch=master" alt="Build Status">
|
12 | </a>
|
13 |
|
14 | <a href="https://www.npmjs.com/package/@accordproject/markdown-editor">
|
15 | <img src="https://img.shields.io/npm/dm/@accordproject/markdown-editor" alt="npm version" />
|
16 | </a>
|
17 |
|
18 | <a href="https://badge.fury.io/js/%40accordproject%2Fmarkdown-editor">
|
19 | <img src="https://badge.fury.io/js/%40accordproject%2Fmarkdown-editor.svg" alt="downloads" />
|
20 | </a>
|
21 |
|
22 | <a href="https://github.com/accordproject/markdown-editor/blob/master/LICENSE">
|
23 | <img src="https://img.shields.io/github/license/accordproject/markdown-editor" alt="GitHub license">
|
24 | </a>
|
25 |
|
26 | <a href="https://accord-project-slack-signup.herokuapp.com/">
|
27 | <img src="https://img.shields.io/badge/Accord%20Project-Join%20Slack-blue" alt="Join the Accord Project Slack" />
|
28 | </a>
|
29 |
|
30 | </p>
|
31 |
|
32 | This repo contains two React-based editors:
|
33 | 1. A WYSIWYG [Slate][slate]-based editor that edits rich text and calls an `onChange`
|
34 | callback with the modified Slate DOM.
|
35 | 2. A TextArea-based markdown editor that edits markdown text and calls an `onChange`
|
36 | callback with the modified markdown text.
|
37 |
|
38 | The demo editor uses the `markdown-transform` package to transform Slate DOM
|
39 | to/from markdown text.
|
40 |
|
41 | Using these editors you could allow people to either edit rich formatted text using
|
42 | markdown (and provide a WYSIWYG preview), or allow them to edit using a WYSIWYG
|
43 | editor and use markdown for persistence.
|
44 |
|
45 | The editor includes a formatting toolbar.
|
46 |
|
47 | This component is Apache-2 licensed Open Source. Contributors welcome!
|
48 |
|
49 | ### [Demo](https://accordproject-markdown-editor.netlify.com/examples/)
|
50 |
|
51 | ### Usage
|
52 |
|
53 | ```
|
54 | npm install @accordproject/markdown-editor
|
55 | ```
|
56 |
|
57 | You'll also need to be sure to install this package's peer dependencies:
|
58 | ```
|
59 | npm install react react-dom slate slate-react styled-components
|
60 | ```
|
61 |
|
62 | ```
|
63 | import { SlateAsInputEditor } from '@accordproject/markdown-editor';
|
64 | import List from '@accordproject/markdown-editor/dist/plugins/list';
|
65 | import Video from '@accordproject/markdown-editor/dist/plugins/video';
|
66 | import { SlateTransformer } from '@accordproject/markdown-slate';
|
67 |
|
68 | const plugins = [List()];
|
69 | const slateTransformer = new SlateTransformer();
|
70 |
|
71 | function storeLocal(slateValue) {
|
72 | const markdown = slateTransformer.toMarkdown(slateValue);
|
73 | localStorage.setItem('markdown-editor', markdown);
|
74 | }
|
75 |
|
76 | ReactDOM.render(<SlateAsInputEditor plugins={plugins} lockText={false} onChange={storeLocal}/>
|
77 | , document.getElementById('root'));
|
78 | ```
|
79 |
|
80 | ### Example
|
81 |
|
82 | For an example React App see the `./examples/` folder.
|
83 |
|
84 | A `TextArea` containing [CommonMark][CommonMark] synchronized with a `MarkdownEditor` component, rendered using [Slate][slate].
|
85 |
|
86 | ![overview image](overview.png)
|
87 |
|
88 | In order to run an isolated local development example, run `npm run dev` and then navigate to: http://localhost:3001/examples
|
89 |
|
90 | ## Available Scripts
|
91 |
|
92 | In the project directory, you can run:
|
93 |
|
94 | #### `npm run dev`
|
95 |
|
96 | Runs the app in the development mode.<br>
|
97 | Open [http://localhost:3001/examples](http://localhost:3001/examples) to view it in the browser.
|
98 |
|
99 | The page will reload if you make edits.<br>
|
100 |
|
101 | #### `npm run build`
|
102 |
|
103 | Builds the app for production to the `build` folder.<br>
|
104 | It correctly bundles React in production mode and optimizes the build for the best performance.
|
105 |
|
106 | The build is minified and the filenames include the hashes.<br>
|
107 | Your app is ready to be deployed!
|
108 |
|
109 | ## Styling
|
110 |
|
111 | You can style the toolbar of this component, as well as the width of the editor:
|
112 |
|
113 | #### `editorProps`
|
114 |
|
115 | This is an object with the following possible css inputs as strings:
|
116 | - `BUTTON_BACKGROUND_INACTIVE`
|
117 | - `BUTTON_BACKGROUND_ACTIVE`
|
118 | - `BUTTON_SYMBOL_INACTIVE`
|
119 | - `BUTTON_SYMBOL_ACTIVE`
|
120 | - `DROPDOWN_COLOR`
|
121 | - `TOOLBAR_BACKGROUND`
|
122 | - `EDITOR_BORDER_RADIUS`
|
123 | - `EDITOR_SHADOW`
|
124 | - `EDITOR_BORDER`
|
125 | - `TOOLTIP_BACKGROUND`
|
126 | - `TOOLTIP`
|
127 | - `TOOLBAR_SHADOW`
|
128 | - `WIDTH`
|
129 |
|
130 | ---
|
131 |
|
132 | <p align="center">
|
133 | <a href="https://www.accordproject.org/">
|
134 | <img src="assets/APLogo.png" alt="Accord Project Logo" width="400" />
|
135 | </a>
|
136 | </p>
|
137 |
|
138 | Accord Project is an open source, non-profit, initiative working to transform contract management and contract automation by digitizing contracts. Accord Project operates under the umbrella of the [Linux Foundation][linuxfound]. The technical charter for the Accord Project can be found [here][charter].
|
139 |
|
140 | ## Learn More About Accord Project
|
141 |
|
142 | ### Overview
|
143 | * [Accord Project][apmain]
|
144 | * [Accord Project News][apnews]
|
145 | * [Accord Project Blog][apblog]
|
146 | * [Accord Project Slack][apslack]
|
147 | * [Accord Project Technical Documentation][apdoc]
|
148 | * [Accord Project GitHub][apgit]
|
149 |
|
150 |
|
151 | ### Documentation
|
152 | * [Getting Started with Accord Project][docwelcome]
|
153 | * [Concepts and High-level Architecture][dochighlevel]
|
154 | * [How to use the Cicero Templating System][doccicero]
|
155 | * [How to Author Accord Project Templates][docstudio]
|
156 | * [Ergo Language Guide][docergo]
|
157 |
|
158 | ### Ecosystem
|
159 |
|
160 |
|
161 | #### Core libraries:
|
162 | <table>
|
163 | <tr>
|
164 | <th headers="blank">Projects</th>
|
165 | <th headers="blank">Package name</th>
|
166 | <th headers="blank">Version</th>
|
167 | <th headers="blank">Description</th>
|
168 | </tr>
|
169 | <tr>
|
170 | <td headers><a href="https://github.com/accordproject/cicero">Cicero</a></td>
|
171 | <td headers> <a href="https://github.com/accordproject/cicero/tree/master/packages/cicero-core">cicero-core</a></td>
|
172 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fcicero-core"><img src="https://badge.fury.io/js/%40accordproject%2Fcicero-core.svg" alt="npm version"></a></td>
|
173 | <td headers>Templates Core</td>
|
174 | </tr>
|
175 | <tr>
|
176 | <td headers></td>
|
177 | <td headers> <a href="https://github.com/accordproject/cicero/tree/master/packages/cicero-cli">cicero-cli</a></td>
|
178 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fcicero-cli"><img src="https://badge.fury.io/js/%40accordproject%2Fcicero-cli.svg" alt="npm version"></a></td>
|
179 | <td headers> Cicero CLI </td>
|
180 | </tr>
|
181 | <tr>
|
182 | <td headers></td>
|
183 | <td headers> <a href="https://github.com/accordproject/cicero/tree/master/packages/cicero-engine">cicero-engine</a></td>
|
184 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fcicero-engine"><img src="https://badge.fury.io/js/%40accordproject%2Fcicero-engine.svg" alt="npm version"></a></td>
|
185 | <td headers>Node.js VM based implementation of Accord Protcol Template Specification execution</td>
|
186 | </tr>
|
187 | <tr>
|
188 | <td headers></td>
|
189 | <td headers> <a href="https://github.com/accordproject/cicero/tree/master/packages/cicero-server">cicero-server</a></td>
|
190 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fcicero-server"><img src="https://badge.fury.io/js/%40accordproject%2Fcicero-server.svg" alt="npm version"></a></td>
|
191 | <td headers>Wraps the Cicero Engine and exposes it as a RESTful service<td>
|
192 | </tr>
|
193 | <tr>
|
194 | <td headers></td>
|
195 | <td headers> <a href="https://github.com/accordproject/cicero/tree/master/packages/cicero-test">cicero-test</a></td>
|
196 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fcicero-test"><img src="https://badge.fury.io/js/%40accordproject%2Fcicero-test.svg" alt="npm version"></a></td>
|
197 | <td headers> Testing support for Cicero based on cucumber</td>
|
198 | </tr>
|
199 | <tr>
|
200 | <td headers></td>
|
201 | <td headers> <a href="https://github.com/accordproject/cicero/tree/master/packages/cicero-tools">cicero-tools</a></td>
|
202 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fcicero-tools"><img src="https://badge.fury.io/js/%40accordproject%2Fcicero-tools.svg" alt="npm version"></a></td>
|
203 | <td headers>Cicero Tools</td>
|
204 | </tr>
|
205 | <tr>
|
206 | <td headers="co1 c1"></td>
|
207 | <td headers="co2 c1"> <a href="https://github.com/accordproject/cicero/tree/master/packages/generator-cicero-template">generator-cicero-template</a></td>
|
208 | <td headers="co3 c1"> <a href="https://badge.fury.io/js/%40accordproject%2Fgenerator-cicero-template"><img src="https://badge.fury.io/js/%40accordproject%2Fgenerator-cicero-template.svg" alt="npm version"></a></td>
|
209 | <td headers="co4 c1">Code generator for a Cicero Template</td>
|
210 | </tr>
|
211 | <tr>
|
212 | <td headers><a href="https://github.com/accordproject/concerto">Concerto</a></td>
|
213 | <td headers><a href="https://github.com/accordproject/concerto/tree/master/packages/concerto-core">concerto-core</a></td>
|
214 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fconcerto-core"><img src="https://badge.fury.io/js/%40accordproject%2Fconcerto-core.svg" alt="npm version"></a></td>
|
215 | <td headers> Core Implementation for the Concerto Modeling Language</td>
|
216 | </tr>
|
217 | <tr>
|
218 | <td headers></td>
|
219 | <td headers><a href="https://github.com/accordproject/concerto/tree/master/packages/concerto-tools">concerto-tools</a></td>
|
220 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fconcerto-tools"><img src="https://badge.fury.io/js/%40accordproject%2Fconcerto-tools.svg" alt="npm version"></a></td>
|
221 | <td headers> Tools for the Concerto Modeling Language</td>
|
222 | </tr>
|
223 | <tr>
|
224 | <td headers></td>
|
225 | <td headers><a href="https://github.com/accordproject/concerto/tree/master/packages/concerto-cli">concerto-cli</a></td>
|
226 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fconcerto-cli"><img src="https://badge.fury.io/js/%40accordproject%2Fconcerto-cli.svg" alt="npm version"></a></td>
|
227 | <td headers>command-line interface for Concerto</td>
|
228 | </tr>
|
229 | <tr>
|
230 | <td headers><a href="https://github.com/accordproject/ergo">Ergo</a></td>
|
231 | <td headers><a href="https://github.com/accordproject/ergo/tree/master/packages/ergo-cli">ergo-cli</a></td>
|
232 | <td headers><a href="https://badge.fury.io/js/%40accordproject%2Fergo-cli"><img src="https://badge.fury.io/js/%40accordproject%2Fergo-cli.svg" alt="npm version"></a></td>
|
233 | <td headers>Ergo CLI</td>
|
234 | </tr>
|
235 | <tr>
|
236 | <th id="blank"></th>
|
237 | <td headers><a href="https://github.com/accordproject/ergo/tree/master/packages/ergo-compiler">ergo-compiler</a></td>
|
238 | <td headers><a href="https://badge.fury.io/js/%40accordproject%2Fergo-compiler"><img src="https://badge.fury.io/js/%40accordproject%2Fergo-compiler.svg" alt="npm version"></a></td>
|
239 | <td headers>Ergo compiler</td>
|
240 | </tr>
|
241 | <tr>
|
242 | <th id="blank"></th>
|
243 | <td headers><a href="https://github.com/accordproject/ergo/tree/master/packages/ergo-test">ergo-test</a></td>
|
244 | <td headers><a href="https://badge.fury.io/js/%40accordproject%2ergo-test"><img src="https://badge.fury.io/js/%40accordproject%2Fergo-test.svg" alt="npm version"></a></td>
|
245 | <td headers>Ergo test</td>
|
246 | </tr>
|
247 | <tr>
|
248 | <th id="blank"></th>
|
249 | <td headers><a href="https://github.com/accordproject/ergo/tree/master/packages/ergo-engine">ergo-engine</a></td>
|
250 | <td headers><a href="https://badge.fury.io/js/%40accordproject%2Fergo-engine"><img src="https://badge.fury.io/js/%40accordproject%2Fergo-engine.svg" alt="npm version"></a></td>
|
251 | <td headers>Ergo engine</td>
|
252 | </tr>
|
253 | <tr>
|
254 | <td headers><a href="https://docs.accordproject.org/docs/next/markup-cicero.html">Markdown</a></td>
|
255 | <td headers><a href="https://github.com/accordproject/markdown-transform/tree/master/packages/markdown-common">markdown-common</a></td>
|
256 | <td headers><a href="https://badge.fury.io/js/%40accordproject%2Fmarkdown-common"><img src="https://badge.fury.io/js/%40accordproject%2Fmarkdown-common.svg" alt="npm version"></a></td>
|
257 | <td headers>A framework for transforming markdown</td>
|
258 | </tr>
|
259 | <tr>
|
260 | <th id="blank"></th>
|
261 | <td headers><a href="https://github.com/accordproject/markdown-transform/tree/master/packages/markdown-slate">markdown-slate</a></td>
|
262 | <td headers><a href="https://badge.fury.io/js/%40accordproject%2Fmarkdown-slate"><img src="https://badge.fury.io/js/%40accordproject%2Fmarkdown-slate.svg" alt="npm version"></a></td>
|
263 | <td headers>Transform markdown to/from CommonMark DOM</td>
|
264 | </tr>
|
265 | <tr>
|
266 | <td headers></td>
|
267 | <td headers><a href="https://github.com/accordproject/markdown-transform/tree/master/packages/markdown-cli"> markdown-cli </a></td>
|
268 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fmarkdown-cli"><img src="https://badge.fury.io/js/%40accordproject%2Fmarkdown-cli.svg" alt="npm version"></a></td>
|
269 | <td headers> CLI for markdown transformations.</td>
|
270 | </tr>
|
271 | <tr>
|
272 | <th id="blank"></th>
|
273 | <td headers><a href="https://github.com/accordproject/markdown-transform/tree/master/packages/markdown-cicero">markdown-cicero</a></td>
|
274 | <td headers><a href="https://badge.fury.io/js/%40accordproject%2Fmarkdown-cicero"><img src="https://badge.fury.io/js/%40accordproject%2Fmarkdown-cicero.svg" alt="npm version"></a></td>
|
275 | <td headers>CiceroDOM: Markdown extensions for contracts, clauses, variables etc.</td>
|
276 | </tr>
|
277 | <tr>
|
278 | <th id="blank"></th>
|
279 | <td headers><a href="https://github.com/accordproject/markdown-transform/tree/master/packages/markdown-html">markdown-html</a></td>
|
280 | <td headers><a href="https://badge.fury.io/js/%40accordproject%2Fmarkdown-html"><img src="https://badge.fury.io/js/%40accordproject%2Fmarkdown-html.svg" alt="npm version"></a></td>
|
281 | <td headers>Transform CiceroDOM to HTML</td>
|
282 | </tr>
|
283 |
|
284 | </table>
|
285 |
|
286 | #### UI Components:
|
287 |
|
288 | <table>
|
289 | <tr>
|
290 | <th headers="blank">Projects</th>
|
291 | <th headers="blank">Package name</th>
|
292 | <th headers="blank">Version</th>
|
293 | <th headers="blank">Description</th>
|
294 | </tr>
|
295 | <tr>
|
296 | <td headers>Markdown Editor</td>
|
297 | <td headers><a href="https://github.com/accordproject/markdown-editor">markdown-editor</a></td>
|
298 | <td headers> <a href="https://badge.fury.io/js/%40accordproject%2Fmarkdown-editor">
|
299 | <img src="https://badge.fury.io/js/%40accordproject%2Fmarkdown-editor.svg" alt="npm version"></a></td>
|
300 | <td headers>WYSIWYG rich text web editor that persists text as markdown. Based on Slate.js</td>
|
301 | </tr>
|
302 | <tr>
|
303 | <td headers="co1 c1">Cicero UI</td>
|
304 | <td headers="co2 c1"><a href="https://github.com/accordproject/cicero-ui">cicero-ui</a></td>
|
305 | <td headers="co3 c1"> <a href="https://badge.fury.io/js/%40accordproject%2Fcicero-ui"><img src="https://badge.fury.io/js/%40accordproject%2Fcicero-ui.svg" alt="npm version"></a></td>
|
306 | <td headers="co4 c1">WYSIWYG contract editor, template libary browser, error panel component</td>
|
307 | </tr>
|
308 | <tr>
|
309 | <td headers="co1 c1">Concerto UI</td>
|
310 | <td headers="co2 c1"><a href="https://github.com/accordproject/concerto-ui">concerto-ui</a></td>
|
311 | <td headers="co3 c1"> <a href="https://badge.fury.io/js/%40accordproject%2Fconcerto-ui-react"><img src="https://badge.fury.io/js/%40accordproject%2Fconcerto-ui-react.svg" alt="npm version"></a></td>
|
312 | <td headers="co4 c1">Dynamic web forms generated from Concerto models</td>
|
313 | </tr>
|
314 | </table>
|
315 |
|
316 |
|
317 | #### Template Editors:
|
318 |
|
319 | <table>
|
320 | <tr>
|
321 | <th headers="blank">Projects</th>
|
322 | <th headers="blank">Cicero ver.</th>
|
323 | <th headers="blank">Description</th>
|
324 | </tr>
|
325 | <tr>
|
326 | <td headers><a href="https://github.com/accordproject/template-studio">Template Studio v1</a></td>
|
327 | <td headers> <b>0.13.4</b></td>
|
328 | <td headers>Web UI for creating, editing and testing Accord Project templates</td>
|
329 | </tr>
|
330 | <tr>
|
331 | <td headers><a href="https://github.com/accordproject/template-studio-v2">Template Studio v2</a></td>
|
332 | <td headers> <b>0.13.4</b></td>
|
333 | <td headers>Web UI for creating, editing and testing Accord Project templates</td>
|
334 | </tr>
|
335 | <tr>
|
336 | <td headers><a href="https://github.com/accordproject/cicero-vscode-extension">VSCode Extension</a></td>
|
337 | <td headers><b>0.13.4</b></td>
|
338 | <td headers>VS Code extension for editing Cicero templates and Ergo logic</td>
|
339 | </tr>
|
340 | </table>
|
341 |
|
342 |
|
343 | #### Public templates and models:
|
344 |
|
345 | <table>
|
346 | <tr>
|
347 | <th headers="blank">Projects</th>
|
348 | <th headers="blank">Description</th>
|
349 | </tr>
|
350 | <tr>
|
351 | <td headers><a href="https://github.com/accordproject/models">Models</a></td>
|
352 | <td headers>Accord Project Model Library </td>
|
353 | </tr>
|
354 | <tr>
|
355 | <td headers><a href="https://github.com/accordproject/cicero-template-library">Template Library</a></td>
|
356 | <td headers>Accord Project Template Library </td>
|
357 | </tr>
|
358 |
|
359 | </table>
|
360 |
|
361 |
|
362 | #### Documentation:
|
363 |
|
364 | <table>
|
365 | <tr>
|
366 | <th headers="blank">Project</th>
|
367 | <th headers="blank">Repo</th>
|
368 | </tr>
|
369 | <tr>
|
370 | <td headers><a href="https://docs.accordproject.org/">Documentation</a></td>
|
371 | <td headers><a href="https://github.com/accordproject/techdocs">techdocs</a></td>
|
372 | </tr>
|
373 | </table>
|
374 |
|
375 | ## Contributing
|
376 |
|
377 | The Accord Project technology is being developed as open source. All the software packages are being actively maintained on GitHub and we encourage organizations and individuals to contribute requirements, documentation, issues, new templates, and code.
|
378 |
|
379 | Find out what’s coming on our [blog][apblog].
|
380 |
|
381 | Join the Accord Project Technology Working Group [Slack channel][apslack] to get involved!
|
382 |
|
383 | For code contributions, read our [CONTRIBUTING guide][contributing] and information for [DEVELOPERS][developers].
|
384 |
|
385 | ## License <a name="license"></a>
|
386 |
|
387 | Accord Project source code files are made available under the [Apache License, Version 2.0][apache].
|
388 | Accord Project documentation files are made available under the [Creative Commons Attribution 4.0 International License][creativecommons] (CC-BY-4.0).
|
389 |
|
390 | [CommonMark]: https://commonmark.org
|
391 | [slate]: https://docs.slatejs.org/
|
392 |
|
393 | [linuxfound]: https://www.linuxfoundation.org
|
394 | [charter]: https://github.com/accordproject/markdown-editor/blob/master/CHARTER.md
|
395 | [apmain]: https://accordproject.org/
|
396 | [apworkgroup]: https://calendar.google.com/calendar/event?action=TEMPLATE&tmeid=MjZvYzIzZHVrYnI1aDVzbjZnMHJqYmtwaGlfMjAxNzExMTVUMjEwMDAwWiBkYW5AY2xhdXNlLmlv&tmsrc=dan%40clause.io
|
397 | [apblog]: https://medium.com/@accordhq
|
398 | [apnews]: https://www.accordproject.org/news/
|
399 | [apgit]: https://github.com/accordproject/
|
400 | [apdoc]: https://docs.accordproject.org/
|
401 | [apslack]: https://accord-project-slack-signup.herokuapp.com
|
402 |
|
403 | [docspec]: https://docs.accordproject.org/docs/spec-overview.html
|
404 | [docwelcome]: https://docs.accordproject.org/docs/accordproject.html
|
405 | [dochighlevel]: https://docs.accordproject.org/docs/spec-concepts.html
|
406 | [docergo]: https://docs.accordproject.org/docs/logic-ergo.html
|
407 | [docstart]: https://docs.accordproject.org/docs/accordproject.html
|
408 | [doccicero]: https://docs.accordproject.org/docs/basic-use.html
|
409 | [docstudio]: https://docs.accordproject.org/docs/advanced-latedelivery.html
|
410 |
|
411 | [contributing]: https://github.com/accordproject/markdown-editor/blob/master/CONTRIBUTING.md
|
412 | [developers]: https://github.com/accordproject/markdown-editor/blob/master/DEVELOPERS.md
|
413 |
|
414 | [apache]: https://github.com/accordproject/markdown-editor/blob/master/LICENSE
|
415 | [creativecommons]: http://creativecommons.org/licenses/by/4.0/
|