react-directive
A conditional, className and listing directive for react apps
Install
# npm
npm i react-directive
# yarn
yarn add react-directive
Introduction
react-directive
is a library for creating conditional or listing directives in a react app.
It takes a lot of inspiration from Vue.js' conditional, class bindings and listing directives such as v-if
, v-for
, and :class
.
Its two main purposes are listed below.
- To avoid the headache of using conditional (ternary) operators or higher order array methods such as map for list rendering.
- To keep JSX clean and concise.
Usage
To use react-directive
import it to your component.
import React from 'react';
import directive from 'react-directive';
const Component = props => {
return (
// This is the same as a div element but has more features like dirIf, dirFor, extended className etc.
<directive.div>
{ /* Any children for the div element */ }
</directive.div>
);
}
Conditional Rendering
react-directive
provides directives such as dirIf
, dirShow
and components <Switch />
, <Case />
<Default />
to render (show) or remove (hide) an element from the page.
dirIf
Prop The prop dirIf
is used to conditionally render a block. The block will only be rendered if the directive’s expression returns a truthy value (Check out truthy and falsy values in javascript here). It works like v-if
for Vuejs
and ngIf
for angular.
Currently, there is no support for else
and else-if
yet. ( You can use the <Switch />
component to handle such cases. More on that later ).
Note: undefined
is the only exceptional falsy value that evaluates to true
because the default value for dirIf
is true
Usage
The example component below renders a div
element only if the name's length is greater than 4
otherwise it will just render a null
value. If you want to keep the node
and only toggle style's display value, use the dirShow
prop instead.
import React, { useState } from 'react';
import directive from 'react-directive';
const Component = props => {
const [currentName, setCurrentName] = useState('Musah');
return (
<directive.div dirIf={currentName.length > 4}>
Div contents
</directive.div>
);
}
/*
When dirIf is truthy, it renders
<div>Div contents</div>
When dirIf is falsy, it renders nothing
*/
dirShow
Prop The prop dirShow
is used to conditionally show a block. The block will only be shown if the directive’s expression returns a truthy value. It works like v-show
for Vuejs
.
Note: dirShow
takes precedence to any another display for styles. For example if you set the display to block and dirShow
is false
, it will still hide the element.
Note: undefined
is the only exceptional falsy value that evaluates to true
because the default value for dirShow
is true
Usage
The example component below shows a div
element only if the name's length is greater than 4
otherwise it will hide the div
element by add the display:none
to its styles.
import React, { useState } from 'react';
import directive from 'react-directive';
const Component = props => {
const [currentName, setCurrentName] = useState('Musah');
return (
<directive.div dirShow={currentName.length > 4}>
Div contents
</directive.div>
);
}
/*
When dirShow is truthy, it renders
<div>Div contents</div>
When dirShow is falsy, it renders
<div style="display:none;">Div contents</div>
*/
<Switch />
Component The <Switch />
component is used to conditionally render a <Case />
that resolves to a truthy value. Otherwise it renders <Default />
(If it exists).
Usage
The example component below shows the div
element with contents Case 2
because it is the first case that resolves to a truthy value.
Note that the when
prop for the <Case />
component supports either a (truthy | falsy) value or a function that returns a (truthy | falsy) value.
It is recommended to use the function version if the calculation is intensive. This helps in short circuiting (lazy evaluation) when the case is not reached.
import React, { useState } from 'react';
import { Switch, Case, Default } from 'react-directive';
const Component = props => {
const [currentName, setCurrentName] = useState('Musah');
// This makes it more concise to render an element instead of using nested ternary operator.
// Fun fact: I was getting lots of eslint problems because of using ternary operators which was one of my main motivations for building this library.
return (
<Switch>
<Case when={!currentName.length}>
<div>Case 1</div>
</Case>
<Case when={currentName.length > 4}>
<div>Case 2</div>
</Case>
{/* The case below resolves to true but the case above will render because it is the first match */}
<Case when>
<div>Case 3</div>
</Case>
<Default>
<div>Default</div>
</Default>
</Switch>
);
}
/*
renders
<div>Case 2</div>
*/
Class Names
A common need for data binding is manipulating an element's class list.
react-directive
provides the hook useClassName
and an extended version of the className
prop to handle this issue
useClassName
Hook A simple hook for generating class names.
Usage
The hook takes in two optional parameters: classes
and deps
(dependencies).
-
classes
can be a string, an array of strings, an object with class names as keys and truthy values, or an array of such objects. -
deps
is an optional dependencies array that tells the hook to re-run when one of its values changes.
Note: When dependencies are not passed, it will fallback to re-calculate the class names based on the classes argument itself. Also it is advisable to pass an array of primitive values or cached values. So that it doesn't re-calculate unnecessarily. Think of it like a useEffect
dependency.
Here are some basic examples:
import { useClassName } from 'react-directive';
function Component() {
const className = useClassName({
active: true,
disabled: false,
});
return <div className={className}>Contents</div>;
}
// Renders <div class="active">Contents</div>
You can also pass an array of objects, strings, or a combination of both:
import { useState } from 'react'
import { useClassName } from 'react-directive';
function Component() {
const [isActive, setIsActive] = useState(true);
const [isDisabled, setIsDisabled] = useState(false);
const className = useClassName([
'highlighted',
{ active: isActive, disabled: isDisabled },
], [isActive, isDisabled]); // Optional deps to tell it to re-calculate based on isActive and isDisabled values
return <div className={className}>Contents</div>;
}
// Renders <div class="highlighted active">Contents</div>
// This is useful if you have some other classNames that doesn't have to react to any value
className
Prop react-directive
extends the default className of React to support what the useClassName
hook supports above. To pass the dependencies, use classNameDeps
props.
Here are some basic examples:
import { useState } from 'react'
import directive from 'react-directive';
function Component() {
const [isActive, setIsActive] = useState(true);
const [isDisabled, setIsDisabled] = useState(false);
return <directive.div className={{isActive, isDisabled}} classNameDeps={[isActive, isDisabled]} >Contents</directive.div>;
}
// Renders <div class="active">Contents</div>
// classNameDeps is optional but makes it more efficient and performant.
List Rendering
react-directive
provides the directive dirFor
and the component <For />
to generate lists. This makes it more readable than using Array.prototype.map
.
Usage
The <For />
component takes in an object with two properties: each
and children
-
each
is an array of items that you want to render. -
children
is a function that takes in two arguments:value
andindex
.value
is the current item in theeach
array andindex
is its index in the array.
Note: Make sure to pass a unique key prop to the elements rendered by the children function. This is important for React to keep track of the elements and render updates efficiently.
Here is a basic example:
import { For } from 'react-directive';
function Component() {
const items = ['Item 1', 'Item 2', 'Item 3'];
return (
<For each={items}>
{(value, index) => <div key={index}>{value}</div>}
</For>
);
}
/* renders
<div>Item 1</div>
<div>Item 2</div>
<div>Item 3</div>
*/
You might have noticed that we are passing a function. It is important to know that if you want the values of the elements in the list you passed, you must pass a function to be able to access them. This works like <Context.Consumer />
.
dirFor
Prop react-directive
includes a prop that takes an array
of items you want to render. The children prop is a function or React node that is called for each item in the array. The function takes in two arguments: currentItem
and index
. currentItem
is the current item in the dirFor
array and index
is its index in the array.
Note: Make sure to pass a unique key prop to the elements rendered by the children function using dirKey
(More later). This is important for React to keep track of the elements and render updates efficiently. It fallsback to using the index
of the item if not provided
Here is a basic example:
import directive from 'react-directive';
function Component() {
const items = ['Item 1', 'Item 2', 'Item 3'];
return (
// this means the current item. Used because the items are unique. If they are not
<directive.div dirKey="this" dirFor={items}>{(currentItem) => currentItem}</directive.div>
);
}
/* renders
<div>Item 1</div>
<div>Item 2</div>
<div>Item 3</div>
*/
Other props
dirKey
Prop The dirKey
prop is used to specify the key of the current element which is in the loop while using dirFor
.
- It can take
this
, which means the current item should be used as a key. Note: use this for only primitive values that you know will be unique in the list. - It can take any other string. When any other string is specified, it will use that to find the index in the current item according to the name. It will assume that the current item is an object. If null or undefined is returned, it will fallback to the index
- It can take in a function that returns the key
dirRef
Prop The dirRef
prop is used instead of the standard ref prop for directives. The ref
prop will not work when using directives.
Here is a basic example:
import { useRef } from 'react';
import directive from 'react-directive';
function Component() {
const inputRef = useRef<HTMLInputElement>(null);
const focusInput = () => {
inputRef.current.focus();
}
return (
<React.Fragement>
<directive.input placeholder="Name" dirRef={inputRef} />
<button onClick={focusInput}>Focus Input</button>
</React.Fragement>
);
}
/* renders
<div>Item 1</div>
<div>Item 2</div>
<div>Item 3</div>
*/
Pull Requests
Pull requests are welcome. Open pull requests to the pr branch and make sure it all lint and tests are passing.
License
MIT © Musah Kusi Hussein