From 20b650cb39a783150aed1df01edcd32c8076eb2b Mon Sep 17 00:00:00 2001 From: fzaninotto Date: Sun, 7 Mar 2021 21:54:07 +0100 Subject: [PATCH] [Doc] Fix section title level in List and Create/Edit chapters --- docs/CreateEdit.md | 26 +++--- docs/List.md | 194 +++++++++++++++++++++++---------------------- 2 files changed, 113 insertions(+), 107 deletions(-) diff --git a/docs/CreateEdit.md b/docs/CreateEdit.md index 10e6f6a75a5..c4508508cfc 100644 --- a/docs/CreateEdit.md +++ b/docs/CreateEdit.md @@ -76,7 +76,7 @@ That's enough to display the post edit form: ![post edition form](./img/post-edition.png) -**Tip**: You might find it cumbersome to repeat the same input components for both the `` and the `` view. In practice, these two views almost never have exactly the same form inputs. For instance, in the previous snippet, the `` views show related comments to the current post, which makes no sense for a new post. Having two separate sets of input components for the two views is, therefore, a deliberate choice. However, if you have the same set of input components, export them as a custom Form component to avoid repetition. +**Tip**: You might find it cumbersome to repeat the same input components for both the `` and the `` view. In practice, these two views almost never have exactly the same form inputs. For instance, in the previous snippet, the `` view shows related comments to the current post, which makes no sense for a new post. Having two separate sets of input components for the two views is, therefore, a deliberate choice. However, if you have the same set of input components, export them as a custom Form component to avoid repetition. React-admin injects a few props to the `create` and `edit` views: the `resource` name, the `basePath` (the root URL), the `permissions`, and, in the case of the `edit` view, the record `id`. That's why you need to pass the `props` to the `` and `` components. @@ -245,7 +245,7 @@ const Aside = ({ record }) => ( By default, the `Create` and `Edit` views render the main form inside a material-ui `` element. The actual layout of the form depends on the `Form` component you're using (``, ``, or a custom form component). -Some form layouts also use `Card`, in which case the user ends up seeing a card inside a card, which is bad UI. To avoid that, you can override the main form container by passing a `component` prop : +Some form layouts also use `Card`, in which case the user ends up seeing a card inside a card, which is bad UI. To avoid that, you can override the main page container by passing a `component` prop : ```jsx // use a div as root component @@ -325,7 +325,7 @@ The `` view exposes two buttons, Save and Delete, which perform "mutations - `optimistic`: The mutation is applied locally and the side effects are executed immediately. Then the mutation is passed to the dataProvider. If the dataProvider returns successfully, nothing happens (as the mutation was already applied locally). If the dataProvider returns in error, the page is refreshed and an error notification is shown. - `undoable` (default): The mutation is applied locally and the side effects are executed immediately. Then a notification is shown with an undo button. If the user clicks on undo, the mutation is never sent to the dataProvider, and the page is refreshed. Otherwise, after a 5 seconds delay, the mutation is passed to the dataProvider. If the dataProvider returns successfully, nothing happens (as the mutation was already applied locally). If the dataProvider returns in error, the page is refreshed and an error notification is shown. -By default, pages using `` use the `undoable` mutation mode. This is part of the "optimistic rendering" strategy of react-admin ; it makes the user interactions more reactive. +By default, pages using `` use the `undoable` mutation mode. This is part of the "optimistic rendering" strategy of react-admin ; it makes user interactions more reactive. You can change this default by setting the `mutationMode` prop - and this affects both the Save and Delete buttons. For instance, to remove the ability to undo the changes, use the `optimistic` mode: @@ -557,7 +557,7 @@ const PostEdit = props => ( ## Prefilling a `` Record -You may need to prepopulate a record based on another one. For that use case, use the `` component. It expects a `record` and a `basePath` (usually injected to children of ``, ``, ``, etc.), so it's as simple to use as a regular field or input. +Users may need to create a copy of an existing record. For that use case, use the `` component. It expects a `record` and a `basePath` (usually injected to children of ``, ``, ``, etc.), so it's as simple to use as a regular field or input. For instance, to allow cloning all the posts from the list: @@ -575,9 +575,9 @@ const PostList = props => ( ); ``` -Alternately, you may need to prepopulate a record based on a *related* record. For instance, in a `PostList` component, you may want to display a button to create a comment related to the current post. Clicking on that button would lead to a `CommentCreate` page where the `post_id` is preset to the id of the Post. +**Note**: `` is designed to be used in a `` and in an edit view `` component, not inside the form ``. The `Toolbar` is basically for submitting the form, not for going to another resource. -**Note** `` is designed to be used in an edit view `` component, not inside a ``. The `Toolbar` is basically for submitting the form, not for going to another resource. +Alternately, users need to prepopulate a record based on a *related* record. For instance, to create a comment related to an exising post. By default, the `` view starts with an empty `record`. However, if the `location` object (injected by [react-router-dom](https://reacttraining.com/react-router/web/api/location)) contains a `record` in its `state`, the `` view uses that `record` instead of the empty object. That's how the `` works under the hood. @@ -637,6 +637,8 @@ const CreateRelatedCommentButton = ({ record }) => ( ``` {% endraw %} +Should you use the location `state` or the location `search`? The latter modifies the URL, so it's only necessary if you want to build cross-application links (e.g. from one admin to the other). In general, using the location `state` is a safe bet. + ## The `` component Instead of a custom `Edit`, you can use the `EditGuesser` to determine which inputs to use based on the data returned by the API. @@ -2078,7 +2080,9 @@ export const PostEdit = (props) => ( ); ``` -## Displaying Fields or Inputs Depending on the User Permissions +## Recipes + +### Displaying Fields or Inputs Depending on the User Permissions You might want to display some fields, inputs or filters only to users with specific permissions. @@ -2140,7 +2144,7 @@ export const UserEdit = ({ permissions, ...props }) => ``` {% endraw %} -## Changing The Success or Failure Notification Message +### Changing The Success or Failure Notification Message Once the `dataProvider` returns successfully after save, users see a generic notification ("Element created" / "Element updated"). You can customize this message by passing a custom success side effect function as [the `` prop](#onsuccess): @@ -2193,7 +2197,7 @@ If the form has several save buttons, you can also pass a custom `onSuccess` or **Tip**: The notify message will be translated. -## Altering the Form Values Before Submitting +### Altering the Form Values Before Submitting Sometimes, you may want to alter the form values before sending them to the `dataProvider`. For those cases, use [the `transform` prop](#transform) either on the view component (`` or ``) or on the `` component. @@ -2246,7 +2250,7 @@ const dataProvider = { } ``` -## Using `onSave` To Alter the Form Submission Behavior +### Using `onSave` To Alter the Form Submission Behavior **Deprecated**: use the `` prop instead. @@ -2293,7 +2297,7 @@ const SaveWithNoteButton = props => { The `onSave` value should be a function expecting 2 arguments: the form values to save, and the redirection to perform. -## Grouping Inputs +### Grouping Inputs Sometimes, you may want to group inputs in order to make a form more approachable. You may use a [``](#the-tabbedform-component), an [``](#the-accordionform-component) or you may want to roll your own layout. In this case, you might need to know the state of a group of inputs: whether it's valid or if the user has changed them (dirty/pristine state). diff --git a/docs/List.md b/docs/List.md index a4117311d1c..09dfc8212e5 100644 --- a/docs/List.md +++ b/docs/List.md @@ -731,7 +731,7 @@ const PostList = props => ( The default value for the `component` prop is `Card`. -## Synchronize With URL +### Synchronize With URL When a List based component (eg: `PostList`) is passed to the `list` prop of a ``, it will automatically synchronize its parameters with the browser URL (using react-router location). However, when used anywhere outside of a ``, it won't synchronize, which can be useful when you have multiple lists on a single page for example. @@ -2004,7 +2004,7 @@ export const PostList = (props) => ( **Tip**: To let users hide or show columns at will, check the [``](https://marmelab.com/ra-enterprise/modules/ra-preferences#selectcolumnsbutton-store-datagrid-columns-in-preferences), an [Enterprise Edition](https://marmelab.com/ra-enterprise) component. -The `` is an **iterator** component: it gets an array of ids and a data store from the `ListContext`, and iterates over the ids to display each record. Another example of iterator component is [``](#the-singlefieldlist-component). +The `` is an **iterator** component: it gets an array of ids and a data store from the `ListContext`, and iterates over the ids to display each record. Other examples of iterator component are [``](#the-simplelist-component) and [``](#the-singlefieldlist-component). ### Body element @@ -2213,6 +2213,28 @@ export const PostList = props => ( ``` {% endraw %} +### Performance + +When displaying large pages of data, you might experience some performance issues. +This is mostly due to the fact that we iterate over the `` children and clone them. + +In such cases, you can opt-in for an optimized version of the `` by setting its `optimized` prop to `true`. +Be aware that you can't have dynamic children, such as those displayed or hidden by checking permissions, when using this mode. + +```jsx +const PostList = props => ( + + + + + + + +); + +export default withStyles(styles)(PostList); +``` + ### CSS API The `Datagrid` component accepts the usual `className` prop but you can override many class names injected to the inner components by React-admin thanks to the `classes` property (as most Material UI components, see their [documentation about it](https://material-ui.com/customization/components/#overriding-styles-with-classes)). This property accepts the following keys: @@ -2323,92 +2345,6 @@ export const ProductList = (props) => ( For this kind of use case, you need to use a [custom datagrid body component](#body-element). -### Performance - -When displaying large pages of data, you might experience some performance issues. -This is mostly due to the fact that we iterate over the `` children and clone them. - -In such cases, you can opt-in for an optimized version of the `` by setting its `optimized` prop to `true`. -Be aware that you can't have dynamic children, such as those displayed or hidden by checking permissions, when using this mode. - -```jsx -const PostList = props => ( - - - - - - - -); - -export default withStyles(styles)(PostList); -``` - -### With Custom Query - -You can use the `` component with [custom queries](./Actions.md#usequery-hook), provided you pass the result to a ``: - -{% raw %} -```jsx -import keyBy from 'lodash/keyBy' -import { - useQuery, - ResourceContextProvider, - ListContextProvider, - Datagrid, - TextField, - Pagination, - Loading, -} from 'react-admin' - -const CustomList = () => { - const [page, setPage] = useState(1); - const perPage = 50; - const { data, total, loading, error } = useQuery({ - type: 'getList', - resource: 'posts', - payload: { - pagination: { page, perPage }, - sort: { field: 'id', order: 'ASC' }, - filter: {}, - } - }); - - if (loading) { - return - } - if (error) { - return

ERROR: {error}

- } - return ( - - id), - currentSort: { field: 'id', order: 'ASC' }, - selectedIds: [], - }} - > - - - - - - - - ); -} -``` -{% endraw %} - ## The `` component The `<SimpleList>` component @@ -2736,7 +2672,9 @@ export const Basic = () => ( Check [the `ra-calendar` documentation](https://marmelab.com/ra-enterprise/modules/ra-calendar) for more details. -## Using a Custom Iterator +## Recipes + +### Using a Custom Iterator A `` can delegate to any iterator component - `` is just one example. An iterator component can get the data to display from [the `useListContext` hook](#uselistcontext). The data comes in two constants: @@ -2803,13 +2741,7 @@ export const CommentList = (props) => ( As you can see, nothing prevents you from using `` components inside your own components... provided you inject the current `record`. Also, notice that components building links require the `basePath` component, which is also available from `useListContext`. -## Third-Party Components - -You can find components for react-admin in third-party repositories. - -- [ra-customizable-datagrid](https://github.com/fizix-io/ra-customizable-datagrid): plugin that allows to hide / show columns dynamically. - -## Displaying Fields Depending On The User Permissions +### Displaying Fields Depending On The User Permissions You might want to display some fields or filters only to users with specific permissions. @@ -2858,3 +2790,73 @@ export const UserList = ({ permissions, ...props }) => { {% endraw %} **Tip**: Note how the `permissions` prop is passed down to the custom `filters` component. + +### Rendering `` With A Custom Query + +You can use the `` component with [custom queries](./Actions.md#usequery-hook), provided you pass the result to a ``: + +{% raw %} +```jsx +import keyBy from 'lodash/keyBy' +import { + useQuery, + ResourceContextProvider, + ListContextProvider, + Datagrid, + TextField, + Pagination, + Loading, +} from 'react-admin' + +const CustomList = () => { + const [page, setPage] = useState(1); + const perPage = 50; + const { data, total, loading, error } = useQuery({ + type: 'getList', + resource: 'posts', + payload: { + pagination: { page, perPage }, + sort: { field: 'id', order: 'ASC' }, + filter: {}, + } + }); + + if (loading) { + return + } + if (error) { + return

ERROR: {error}

+ } + return ( + + id), + currentSort: { field: 'id', order: 'ASC' }, + selectedIds: [], + }} + > + + + + + + + + ); +} +``` +{% endraw %} + +## Third-Party Components + +You can find components for react-admin in third-party repositories. + +- [ra-customizable-datagrid](https://github.com/fizix-io/ra-customizable-datagrid): plugin that allows to hide / show columns dynamically.