Skip to content

Danilo-Zekovic/react-tableql

Repository files navigation

react-tableql

Build Status codecov downloads npm version GitHub license PRs Welcome

stars fork watch

Twitter Follow

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

NOTE: rewrite is in progress, check out the rewrite branch https://github.com/Danilo-Zekovic/react-tableql/tree/rewrite

Installing and Using react-tableql

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

Dependencies

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

TypeScript (>= v0.1.0-beta.0)

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

Server-Side Rendering (SSR)

Next.js

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 })

Props

Prop Required Type Default Value Description TableQL 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: ApolloTableQL 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.

query

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.

data

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.

columns

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.

Columns Object Options

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
id

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'.

label

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'
    ]} />
component

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() />}
    ]} />
customColumn

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,
    },
  ]} />
headerStyle

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'
    },
  ]} />
nodeStyle

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

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),
    },
  ]}
/>

pagination

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)
      }
    }
  } />

onRowClick

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)} />

onEmpty

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} />

loading

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} />

error

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!')} />

sort

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 />

errorMessage

Error message to be displayed when error occurs.

Example:

<ApolloTableQL query={GET_ALL_FILMS} errorMessage="There was an error! Try again." />

Apollo Options

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} />

styles

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.

debug

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.


Licence

MIT


Scripts

start

Builds the component with hot reload. Useful when developing and working on the TableQL.

build

Builds production ready component using Webpack.

build:all

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.

test

Run all the tests.

test:watch

Runs tests in watch mode.

test:coverage

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.

format

Formats the code using Prettier by rules defined in Pretier config file.

format:check

Checks if all the formating follows the defined rules.

storybook

Starts Storybook locally on port 6006, http://localhost:6006/

build-storybook

Builds static Storybook files that will be used on GitHub pages as online TableQL documentation and component demonstration.

Travis

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.


TODO

  • 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.

About

React table component that is harnessing the power GraphQL. Simple way to display your data.

Resources

License

Code of conduct

Stars

Watchers

Forks

Packages

No packages published