Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/tomashubelbauer/rbc-month-view-slot-grouping-by-type

An experiment in making RBC month view collapse items into represenatives per kind
https://github.com/tomashubelbauer/rbc-month-view-slot-grouping-by-type

rbc react-big-calendar

Last synced: 21 days ago
JSON representation

An experiment in making RBC month view collapse items into represenatives per kind

Awesome Lists containing this project

README

        

# React Big Calendar Month View Slot Grouping By Type

[**Online demo**](https://tomashubelbauer.github.io/rbc-month-view-slot-grouping-by-type)

For a project I am working on, I need to have an option to adjust RBC to display
a different version of month view than it does by default. I am going to see if
I can prototype a standalone solution to that before I attempt to integrate the
changes needed to support it in my project.

By default, the RBC week view looks like this:

![](week-view.png)

A month view containing the above week then looks like this:

![](month-view.png)

In my project, slots of the calendar can have multiple types and are color-coded.
In the screenshots above, which come from the
[RBC demo app](http://intljusticemission.github.io/react-big-calendar/examples/index.html),
there is only one kind of a slot (in blue color).

The change I am looking to implement entails grouping all items of the same type
to a single slot and displaying the number of items in the group on that one
slot.

RBC month view will display up to three slots in a day cell of the month view.
Coincidentally, I know I will only ever have three types of slots in my application.
So after grouping, I should never run into a case where the *+X more* links gets
displayed. I can hide that using CSS, then, but the grouping itself I think I
will be best off doing using a custom implementation provided to one of the RBC's
`components` prop fields. I am not sure which one, yet.

I do not want to change the data model in the component which displays RBC as it
would require hoisting the logic which deals with view change and data restructuring
for the given view up. This is something I want to avoid as I consider the current
calendar view to be an encapsulated concern of the calendar component which
components which host it should have no information about unless they want to
hook additional logic up to that which affects their own state and presentation,
not just the calendars.

I'm going to start by scaffolding a new Create React App instance from a template.
`npx create-react-app . --typescript`.

CRA being annoying, it will require `README.md` and the image files to be temporarily
moved to another directory and then moved back after it has scaffolded its files,
because it is not smart enough to realize they would not conflict.

After scaffolding the application and cleaning it up a bit (removing useless files,
rewriting questionable syntax to a more appropriate way the way I view the world),
we are ready to integrate RBC into the project.

`npm install --save react-big-calendar moment` is followed by
`npm install --save-dev @types/react-big-calendar` because Moment bundles its
own types in its package, as it should. We need Moment because RBC won't work
without a localizer and Moment is the easiest choice for a demo like this one.

Do not forget to include RBC styles in `App.tsx` so that the calendar is not
broken.

With a few lines we can demonstrate a week view calendar with a number of events
and upon switching to the month view we see the preview + expansion link mode.

---

The next objective is to demonstrate slots of multiple kinds. We can get rid of
the default `Event` type and provide our own to RBC as the component is generic.
If we keep the names of the `start` and `end` fields, we do not need to provide
accessor props for it, so we will do that to keep things simple.

To render the slots differently based on the `kind` property, we need to implement
a custom `eventWrapper` `component` prop field. Since the `children` passed in
the `props` of that wrapper component are absolutely positioned (at least in the
week view), we cannot just wrap them in another `div` with a certain background
color, as it would end up stacked at the top of the week day column.

Instead, we will make use of `React.Children.only` and adjust the props of the
passed-in `div` to extend its `className`.

We can now verify both the week view and the month view show correctly color-
coded slots.

---

To start figuring out how the month view could be adapted to group the slots and
display only one representative slot for each kind which has members, we will
take a look at the RBC documentation.

By that, I of course mean the RBC typings, as the actual RBC documentation for
the `components` prop is rather… scarce:

http://intljusticemission.github.io/react-big-calendar/examples/index.html#prop-components

So what do we have in the `component` prop typings?

https://github.com/DefinitelyTyped/DefinitelyTyped/blob/master/types/react-big-calendar/index.d.ts#L134

`event` and `eventWrapper` both deal with the individual slots. On top of that,
the field `month` for month view specific overrides does not include anything
which looks like it would be a component for the month view cell.

We do have `eventContainerWrapper` there so let's figure out what that does.
By cloning the sole child element and appending a marker class to its `className`
prop, we are able to identity that in the week view, this is a `div` which
encapsulates all the individual event wrappers in the day column.
Unfortunately this component is never called for the month view.

Applying the same technique (an extra, marker class) and querying the DOM with
`document.querySelectorAll`, we find that `dateCellWrapper` is a component which
gets called for each cell of the month view calendar. We can scope it to just it
by nesting it in the `month` field of the `components` prop.

Unfortunately the typings here lack this particular component underneath the
`month` field so we have to enter `any`-territory.

- [ ] Create a PR for `components`

We also find that this particular component doesn't receive any `props` containing
the models of the events to be displayed. It is merely a background cell which
we can adjust, but the actual contents of the calendar are still handled by
`event` and `eventWrapper` but not `eventContainerWrapper`.

Our best bet here is to completely disable the events by overriding `eventWrapper`
for month and returning `null` for each invocation. This will render the
calendar empty and we can then adjust the `dateCellWrapper` implementation so it
computes and materializes the representative events on the fly base on the calendar
`events` prop.

We also need to hide the *+X more* link, which luckily for us has a CSS class
`rbc-show-more` so we can hide it using plain CSS with a selector with higher
specificity than the one from RBC. We do not need to rely on extensibility points
provided by RBC, which we already know it doesn't have enough of to enable us to
hide this link using other means.

---

With all of the above done, we can now marvel at our creation:

![](final.png)

The published artifacts can be found on GitHub Pages:

[**Online demo**](https://tomashubelbauer.github.io/rbc-month-view-slot-grouping-by-type)