Easy to use React components for creating data grids with zero to no configuration. Simple way to display your data.
If this project helped you improve and simplify your application, consider supporting it by buying me a beer.
Official React TableQL Storybook: https://danilo-zekovic.github.io/react-tableql/
Example app: https://github.com/Danilo-Zekovic/react-tableql-example-app
TypeScript example app: https://github.com/Danilo-Zekovic/react-tableql-typescript-example-app
Next.js example app: https://github.com/Danilo-Zekovic/react-tableql-nextjs-example-app
If you have any suggestions please let me know or join the project.
Latest code can be found on GitHub
First install it:
npm install react-tableql
Then import it into your React component:
import TableQL from 'react-tableql'
or for ApolloTableQL:
import { ApolloTableQL } from 'react-tableql'
Component in which is ApolloTableQL used must be wrapped in Apollo Client Provider:
ReactDOM.render(
(
<ApolloProvider client={client}>
<App />
</ApolloProvider>
),
document.getElementById('root')
)
For more information how to setup Apollo Client please visit their Get Started page: https://www.apollographql.com/docs/react/essentials/get-started.html
These packages are required to be able to use TableQL and ApolloTableQL:
- React
npm install --save react
Additionally for ApolloTableQL:
- GraphQL
- Apollo Client
npm install --save graphql @apollo/client
For version 0.1.0-beta.4 or older, or 0.0.10 and older
npm install --save graphql apollo-boost react-apollo
react-tableql supports type definitions for TypeScript out of the box. It ships with definitions in its npm package, so installation should be done for you after the library is included in your project. Available from version v0.1.0-beta.0
import dynamic from 'next/dynamic'
const ApolloTableQL = dynamic(() => import('react-tableql').then((m) => {
const { ApolloTableQL } = m
return ApolloTableQL
}), { ssr: false })
import dynamic from 'next/dynamic'
const TableQL = dynamic(() => import('react-tableql'), { ssr: false })
Prop | Required | Type | Default Value | Description | TabelQL | ApolloTableQL |
---|---|---|---|---|---|---|
query | Yes | String | - | Graphql query | ❌ | ✅ |
data | Yes | Array or Object | - | Data to be parsed and rendered | ✅ | ❌ |
columns | No | Array | null |
Customization of columns | ✅ | ✅ |
pagination | No | Object or Boolean | false |
Adds pagination | ✅ | ✅ |
onRowClick | No | Function | null |
Handle row click | ✅ | ✅ |
onEmpty | No | React Functional Component | null |
Component to render if data is empty | ✅ | ✅ |
styles | No | Object | {} |
Change the look of part of a table | ✅ | ✅ |
debug | No | Boolean | false |
Turns on and off the debug mode | ✅ | ✅ |
loading | No | Boolean | false |
Indicates if data is loading | ✅ | ❌ |
error | No | Error | '' |
Error object | ✅ | ❌ |
sort | No | Boolean | false |
All columns can be sorted by | ✅ | ✅ |
errorMessage | No | String | '' | When error occurs display this message | ✅ | ✅ |
Apollo Options | No | - | - | See Apollo Client site for more details | ❌ | ✅ |
NOTE: ApolloTabelQL has yet to be tested for all the options that Apollo Client has to offer. See the once that have been tested in the section Apollo Options bellow.
To pass the GraphQL query to the component, all you have to do is create a string with backticks as follows:
const GET_ALL_FILMS = `
{
allFilms(first:7){
films{
title
episodeID
}
}
}
`
Then pass it as a query prop:
<ApolloTableQL query={GET_ALL_FILMS} />
This is the minimum requirement for ApolloTableQL to work, rest of the props are used to modify and customize it. query is mandatory prop, all other are optional.
It should be an array of objects, or an object which has at least one property that is an array. TableQL will take it and parse it.
Example:
const PEOPLE = [
{ id:1, name: 'Luke Skywalker', age: 25 },
{ id:2, name: 'Leia Organa' , age: 25 },
{ id:3, name: 'R2-D2', age: 100 }
]
<TableQL data={PEOPLE} />
This example will render a table with three columns and four rows (including headers). Where columns will be id, name, and age.
Prop columns
is very important when one wants to customize the columns.
It is an array. The simplest usage for it is making the order of the columns, of
course order can be made in query as well.
Example of order:
<ApolloTableQL
query={GET_ALL_FILMS}
columns={['episodeID', 'releaseDate', 'title']} />
NOTE: Order can be controlled in GraphQL query, which is recommended way as that improves performance because TableQL does not have to do extra work.
Property | Required | Type | Default Value | Description |
---|---|---|---|---|
id | Yes | String | - | Accessor |
label | No | String | - | Table header label |
component | No, Yes when customColumn true |
Function | null |
Change the value, or inject a component or value |
customColumn | No | Boolean | false |
Adds a column that is not populated by data from query |
headerStyle | No | String | '' |
CSS class for header style |
nodeStyle | No | String or Function | '' |
CSS class style for nodes in column |
sort | No | Boolean or Function | false |
Sort by this column |
When column is represented as object then id
property is required, so that TableQL knows which value to grab.
If data that needs to be displayed is in nested object then id
should be a chain of keys separated with periods, Ex. 'foo.bar.blah'.
If you want to have a custom header label for the column use property label
.
Example:
<ApolloTableQL
query={GET_ALL_FILMS}
columns={[
{id:'episodeID', label:'Episode Identification'},
'releaseDate',
'title'
]} />
To add your own column formating or custom React component use component
parameter. Pass it a function that returns a React component or a function that formats the data. TableQL will return data that would have been displayed
Example with React component:
<ApolloTableQL
query={GET_ALL_FILMS}
columns={[
'episodeID',
'releaseDate',
{ id: 'title', component: data => <Button data={data} />}
]} />
Example with formating function:
<ApolloTableQL
query={GET_ALL_FILMS}
columns={[
'episodeID',
'releaseDate',
{ id: 'title', component: data => data.toUpperCase() />}
]} />
This property indicates if the column is addition to existing columns that are extracted from returned data or if it is something that developer is adding to rest of the table. Pass true
to customColumn
property, and provide component so that TableQL knows how to render it. When customColumn: true
then all the data for the given row is returned to component, which makes it easy to do manipulations such as edit, delete...
Example:
<ApolloTableQL
query={GET_ALL_FILMS}
columns={[
'episodeID',
'releaseDate',
'title',
{
id: 'actions',
component: props => <Button data={props} />,
customColumn: true,
},
]} />
Pass a class name to headerStyle
to only style the header of that one column.
Example:
<ApolloTableQL
query={GET_ALL_FILMS}
columns={[
'episodeID',
'releaseDate',
{
id: 'title',
headerStyle: 'custom-style-class'
},
]} />
To style nodes/cells of the column you can pass a string or function. When string passed then class name is passed. When function is passed TableQl will return the data for that row as parameters. It can be used for conditional styling such as make the color green when true.
Example with string:
<ApolloTableQL
query={GET_ALL_FILMS}
columns={[
'episodeID',
'releaseDate',
{
id: 'title',
nodeStyle: 'custom-style-class',
},
]} />
Example with function:
<ApolloTableQL
query={GET_ALL_FILMS}
columns={[
'episodeID',
'releaseDate',
{
id: 'title',
nodeStyle: data => {
if (data.title.length <= 10) return 'custom-style-class'
},
},
]} />
Sort can be boolean or a function that takes array of objects as first argument and string property which was clicked to be sorted, and it should return the array of objects that were reorder:
Example when sort boolean:
<ApolloTableQL
query={GET_ALL_FILMS}
columns={[
'episodeID',
'releaseDate',
{
id: 'title',
sort: true,
},
]} />
Example when sort function:
<ApolloTableQL
query={GET_ALL_FILMS}
columns={[
'episodeID',
'releaseDate',
{
id: 'title',
sort: (data, property) => data.sort(property),
},
]}
/>
To enable pagination pass true
or config object to pagination
prop. When true
passed then component will use its default values to configure the pagination.
Config object has the following properties:
{
pageLimit: 10,
pageNeighbors: 0,
currentPage: 1,
onPageChanged: (currentPage, totalPages, pageLimit, totalRecords) => void
}
Property | Required | Type | Default Value | Description |
---|---|---|---|---|
pageLimit | No | Number | 10 |
Number of rows per page |
pageNeighbors | No | Number | 0 |
Number of pages that will be shown around current page in pagination component |
currentPage | No | Number | 1 |
The initial page |
onPageChanged | No | Function | null |
On page change returns data about pagination |
Example:
<ApolloTableQL
query={GET_ALL_FILMS}
pagination={
{
pageLimit: 5,
pageNeighbors: 2,
currentPage: 1,
onPageChanged: ({currentPage, totalPages, pageLimit, totalRecords}) => {
alert(currentPage)
}
}
} />
If something is needed to happen when the row is clicked then pass a function to onRowClick
prop. TableQL will return data that is found in the row that was clicked.
Example:
<ApolloTableQL
query={GET_ALL_FILMS}
onRowClick={data => alert(data)} />
If you need to render another component in case of an empty table, TableQL will do so calling onEmpty, a React Functional Component, with "reason" (string) as a prop.
Example:
const FailSafe = ({reason}) => <h3>You don't have data yet</h3>;
<ApolloTableQL
query={GET_ALL_FILMS}
onEmpty={FailSafe} />
Pass it to indicate if data loading. This is only for TableQL, ApolloTableQL takes care of it.
Example:
<TableQL
data={GET_ALL_FILMS}
loading={true} />
Pass the JavaScript Error object if there is an error while fetching the data, or if server returned an error. This is only for TableQL, ApolloTableQL takes care of it.
Example:
<TableQL
data={GET_ALL_FILMS}
error={new Error('Some error!')} />
Pass true
to add sort on all of the columns. When header label clicked sort the table by that column.
Example:
<ApolloTableQL query={GET_ALL_FILMS} sort />
Error message to be displayed when error occurs.
Example:
<ApolloTableQL query={GET_ALL_FILMS} errorMessage="There was an error! Try again." />
Pass data as you would to Apollo client to following props that match Apollo Client props. For details on Apollo Query Props please visit official documentation https://www.apollographql.com/docs/react/essentials/queries
Apollo Query Props | Tested | Works |
---|---|---|
query | ✅ | ✅ |
variable | ✅ | ✅ |
pollInterval | ✅ | ✅ |
notifyOnNetworkStatusChange | ✅ | ✅ |
fetchPolicy | ❌ | |
errorPolicy | ❌ | |
ssr | ❌ | |
displayName | ✅ | |
skip | ✅ | ✅ |
onCompleted | ✅ | ✅ |
onError | ✅ | ✅ |
context | ❌ | |
partialRefetch | ❌ | |
children | ✅ | ❌ |
client | ❌ | |
returnPartialData | ❌ |
NOTE: children is already taken care of by ApolloTableQL. There is no option to pass children to ApolloTableQL yet.
Those props that are not yet tested can also be passed to the ApolloTableQL just as you would to Query component from Apollo Client.
Example:
<ApolloTableQL query={SOME_QUERY_WITH_VAR} variable={someValue} />
To add custom styling to the table create CSS class(es) and pass them in an object with properties as shown bellow:
{
table: '',
thead: '',
theadTr: '',
theadTh: '',
tbody: '',
tbodyTr: '',
tbodyTd: '',
}
None of the properties are required, user can pass the classes just to the parts of table that they wish to change.
Example:
<ApolloTableQL
query={GET_ALL_FILMS}
styles={
{
table:'custom-class',
tbodyTd: 'custom-body-node-class'
}
}
/>
If no custom styling is passed component will use its own styling.
To enable debug mode pass true
to component props:
<ApolloTableQL query={GET_ALL_FILMS} debug={true} />
This mode will enable logs that are part of the component and are disabled by default.
MIT
Builds the component with hot reload. Useful when developing and working on the TableQL.
Builds production ready component using Webpack.
Builds production ready component using Webpack, and builds static Storybook files that will be used on GitHub pages as online TableQL documentation and component demonstration.
Run all the tests.
Runs tests in watch mode.
Check how much of the code do tests cover. To see it visually and explore what is not covered open ./coverage/lcov-report/index.html
in your browser.
Formats the code using Prettier by rules defined in Pretier config file.
Checks if all the formating follows the defined rules.
Starts Storybook locally on port 6006, http://localhost:6006/
Builds static Storybook files that will be used on GitHub pages as online TableQL documentation and component demonstration.
Continues Deployment is triggered on tag change. On every commit Travis will run tests, format check, Storybook build, and component build. It needs to be built with Node >= 9 for everything to pass.
- Column sorting when pagination enabled
- Sticky header
- Special cases (value is an array, ...)
- Continues integration
- Better documentation and user manual
- Search
- Resizable header/columns
- Filters
- Improving the component
- Cleanup of node packages
- Subscribe for live updates
- Context where global info can be set, ex. theming
- Tutorials
- pre-commit hooks (run tests and prettier before every commit)
- add linter: eslint
For more about TableQL visit Official React TableQL Storybook and explore all the options, and see some demo examples.