Scribio is a Javascript library to perform inline editing on a webpage. It is inspired by x-editable, but doesn't rely on jquery, offers a lot of customization options and can be extended by the mean of types and renderers.
This project was mainly an experiment, check Flyter for a library to perform inline editing.
DISCLAIMER: This project is still under active development, use it at your own risks
- Uses DOMPurify to clean templates and markup before appending it to the DOM
- Popup renderer makes use of Popper.JS for perfect positioning
- Can be extended, you can create custom types (user input) and renderers (where the types are rendered). Scribio ships with a few types and a popup renderer by default
- Highly customizable using configuration options
- Possibility to develop and load themes which are based on configuration options (currently a Bootstrap 4 theme is available)
The library comes in two flavors, either including or not Popper.js. You can include the corresponding file
<script src="path/to/scribio/scribio.min.js"></script> <!-- without popper -->
<script src="path/to/scribio/scribio.popper.min.js"></script> <!-- with popper -->
Once the library is loaded on your webpage, you can do the following to trigger Scribio on an element.
const div = document.getElementById('myDiv');
Scribio.span(div, { /* config */ });
You'll notice that Scribio doesn't include any pre-defined style, this is a development choice to avoid enforcing you
with pre-built mandatory styling you'll have to override using !important
rules. Rather than that, it can be easily
customized to fit a global framework, for example Bootstrap.
To load the bootstrap theme simply include the script after including Scribio.
<script src="path/to/scribio/scribio.popper.min.js"></script>
<script src="path/to/scribio/scribio.bootstrap-theme.min.js"></script>
This enforces a new markup for the popup renderer (using bootstrap's popover), buttons and inputs.
You can also directly work with Scribio in your application. Simply require it:
import Scribio from 'Scribio';
const div = document.getElementById('myDiv');
Scribio.span(div, { type: { name: 'text' }, /* other config... */ });
To use the bootstrap theme, include it and load it:
import Scribio, { BootstrapTheme } from 'Scribio';
Scribio.loadTheme(BootstrapTheme('sm')) /* pass a global bootstrap size for buttons and inputs */
Scribio currently ships with a single renderer, being the Popup renderer. It internally uses Popper.js
to display popups and, as such, requires the library to be loaded. It will first try to load it from
window.Popper
but will cry if it can't find it. Another way is to provide Scribio with your custom Popper
instance like so.
import MyPopperInstance from 'popper.js';
Scribio.loadTheme({
renderers: [{ name: 'popup', config: { popper: MyPopperInstance } }],
});
Scribio can be configured with the following options. Each of them has a given type, which should be the returned value. As such, each option can be
a function returning a value. The async
column tells you if your function can return a Promise if necessary. All functions
are automatically binded to:
- The instance by default
- The type when under the
type.
namespace (mainly configuring the type undertype.config
) - The renderer when under the
renderer.
namespace (mainly configuring the renderer underrenderer.config
) This means that you'll have access to it usingthis
.
Note that the configuration is an object, keys like
handler.onSubmit
refers to { handler: { onSubmit: ..., } }
You can check all defaults in the default config file.
key | type | description | async |
---|---|---|---|
trigger | string | Scribio trigger, can be either 'click', 'hover' or 'none' |
no |
emptyValue | any | The empty value, which indicates to Scribio that the current instance has no value | no |
voidDisplay | string | The text displayed if the current value is equal to the config option emptyValue |
no |
initialValue | any | This instance's initialization value | no |
valueDisplay | function | What should be displayed for a given value, by default will use the Type to display it | no |
Quickly override default configuration for type and renderer as well as choose which type and renderer to use for this instance.
key | type | description | async |
---|---|---|---|
type.name | string | Used type's name for this instance | no |
type.config | object | Config passed to the type | no |
renderer.name | string | Used renderer's name for this instance | no |
renderer.config | object | Config passed to the renderer | no |
By default, Scribio ships with a server handler which performs an async request on submit to a server.
This can be easily overriden by setting the handler.onSubmit
option as seen below, but these options
allows you to configure how the request is sent if you keep the default handler.
key | type | description | async |
---|---|---|---|
server.url | string | Server's URL to which the request should be sent | no |
server.requestParams | object | Configuration passed to the fetch request, you could for example set { method: 'POST' } here |
no |
server.resultFormatter | function | This function should return the new value after submit, you can use to change it or extract it from the given fetch response | no |
Multiple handlers allow you to hook into the lifecycle of your Scribio instance.
key | type | description | async |
---|---|---|---|
handler.onOpen | function | Called when the instance is triggered | yes |
handler.onClose | function | Called when the instance is closed | yes |
handler.onSubmit | function | Called when a value is submitted, by default tries to send a server request. Takes three arguments, the new value, an onSuccess callback, which takes the new value, and an onError callback which takes the error | using callbacks |
handler.onError | function | Called when a function arise, takes two arguments, the error and a callback that MUST be called with the error as argument | using callback |
handler.onCancel | function | If the edition is cancelled | yes |
handler.onLoading | function | Called when the loading status change | no |
handler.errorDisplay | function | Called when there's an error to display, should return a string | no |
handler.validate | function | Called right after new value is submitted, should return a boolean indicating if the value is correct or not | no |
Configuration options for the two action buttons. These can be overriden by redefining the used template in the template configuration options.
key | type | description | async |
---|---|---|---|
buttons.enabled | boolean | If the buttons are enabled or not | no |
buttons.submitText | string | The text to display in the submit button | no |
buttons.cancelText | string | The text to display in the cancel button | no |
You can override the global templates here if you would like to change the generated markup for different parts of the library.
Note that each important element has an aria-
attribute that must be set.
key | type | description | async |
---|---|---|---|
template.edit | string | The template used to generate the global edition container which includes type and action buttons | yes |
template.buttons | string | The action buttons template | yes |
template.read | string | The template used to display the value when not triggered | no |
template.loading | string | The template used to indicate a loading state waiting on the renderer to finish rendering itself | no |
Scribio currently ships with 4 types ready-to-use.
You can use it by setting your config as such { type: { name: 'text' }}
. It comes with the following configuration options:
key | type | default | description | async |
---|---|---|---|---|
class | string | '' |
Input additional class | no |
type | string | 'text' |
The input html type, for example text, number... Can also be textarea | no |
attributes | string | '' |
Additional html attributes | no |
You can use it by setting your config as such { type: { name: 'select' }}
. It comes with the following configuration options:
key | type | default | description | async |
---|---|---|---|---|
multiple | bool | false |
If your select accepts multiple values. If yes, its returned value will be an array | no |
dataSource | array | [] |
Your input data source. Must be an array of objects of the type{ value: 1, text: 'My label' } |
yes |
class | string | '' |
Input additional class | no |
displaySeparator | string | ', ' |
If multiple, when the value is displayed, it will be separated by it | no |
You can use it by setting your config as such { type: { name: 'radio' }}
. It comes with the following configuration options:
key | type | default | description | async |
---|---|---|---|---|
dataSource | array | [] |
Your input data source. Must be an array of objects of the type{ value: 1, text: 'My label' } |
yes |
containerClass | string | '' |
Radio container class | no |
labelClass | string | '' |
Radio labels class | no |
radioClass | string | '' |
Radio input class | no |
You can use it by setting your config as such { type: { name: 'checkbox' }}
. It comes with the following configuration options:
key | type | default | description | async |
---|---|---|---|---|
dataSource | array | [] |
Your input data source. Must be an array of objects of the type{ value: 1, text: 'My label' } |
yes |
containerClass | string | '' |
Checkbox container class | no |
labelClass | string | '' |
Checkbox labels class | no |
checkboxClass | string | '' |
Checkbox input class | no |
displaySeparator | string | ', ' |
When the value is displayed it will be separated by it | no |
Currently Scribio ships with a single renderer.
Displays the edition in a popup handled by Popper.js. You can use it by setting your config as such { renderer: { name: 'popup' } }
It comes with the following default configuration:
key | type | default | description | async |
---|---|---|---|---|
popperConfig | object | { placement: 'top' } |
Configuration passed down to Popper on initialization | no |
popper | object | window.Popper |
Popper instance | no |
transitionDuration | number | 300 |
Transition duration in milliseconds | no |
closeOnClickOutside | bool | true |
If clicking outside the popup cancel and close the edition session | no |
popupTemplate | string | Check the source | The popup's template, note the aria elements which are required if you override this option | no |
title | string | null |
Popup title | no |
##Customizing Scribio Scribio can be easily customized by the means of adding new types, renderers and themes which will be explained here.
When you first call Scribio.span
on an element, it will create and attach a new instance to it.
The instance exposes the following methods:
- setNewValue(value) which allows you to dynamically update this instance's current value
- setLoading(status) which will show/hide the loading status
- open() which starts a new edition session if none currently exists
- close() which closes any existing session
- destroy() which destroys the instance's markup and removes existing listeners
- refreshContent() which triggers a new rendering of the displayed value
Furthermore the instance provides the following attributes:
config
which exposes the instance's configcreateType
a function which returns a new instance of the typecreateRenderer
a function which returns a new instance of the renderervalue
the instance's current valuesession
if an edit session exists it will be stored here, null otherwisetarget
the HTML node to which this instance is attachedariaElement
the generated read element
Once your instance was triggered and opens a new edit session, its session
attribute will have a new Session object.
It offers the following methods:
- open() which opens the renderer and displays the type
- cancel() which cancels the edition and closes it
- setLoading(status) which displays a loading status in the renderer and disable/enable the input
- error(error) which displays an error in the renderer
- submit() triggers a submit, retrieves the type's value and handles it
- destroySession() destroys this session, should not be called directly, rather use
instance.close()
It also exposes the following attributes:
instance
which is the instance to which this session is attached and opened fromtype
an instance of the typerenderer
an instance of the renderermarkup
which is the edit markup from thetemplate.edit
configuration
Once a session is created, it will automatically create a new instance of the type and of the renderer. As such, they can store in themselves inner properties and attributes, types and renderer lives as long as the edit session is opened.
You can add new types to Scribio very easily. It all comes down to creating a new class and registering it.
The default type class code is the following
export default class {
constructor(instance, config = {}) {
/* Reference to the instance and the given config */
}
init() {
/* Here you can setup your type and perform async jobs by returning a Promise */
}
show(rootNode, value) {
/* Called when the type is displayed, you should append your type markup to the given rootNode and feed it with the given value */
}
getInputValue() {
/* This should return your type's value */
}
getReadableValue(value) {
/* Here you should convert the given value to a stringified version which will be displayed to the end user */
}
disable(status) {
/* You should enable or disable your type (make it visible in your markup) depending on the given status */
}
onDestroy() {
/* Here you must clean your type, removing event listeners and so on. Your markup will be automatically cleaned for you */
/* This method can also return a Promise */
}
}
To make Scribio aware of your new type, you can simply register it like so:
Scribio.registerType('myType', typeClass, someConfig);
Where:
- myType is your type name, it will be used when spanning new instances
- typeClass is the class of your type
- someConfig, you can eventually pass some config here. This is used if you register a third-party type and want to override default configuration on type registration.
Creating a renderer is a bit more complex than creating a type, but fear not, it remains quite easy. A renderer is responsible to display the type once an edit session is triggered.
Creating a renderer implies creating a class for it. Here is the default skeleton:
export default class {
constructor(instance, config = {}) {
/* Constructor which receives the Scribio instance and given config */
}
init() {
/* This can be used to initialize the renderer, it is called right after it was instanciated. For example in the Popup renderer it is used to parse the markup, attach event listeners and initialize Popper.js. It can return a Promise. */
}
error(error) {
/* This should display the given error string somewhere in your renderer */
}
show(markup) {
/* This is called right after init has finished resolving, in the popup renderer it is used to handle transition. You must insert the given markup (type, actions...) in your renderer */
}
loading(status) {
/* This should display a visual element indicating if a job is running or not depending on the given status */
}
destroy() {
/* This should completely destroy your renderer, clean event listeners and markup. It can return a promise */
}
}
Once your renderer is ready, you can easily register it:
Scribio.registerRenderer('myRenderer', rendererClass, someConfig);
Where:
- myRenderer is your renderer name, it will be used when spanning new instances
- rendererClass is the class of your renderer
- someConfig, you can eventually pass some config here. This is used if you register a third-party renderer and want to override default configuration on registration.
Scribio supports global modifiers by the mean of themes. A theme is, in the end, simply a big configuration object. You can override configuration for everything in a theme. It still differs a little bit from the standard default configuration. Here is a basic theme:
export default {
types: [
{
name: 'text',
config: { /* text type config */},
},
],
renderers: [
{
name: 'popup',
config: { /* popup renderer config */},
},
],
config: {
/* Global config */
},
}
As you can see, a theme can override configuration per type and renderer, as well as override any global configuration option. You don't have to override everything, only the parts you want. The configuration tree is then progressively merged as detailed in the next section.
Scribio is based on a lot of configuration options. Here is the order in which it is merged:
- Default configuration, first we take the base configuration object given to have all properties defined
- Loaded configuration, only for types and renderers, we first merge it with the configuration given on registration
- Theme configuration, we iterate over each theme by order of registration and merge it
- Spanning configuration, we finish by merging it with the configuration given on a
Scribio.span(node, config)
For Scribio core types and renderers as well as instance configuration, the whole tree is flattened to a one level object.
Objets like { handler: { onSubmit: fn } }
become { 'handler.onSubmit': fn }
.
Each configuration option has a type (number, string, function as detailed in the upper section), but can also be a function returning a value
of the given type (except for functions as explained in the section below). As such, for each detected function, we bind
the this
to an object, for the instance configuration it is the instance itself, for a renderer it is the renderer itself
and the same goes for the type. For example you can do something like this if the type is stored in an attribute:
<div id="scribio" data-type="text" data-value="Initial value"></div>
<script type="text/javascript">
Scribio.span(document.getElementById('scribio'), {
type: {
name() {
return this.target.getAttribute('data-type');
},
},
initialValue() {
return this.target.getAttribute('data-value');
},
});
</script>
Please be aware that arrow function won't work because of the this
context which cannot be bound dynamically.
If you need to read the instance, type or renderer's configuration once you have access to it, you can do it by the following:
instance.config('config.key')
If the config value you expect is a function you wish to call, you rather have to retrieve it like this:
instance.config.fn('config.key')
This code is licensed under the MIT license