100DaysOfGatsby
Day 100
Apr 9 2020
Stick a fork in me, iām done! ā
I canāt quite believe it but for 100 days iāve written code and updated this blog post with what iāve been doing as part of the #100DaysOfGatsby challenge.
Itās been a great experience and iāve learned so much but to save you some time iāve summarized all the best bits in a separate post: 100DaysOfGatsby - TheRoundup
Ta ta for now!
Paul š
Day 99
Apr 8 2020
Still taking it easy.
This morning i only did a very small amount of work on my commercial portfolio
www.pauliescanlon.io and added some extra dates to the cards on the home page. Under
the hood iāve used the dataModified
field from frontmatter and am combining it with the date
field which allows me
to show when i started and ended each contract.
Iām not sure how closely employers look at these details but it does at least demonstrate how flexible gatsby-theme-terminal is. Those fields technically relate to blog posts but since they are exposed by a data component i can render whatever i want in the UIā¦ super cool! š
Iāve also been planning what life might look like after this challenge and stumbled upon this tweet from Fauna
Maybe Write With Fauna will be my new jam and iāve submitted a proposal for a Full Stack Gatsby App
I think itāll mainly focus on creating commenting functionality since this was one of the #100DaysOfGatsby challenges but none of the suggested options really worked for me, and with the work iāve done on skin-ui.com i have all the code examples needed to document how to make a Gatsby App with the following
- Gatsby App / With Theme UI (of course)
- Client only routes (Admin Page)
- Netlify identity
- Apollo/GraphQL api
- Fauna Database
It does sound like itāll be quite a lot of work and iāve never attempted to write anything like this before but iāve rather enjoyed blogging EVERY DAY so maybe iāll be able to handle it.
Other life work is beginning to resume to and iāve made a list of jobs that iād like to apply for, they are all top tier tech firms so iām expecting many rejections but if i fire off enough applications hopefully one of them will come to fruition.
Naturally Gatsby is on this list but sadly they donāt have any open positions thatās iām suitable for but i think i will submit a āDream Jobā application anyway. Fingers crossed ay š¤
Tomorrow is the final day and iām all but out of things to write about so i think tomorrow will be a rundown of everything iāve done over the past 100 daysā¦ i know already itās a lot but iāve enjoyed every minute of it and if youāve been reading along i hope you have too!
Day 98
Apr 7 2020
This morning i was doing a bit of house keeping on this blog.
Previously iād been dumping all manner of @theme-ui/components in my .mdx
so i could give the data components a test
drive with some of my real data but unfortunately this led to a lot of duplication.
Iāve now abstracted some of the charts and the card component from raw @theme-ui/components in .mdx
to composed React
components in .js
It makes it much easier to see whatās going on in the .mdx
files and means i can re-use those components in any pages,
or posts since everything within them is now encapsulated.
The home page of this blog is a bit gross so i might create a separate dashboard or statistics page to showcase all the data related stuff and leave the home page to just display some intro text and the latest blog posts.
I noticed this tweet today from @AskGatsbyJS
ā¦and whilst Component Shadowing is an important concept with Gatsby themes i feel my theme gatsby-theme-terminal side steps the whole requirement to shadow components because there are no components to shadow.
Everything you see in gatsby-theme-terminal is something from @theme-ui/components and since the only ācomponentsā that exist in the theme are data components you can render your UI in any way or style you likeā¦ maybe i should write a blog post about what iām calling Zero Component Themes
Day 97
Apr 6 2020
Another day of data!
This morning i had another crack at the <SourceWords />
component in
gatsby-theme-terminal and have exposed a new array of data,
itās called wordCountByMonth
and yeap it does what it says on the tin.
Itās very similar to sourceMonths
and provides a nice way to visualize how many words have been written in any given
month.
Hereās another lame demo.
<SourceWords>
{(sourceWords) => {
const currentYear = sourceWords.wordCountByMonth[sourceWords.wordCountByMonth.length - 1];
return (
<Box>
{currentYear.map((month, index) => {
const { name, words } = month;
return (
<Box key={index}>
<Text>{`${name} | ${words}`}</Text>
</Box>
);
})}
</Box>
);
}}
</SourceWords>
Thereās a better looking bar chart version of this on the demo site if youāre interested.
Day job wise iāve been really getting stuck in to Testing Library.
Now that iām over the whole needing to add a data-testid
to dom elements it is pretty cool to work with, i expect iāll
have a few more things to learn before i really suss it out and i havenāt written any tests that require mocked api
responses so thatāll be fun when i get to it!
Day 96
Apr 5 2020
Keep it simple stupid.
Thatās how iām planning on winding up this challenge, thereās no point in starting anything new if iām not going to be able to finish it off and so close to Day 100 iām just combing over the work iāve done over the last 3 months and tweaking where needed.
Today iāve released a new patch for gatsby-theme-terminal
which now includes a <SourceWords />
component.
You can see a demo of it here.
It works in much the same way as the other data components by only being responsible for returning data, itās up to you the theme user to determine what UI element youād like to render.
And for my next trickā¦ because this blog is built on
gatsby-theme-terminal i can use the <SourceWords />
component right here in the .mdx
<SourceWords>
{(source) => {
const { wordCountTotal, wordCountAverage, wordCountHighest, wordCountLowest, timeToReadTotal, timeToReadAverage } =
source;
return (
<ul>
<li>{`wordCountTotal: ${wordCountTotal}`}</li>
<li>{`wordCountAverage: ${wordCountAverage}`}</li>
<li>{`wordCountHighest: ${wordCountHighest}`}</li>
<li>{`wordCountLowest: ${wordCountLowest}`}</li>
<li>{`timeToReadTotal: ${timeToReadTotal}`}</li>
<li>{`timeToReadAverage: ${timeToReadAverage}`}</li>
</ul>
);
}}
</SourceWords>
Pretty lame list right, but if you like donut charts checkout the demo
Day 95
Apr 4 2020
I spent the morning investigating why gatsby-source-contentful
was throwing errors with my blog but after a few hours
i took a step back and thought about if i really need a CMS in my blog.
My motivation for looking into gatsby-source-contentful
was because itās one of the challenges and i suppose i do have
a requirement for both my blog and my site to share some of the same content, but since this is a source plugin iād
still need to build and deploy both in order to keep them in syncā¦ and if i need to do that itās probably the same
amount of effort involved in just copy and pasting the content they share.
I did then look in to perhaps making a REST-ful request to contentful which would remove the need to re-build and deploy but it would also remove the static data which would probably have a negative effect on SEO but i was curiousā¦
I got all the way through the docs, set up a nice component that will make a request with the correct space
id and
bearer token
and then moved on to work out what the GraphQL query would beā¦ i then discovered that on the free plan
you canāt make GraphQL queries so that was the morning pretty much wasted š
So, instead of messing around with things i donāt really need iāve messed around with some things i do.
There were a few issues with my theme gatsby-theme-terminal which i wanted to look at.
The first relates to the prev
and next
links that were at the bottom of each āsourceā page butā¦
gatsby-theme-terminal has the ability to query from any local
data source and although in āyour siteā you can filter and sort however you want the GraphQL query thatās inside the
theme is still in charge of the source nodes and is whatās deciding what in fact is prev
or next
.
For example if you have two sources and filter out one in your MDX the underlying nodes wonāt know about the filter and
might try to link to a prev
or next
āsourceā that isnāt relevant ā¦ so iāve removed this functionality ā¦ sorry
about that.
There was also an issue with the <SourceDays />
component where iām performing some calculations based on what date
is found in the frontmatter eg.
new Date(item.node.frontmatter.date).getDay();
Unfortunately iād forgotten that arrays start at 0
so all days where shifted forward by one! changing the above line
to the below fixes that problem.
new Date(item.node.frontmatter.date).getDay() - 1;
I also added a word count, time to read and author to the post layout template. These fields were already there in the frontmatter i just forget to add them to the UI.
Iāve had another scan over the challenges and thereās not really anything that i havenāt covered in one form or another so iām hoping iām still on track. I appreciate these are just pointers to help you get started with Gatsby but based on the work iāve done with my themes, plugins and more recently with skin-ui.com i feel iāve got a solid understanding on what you can achieve with Gatsby.
Itās not long until the end now and i gotta be honest iām feeling a little burned out. I donāt regret committing to writing a post every day, itās kept me motivated but it has been a challenge. Every day i have to think about something to do so that i have a reason to update this post which means iāve actually got to do at least one thing every day.
Iām quite looking forward to a day off!
Day 94
Apr 3 2020
Challenge acceptedā¦ challenge failed š
Today was a good opportunity to catch up on the actual #100DaysOfGatsby challenges and the headless CMS one seemed quite pertinent given my recent shift over to one theme two sites. Both my commercial portfolio and blog have the same intro header, and it would be nice if they both pulled that content from a CMS so as and when i update it i only have to do it the once.
The recent challenge to use gatsby-source-contentful
fits my needs so this morning i set about adding the source
plugin to this blogā¦ but sadly ran in to issues immediately. There seems to be some kind of clash going on with my use
of onCreateNode
in the theme. I got as far as console logging out a few values to see if i could work out whatās going
on but i feel like itās a job for the weekend now.
The day job was pretty intense today. Iāve recently completed quite a large feature which involves a number of form elements and some odd / complicated interactivity that results in conditional rendering.
I did think during development that it would be hard to test.. and i was right.
For example, in one part of the feature thereās a form that you canāt submit until all five text inputs have a value, but the last three inputs are only revealed if one of the first two has been touched and has an input value.
The test for this is madā¦ iāve had to use async and await to first fireEvent from one of the first two inputs and only then can i assert if the remaining three are visible.
Iām still not quite there with the test as i then need to assert that the button is disabled until the last three and the first one that wasnāt originally touched also have input values.
Just writing this out is complicated, imagine how odd this form will be to use, those poor users!
Anyway iāve learned a lot about React Testing Library today and am quite liking working with it. Iām still not quite ready to ditch the Enzyme love but maybe i can just like both equally and leave it at that.
Day 93
Apr 2 2020
Never deploy in a hurry!
Thatās the main take away for today. I was working on two new features for skin-ui.com these were:
- Allow the admin user write access to all themes
- Add a save icon button to the editor toolbar
Admin user access was a carry over from yesterday and i still canāt really work out why my environment variables in Netlify are undefined when i deploy but in any case if i run locally and log in as admin i can perform admin operations, one being write access to other users themes. This functionality is possible because i conditionally render the settings icon button if the ālogged inā user id matches the user id from the theme saved in the database OR if it matches the user id of my admin login.
state.isUserOwner || (state.user.id === process.env.SKIN_UI_USER_ID && ( // render UI)
Once this was working i set about adding a save icon button to the editor toolbar, this is to improve the UX. Before this you had to open the settings panel to get at the save button which i thought was a bit shitty.
butā¦
in my haste to add the save icon button i forgot to add the condition, and deployed. It wasnāt until a bit later when i noticed the default theme had changedā¦ i thought ah! iāve been hacked!
but nope, i logged in with my regular user details and viewed the default theme and sure enough the save icon button was there meaning anyone could just save over the top of the default theme!
I was in such a hurry to deploy this new feature this morning before i started work that i hadnāt fully tested the functionalityā¦ a mistake i wonāt make again!
Alsoā¦ if iād actually had unit tests this would have probably been caught before i even committed the changes.
A lesson learned.
On to the day job.
Nothing really new to report, just carrying on with a customer review feature, got all the main UI parts built and am now moving on to improving my tests.
Day 92
Apr 1 2020
Notice anything different?
If youāre one of the 3 people (including my mum) who have actually been reading this you might notice the site has completely changed. Yup! www://paulie.dev now looks very similar to www.pauliescanlon.io
ā¦ and thatās because finally after almost a year iāve set out what i wanted to achieve and the reason i got started with Gatsby in the first placeā¦ Themes
For a long time iāve wanted two sites, one for my commercial portfolio and one for my blog, but i didnāt want to build and maintain two separate entities.
With Gatsby Themes this is no longer an issue!
Iāve built two themes over the past year but gatsby-theme-terminal is the one iāve put more of my learnings in to and because itās really flexible itās easy to create different layouts and query different parts of the file system. More on that here
So there you have it, if you have a requirement like i did, Themes is the way to go.
Today iāve also managed to squeeze in some Skin UI work and am now handling syntax and theme object errors more
gracefully. I do have some problems with creating an admin user as Netlify seems to be ignoring my .env
variablesā¦
this one requires more investigation.
The day jobā¦ a pretty good day today mainly chore
focussed as i was asked to convert the code base over to
TypeScript š
Storybook was the killer here, Storybook 5 is brills but it still took a little while to configure the webpack config to get the prop tables to populate using react-docgen.
After that was done i was back on to Formik forms and wrestled with Radio buttons but got there in the end.
Thatās me for today, cheerio!
Day 91
Mar 31 2020
Iāve become a night owl! š¦
After i posted yesterday, and i donāt know why but i decided to switch this blog over to using my other theme gatsby-theme-terminal, it was a pretty easy switch but i wanna make a couple more changes before i deploy so weāre stuck with gatsby-theme-gatstats for the time being.
Then i went to bed and couldnāt sleep because i kept thinking about an enhancement i wanted to make to skin-ui.com
Thereās a showcase section and it wasnāt really showcasing the themes because i
wasnāt able to pull the theme_object
down from Fauna using an indexes
lookup.
I did some investigation and theme_object
was returning null
which i found very odd because if youāre on the Editor
page it pulls the theme_object
down ok.
Iām not sure if iām doing something wrong with the query but if i query all themes using an indexes
lookup rather than
by a specific id
i get null
I found that if i save the data as a string
rather than an object
in both queries theme_object
comes back okā¦ so
thatās what iāve done.
No thereās some pretty sweet looking Empty States when you visit the showcase section and they represent the colours used in each of the themes that users have saved.
On to the day job.
Today iāve been building out a customers reviews section, nothing new here, just more Formik and using Theme UI to style stuff.
Tomorrow iāve gone a bit more to do but so far so good. Once the UI part is done i hope to be investigating a source plugin which will be great as itās not something iāve done with Gatsby plugins yet so iām excited about that.
Day 90
Mar 30 2020
Back to work!
Early start this morning and i made a few changes to skin-ui.com and exposed a little more
theme information on the your themes section. I exposed the user.id
and the theme.id
. Not huge changes but if i
need to manually delete something from the database if a user emails me iāll be able to find the entry more easily if
they provide one or the other id.
Next up, the day job. Todays task mainly centered around how to build a UI with Theme UI and the code base iām working on has the maddest looking theme object iāve ever seen. It looks like all the styles from Tailwind but in a Theme UI shape.
This is not to say Tailwind isnāt hugely powerful but Tailwind and Theme UI are two very different beasts.
Iāll try and explain, first with one example key from the Theme UI spec.
export default {
...
space: [4, 8, 16, 24, 32, 48],
}
You can see from the space
key that there are only six different sizes, this is so when developing a UI if you need to
add a margin or padding to an element you can use one of these options. Most likely in a good design system itāll be
space[2]
or space[3]
which gives you a nice amount of space of 16px
or 24px
depending on which one you choose.
If you have a theme that has something like this;
export default {
...
space: {
px: "1px",
"0": "0",
"1": "0.25rem",
"2": "0.5rem",
"3": "0.75rem",
"4": "1rem",
"5": "1.25rem",
"6": "1.5rem",
"8": "2rem",
"10": "2.5rem",
"12": "3rem",
"16": "4rem",
"20": "5rem",
"24": "6rem",
"32": "8rem",
"40": "10rem",
"48": "12rem",
"56": "14rem",
"64": "16rem",
},
}
What youāve done is ignore any kind of design system and are saying āuse any padding or spacing you like because thereās loads to choose fromā, which completely defeats the point of a theme.
If youāre going to take this approach you might as well not have any options in the theme and allow anyone developing
the UI to just hard code px
or rem
values wherever they like.
The point, in my opinion of a theme is to restrict css properties to only a few options. The less time you spend thinking about how much padding or margin you need the more time you get actually developing features.
Also, think about how much time youāll spend on PRās, how on earth are you going to know what all those values are and how will you know from looking at the code if something has the correct amount of padding or margin.
To note, this is one of the smaller keys. The sizes
key has 58 possible options and colors has 93!
What could you possibly need 93 colors for?!?!?
My main point being, if youāre exposing that many options in a theme which is supposed to reduce the css complexity you might as well ditch Theme UI altogether
Madness i tell thee, absolute madness!
Next up was forms i used to hate forms in Reactā¦ but then Formik came along and boom! everything changed. Combine this with Yup and youāve rock solid forms which manage their own state and really powerful validation without having to really think about anything.
I believe that like Theme UI, Formikās approach is to remove those lower level problems so that you can start exploring higher level abstractions, and in both cases if you use the technologies for the purpose theyāre were intended you can dramatically speed up UI development. Sadly iāve seen this misunderstanding all to often and it always ends badly.
Day 89
Mar 29 2020
Today i feel good!
Another early start this morning and i was determined to solve this UX Navigation problemā¦ to be honest iām not 100% happy with the solution but you can now navigate around skin-ui.com with a reasonable amount of ease. The experience is a little better if youāre logged in but maybe thatās ok. If youāre just an idle browser then its probably forgivable that you have to work a bit harder to navigate.
Aside from that iāve also tackled a few bugs and put the newly deployed features out to be beta tested by trused Gatsby dev chums Scott Spend, Richard Haines and Eric Howey who to note, iāve all met through Twitter and because we all love Gatsby.
With the new release deployed skin-ui.com is now pretty much a fully functional application.
There are a few issues and enhancements i need to make but iām happy for it to be out there. That said i now have THE FEAR.
Users can now sign up and save their themes which means i now have dataā¦ what do i do with data, what do i do if thereās a security issue and what happens if i need to delete data or change data thatās already been stored in Fauna?
If youāre reading this and have any advice please do get in touch. @pauliescanlon
Day 88
Mar 28 2020
Not the most productive dayā¦ but
Iāve now completed the Gallery / Showcase section on skin-ui.com. I can now pull down all saved themes from Fauna.
The response is paginated but i think i need to investigate how that works at some point, but since iām not expecting a Facebook situation when i release this pulling all themes down at once (all 5 of them) probably wonāt cause anyone any issues.
My biggest issue now though is the navigation. Once youāre in the Editor view thereās no real room for a top nav or a side nav so i think iāll have to go back to the UX drawing board and see if i can figure out a way to maybe keep the top nav so you can navigate between the Gallery / Showcase and the Editor and if youāre logged in Your Themes and keep the Theme title and Theme description at the top of the screenā¦ should be fun!
Day 87
Mar 27 2020
My morning started well, i got up early to look at my UX / Dropdown issue with
www.skin-ui.com and i tried the
reach/menu-button again and got the same āCannot read property
āownerDocumentā of nullā error so i gave up and wrote my own Dropdown component. It was quite an interesting process as
i discovered something new about useEffect
, maybe iāll write that up at some point.
Now with this UX / navigation issue ironed out i can navigate around the app and login / out easily.
Iām gonna spend the weekend working on the gallery pages which gets me very close to the finish line.
The working day wasnāt entirely Gatsby based today but i did run in to a small problem with Jest, React Testing Library and the Theme UI providerā¦ oh and Storybook was giving me a lot of problems with Gatsby links and GraphQL queriesā¦ hereās some comments on an open PR Docs/visual testing with storybook updates
But todayās main battle has been with massive api responses.
Imagine this scenario: You hit an api end point and it returns an absolute ton of data, probably 2% of it useful to drive the UI.
Do youā¦
- A. Reduce it / format it once the response comes back, and only then pass it on to your UI / Context.
- B. Just pass the entire object on to your UI.
If A, do you find it speeds up development as data is clean and easier to type with fewer obscure areas that require conversations.
If B, do you find that a number of smaller components need to run their own operations to wrangle the data in to a format that is helpful, do you find theres a performance impact when doing this?
My preference is A, i think itās more prudent to think ahead rather than re-factor after.
Have your sayā¦ tweet me @pauliescanlon
Day 86
Mar 26 2020
Itās been a day of many halves!
I got up early and tried to implement a user dropdown kind of menu thing on skin-ui.com using @reach/menu-button which kinda worked but i did get some weird āCannot read property āownerDocumentā of nullā error so i gave up and decided to plan my work today.
My main tasks on the new job are to refactor the UI and make a few decisions regarding the following.
- What to name things
- Where to save things
- How to export things / named or default
- How to create stories for things
- How to test things
- How to work round the Gatsby āpagesā issue - more on this later
Iāll start with how to name things. This is my preferred approach.
directory: components/SomeComponent
// components/SomeComponent/SomeComponent.js
import { React } from 'react';
export const SomeComponent = () => <div>SomeComponent</div>;
A few notes on the above; The component is a named export, itās named the same as the directory itās within and wherever possible it uses an implicit return.
// components/SomeComponent/index.js
export { SomeComponent } form './SomeComponent'
This is a named export of a named export but from an index.js
in the same directory.
The usage then becomes;
// some other file
import { SomeComponent } from '../components/SomeComponent';
Then at some point you might have Storybook so youāll probably want SomeComponent.stories.js
and really there should
be tests, SomeComponent.test.js
and maybe even _snapshots_
All of that together might look something like this:
|-- src
|-- components
|-- SomeComponent
_snapshots_
SomeComponent.js
SomeComponent.stories.js
SomeComponent.test.js
index.js
In my opinion itās kind of crucial to get this locked down early on in a project.
Iām not sure why i care so much about these small details. I think it might be because when i first started doing this, Agile didnāt exist and yet we still got shit doneā¦ weird
I think it might also come from my family of tradesman where they will strictly employ a
Measure twice, cut once
approach to work. And just because on computers you and edit undo doesnāt mean you should.
Iāve re-imagined this phrase as;
Do it nice, not twice!
You can have that for free team!
Tomorrow iāve got more of the same but am really looking forward to developing a rock solid testing strategy using React Testing Library, once i have my tests in place iāll feel a bit more confident refactoring the UI.
I reckon with good tests you can re-build anything!
Day 85
Mar 25 2020
The first part of this is an update from last night. I just about had the brain power to tackle the Fork functionality i needed in www.skin-ui.com and am pleased to report that step one is done. I can now hit the Fork button and copy the default theme, give it a random name and push it up to Fauna. From here you can amend as normal.
I have been a bit of a hack monster in the process and now have loads of GraphQL types floating around. Thereās currently only three operations that a user can perform in the application and all, i think should post and return the exact same data shape so i need to have a bit of a tidy up and work out if i can just have one type for all operations.
I also got up early this morning to tackle the delete functionality and.. wallop thatās now done too!
I now have Create Read Update and Delete!
The issue i have at the moment is that in my haste to ship iāve neglected to write testsā¦
not even one! š
For UI stuff iām pretty ok and have confidence in my ability to have visually covered off all states the application could be in, but for the data iām not so sure.
Apollo and Fauna are all totally new to me and whilst every thing appears to be working thereās a chance that itās not, and when you start allowing users to push to a database iām worried that it iāve messed something up iām gonna end up with data in Fauna thatās incorrectā¦ then what do i do, delete it and piss off my users?
I might ask for some help here peeps. I could use Netlifyās preview deploy link and give it to a handful of people to test out and can ask for feedback but anything they do is still gonna end up in the database and as mentioned i might need to delete it later.
Iām gonna sit on this for a while as i have other areas i need to focus on.
I need to build a kind of users themes page and a global themes page and still need to check the sign up process works smoothly.
I have 15 days before the #100DaysOfGatsby challenge is up and iām starting to feel the pressure. Iām really diggin my new job and not sure how much time i can commit to skin-ui.com over the next few weeks.
Iām also really getting in to drinking at home which is a major distraction!
Day 84
Mar 24 2020
Second day on the job and all is still going ok.
My task for today was to implement storybook. I love Storybook and it plays a huge part in my day to day work when developing component libraries but iāve only ever implemented it with a Gatsby project once before and that was for my first theme gatsby-theme-gatstats
Like gatsby-theme-gatstats the project iām working on also uses Theme UI and thereās a couple of little tricks one needs to implement to ensure all Theme UI styles get passed on to the component when viewed in Storybook.
The below is how you set up Theme UI in Storybook 5.3
// .storybook/preview.js
import React from 'react';
import { addDecorator } from '@storybook/react';
import { ThemeProvider } from 'theme-ui';
import yourTheme from '../src/gatsby-plugin-theme-ui';
addDecorator((storyFn) => <ThemeProvider theme={yourTheme}>{storyFn()}</ThemeProvider>);
The addDecorator
works a bit like wrapRootElement
does in Gatsby and provides a nice way to wrap all Storybook
stories with anything you like.
Adding fonts is, according to the Storybook docs pretty straight forward however, nothing i did today using the
preview-head.html
seemed to work so i opted for using the <Global />
component from @emotion/core
// .storybook/preview.js
...
import { Global, css } from "@emotion/core";
addDecorator(storyFn => (
<Fragment>
<Global
styles={css`
@font-face {
font-family: "Some Font";
font-weight: 400;
src: url("/fonts/Some-Font.woff2") format("woff2");
}
`}
/>
<ThemeProvider theme={yourTheme}>{storyFn()}</ThemeProvider>
</Fragment>
));
This is assuming you have the fonts located at public/fonts
and have the -s
flag in package.json
pointing at
./public
"scripts": {
"storybook": "start-storybook -s ./public -p 6006",
"build-storybook": "build-storybook"
}
After iād sorted that out i could now start creating stories for all the components in the code base.
Iāve done this a lot in the past but did want to experiment with pure .mdx
based stories and whilst they work just
fine they require you to import a number of things to get stories to display with props and previews.
Using the .js
method or CSF Component Story Format
in my option is just better and leaves less to get wrong. After all the whole point of having Storybook manage the
documentation is so any engineer coming to the code base can quickly and easily discover how a component works, what
props it needs etc and if you start to be a bit slack with the documentation the whole system breaks down.
I then moved on to unit tests and Alex has already started to implement @testing-library/react.
Iām gonna reserve judgement on this for the moment, but not because i donāt believe itās every bit as good as everyone says it is but because itās new to me and i need to work out if my battle hardened component testing pattern and CSF strategy works as well with Testing Library as it used to with Enzyme
So far today iāve only written a handful of tests and they were pretty straight forward. I should imagine in time once iām a bit more familiar with the methods iāll get rolling but i still gotta learn and get my head round it.
Thatās all for today teamā¦ keep it Gatsby! šŗ
Day 83
Mar 23 2020
This morning i was up earlyā¦ and for good reason. I started a new contract today š.
Theyāre a US based firm and are cool with me being fully remoteā¦ The stack?ā¦ Gatsby and Theme UI, how cool is that!
In fact itās through the #100DaysOfGatsby challenge that Alex Luong found me and very kindly put me forward to work on a project with him.
Itās so cool that itās happened this way and really makes it all worth while. Iāve been absolutely slogging a way of this stuff for the past 83 days and and itās finally paid off.
Right ho, thatās the business stuff outta the way, back to my early start.
From around 5.30 this morning until 9.00 am when i started work i was investigating this pesky Object to String issue i was having.
The theme object is stored in Fauna as an object but i canāt type the response required by GraphQL as an object at this stage as thereās too many key value pairs. Instead iāve typed it as a string which is what GraphQL is expecting.
Once i have the theme object as a string iām storing it in application state and passing it on the preview which has a
Theme Provider and uses JSON.parse
which is how the styles get applied to the markdown and mdx.
In the Editor however i need to keep it as a string so it can be passed on to CodeMirror but i need to remove some of
the quote marks around the key value pairs so it appears as JavaScript
and not JSON
ā¦ but, when CodeMirror passes
this back via the onChange
i intercept it and pop the quote marks back in before storing back in application state.
This means that each of those special requirements are handled in the places they are needed and wonāt affect how the data is stored in application or in Faunaā¦ bit of ball ache but itās working now.
Iāve checked that saving to Fauna still works and everything is looking pretty sweet. With my test logins i can edit the object and save it back to my account
The next step is for new users to be able to Fork the default theme and to save it to their own accountā¦ i also need to think about some kind of users themes pages so you can browse and edit your own themes.
I think iāve got everything in place to get this going but at the moment iām not really sure what extra or temporary things iāll have to add to allow the Fork to really work. I might need to auto generate a temporary made up name just so if you save it before youāve re-named it youāll see it in your users themes area.
Itās a bit of a head scratcher but i think iāve got a planā¦ just need some time to get it all up and running.
Day 82
Mar 22 2020
Motherās Day and Ma comes first. Iāve only done a few hours work today for obvious reason but i have sorted the
useMutation
callback. After an update is applied thereās an onCompleted
arg that can be used to set state or
whatever you need to do once a 200 response comes back. Currently iām using it to hide a spinner that is set to visible
when the save button is clicked.
This i think completes the update journey but yesterdaysā string to json to string parsing issue is still very much in play. Iām gonna need a good few hours to work out what to do with that which i donāt have time for right now so itāll have to wait.
Day 81
Mar 21 2020
Only a few hours work this morning, iāve got plans that require me to leave the house so iām a bit short on time.
What i have managed to accomplish this morning is cracking how to use useMutation
/ @apollo/react-hooks
and can now
pass values from my form back to the Fauna Database via my Apollo / GraphQL api.
This however has un-earthed another problem which i need to un-pick a bit but in short itās related to the transformation iām performing on the Theme UI object. Hereās a list of data types the object currently exists in and where
- Local State : string (modified with regex to remove quotes)
- Application State : object (modified with regex to add quotes)
- Database : string has quotes
- Return type from Database : string has quotes
I think i need to workout how to keep the type the same in Application state and in the Database and leave the local state to manage the funky transformations as they are only required so the object displays nicely in CodeMirror.
Iām not even sure at this stage if i could type the Theme UI object as an object because GraphQL would want the full
blown list of all object keys and their types. Unfortunately it doesnāt look like GraphQL has an any
type like
TypeScript does so itās probably better to type the Theme UI object as string when iām passing it around. It has to be a
string in Fauna anyway so i think thatās pretty much how itāll have to be.
This is quite the ball ache but i suppose it makes sense to keep the type the same in and around the application and in the database and only mutate it where i absolutely have to.
Probably a job for tomorrow now but iām pleased iāve made some progress albeit small.
Day 80
Mar 20 2020
Iām still sloggin a way on www.skin-ui.com and think iāve resolved the
react-codemirror2 syntax try
/ catch
bug. If a syntax error
occurs while editing the Theme UI object iām catching it before it hits the application state and blows up. Iāve had to
implement a 2nd part of local state to manage this but the application doesnāt appear any slower when using it so that
all good.
Iāve now moved on to managing mutations
with Apollo and GraphQL and have come a bit unstuck. As with any new
technology you learn from your mistakes and i think iāve over complicated the way context and database state work, and i
think iām gonna run in to a problem with the database callback when an update is issued. Somewhere along the line iāll
need to re-fetch the newly saved data, my current thinking is to update context which will causes all the right parts of
the app to re-render.
But first things first and i need to get to grips with useMutation
, once thatās done i think iāll be in a better place
to solve some of the UX issues.
I also experimented with Formik which iāve used on a few projects but i donāt think itās really required since i have little to no validation requirements and the value of my inputs all come from the backend.
Itās all part of the process so whilst today has been a challenge iām still making progress.
Day 79
Mar 19 2020
Today hasnāt been all that productive.
It started off well and iāve worked out some of the is the logged in user the theme owner and therefore should the primary call to action be to save the theme or to fork the themeā¦ but then i got caught up in a react-codemirror2 bug chasing nightmare.
I think i mentioned this before regarding some mega regex
that iāve implemented that converts an object to a string
and removes the quotes and then thereās some more regex
to parse the string and re-insert the quotes, which is still
required but i struggled to work out at which point in this transformation should the object be pushed to the database.
Iāve also run in to a few problems with Theme UI components, it appears Buttons error if an invalid string is passed as
a hex. This causes the whole app to break so iāve temporarily removed the <Button />
from the .mdx
preview.
Iāve also got a few issues with how to manage try
/ catch
on the actual object before i save it to state and thereās
also some debounce-ing issue going on which iāve implemented since the switched to a controlled version of CodeMirror.
I think theyāre all resolvable itās just today iāve un-earthed multiple issues and need to pick which one to tackle first.
Either way these all jobs for tomorrow now.
Day 78
Mar 18 2020
react-codemirror2 saves the day!
I can now pass a theme object from my new Apollo / GraphQL server straight to the code editor and itāll style the markdown as it did before. You can still update the theme using the editor and once i get the U of CRUD sorted out youāll be able to save your theme to the Fauna database.
Iām still thinking about some of the UX issues iāll face with this application. For instance if you where to share a
link with someone they could view it but they might also want to fork it and continue to develop it under a different
id. In this case the new Drawer component which houses the save button would first need to display a fork button. I
think i can do this by checking if the user_id
of the theme matches the logged in user_id
if it does then itās safe
to assume that a save button is needed in the UI and if it doesnāt itāll be a fork button in the UI.
This Drawer will also house a delete button and all the fields required to re-name the theme and or update the description and a pair of radio buttons to identify if the theme is a light or dark themeā¦ this i hope will come in handy later in the gallery for filtering purposes.
Iām still wondering what to do about private themes, i had thought about a premium pricing model but the thought of hooking up a payment system at this point is a bit too much to take on. Iāll leave this field in the db but probably wonāt use it for a while.
Iām posting a bit early today because iāve got some life admin to take care off so this is pretty much me for today.
Day 77
Mar 17 2020
At what point can i claim to be full stack?
Iāve had such a belter of a day and got the first half of the Apollo / GraphQL / Funa / Netlify Identity work done.
Iām not even sure if i can fully recount what iāve done today so iāll try putting it in a list.
- Shift Context Provider, Netlify Identity and Apollo Provider in to
wrapRootElement
- Use
gatsby-browser
as import forgatsy-ssr
so the above works in prod and dev - Connect Apollo server to Fauna
- Setup multiple resolvers to query Fauna via GraphQL
- Implement initial functionality to query Fauna served theme object and re-hyrdrate CodeMirror instance / Theme UI Provider.
Actually on that note, the version of CodeMirror iām using doesnāt like to be controlled via state, it appears that once
it loads a string
to use as a starting point for the code no state changes to that string will cause a re-render.
There is another version of CodeMirror out there called
react-codemirror2 which can be setup to handle changes in state but i
played with this a little while back and there was an issue but i canāt remember what the problem was so thatās a job
for tomorrow.
There are a few user journey / functionality things iām trying to solve. These are all based around user sign up / sign in and how iāll handle public / private themesā¦ i might not initially provide the option to make a theme private but i will of course need to restrict saving updates to a theme unless you were the user that created it.
Thereās also quite a bit of work to do around showcasing all the themes that are saved to the database by all users. Iād like a kind of gallery where you can view them or maybe fork them and save them to your own account.
Itās difficult to think in MVP terms because as soon as you introduce user sign up / sign in youāve got a ton of work to do around, sign up verification journeys, forgot password journeys and of course everything to cover CRUD functionality.
Currently iām aiming to have all this finished by the final day of 100DaysOfGatsby butā¦ eeek iāve still got a lot of problems to solve and still have a lot of things to learn so itās a pipe dream at the moment!
Iām still enjoying myself though and itās pretty dope building the full stackā¦ not sure when iāll get the opportunity to create an Apollo / GraphQL / Fauna / Netlify Identity stack again!
Day 76
Mar 16 2020
Full stack attack!
Itās been a day of ups and downs. Iāve been implementing Apollo and got the basic
client Netlify function and a local users array all working and i am able to query the new GraphQL api when running
gatsby develop
Butā¦ thereās some funky stuff going when gatsby build
this is no doubt something to do with fetch
not being
available in Node so iām trying to test and eliminate each of my suspicions one by one.
Iām currently wrestling with client side routes, if i can get this working i should be able to confirm that it is in fact a Node / Fetch issue but thereās quite a few other things i could try but need to do it methodically or else iāll end up chasing my tail.
It is fun learning something new and i enjoy the process but i feel like iāve put myself under pressure to perform as iām writing about my progress each dayā¦ and today iāve not really made great progress.
Butā¦ tomorrow is a new day so with a good nights sleep iāll be ready to tackle it again tomorrow.
Day 75
Mar 15 2020
Not an overly productive day today but i am pretty impressed with last nights post and how well it reads considering my state when i wrote it.
Today iām mainly beein trying to get a plan of action together about how to handle database access in www.skin-ui.com. I think Fauna looks like it will be a good fit but this is all new to me so iām trying to work out how best to structure my repo and where to deploy Netlify functions from, also where in the repo should i build the GraphQL API layer and how is an API request handled dependant on if a user is logged in or not.
Thereās some good tutorials by Chris Biscardi on egghead.io which give me a nice overview on how all of these technologies come together but i feel like i need to be a bit more research before i start hacking something together.
Iām also aware iāve kind of fallen behind on the challenges, although i have seen one email about connecting to a CMS which i donāt really need for any of my current projects. If i get really stuck with the Skin ui database stuff maybe iāll āpivotā to the CMS challenge so i can feel good about my self again.
Day 74
Mar 14 2020
Todayās update has to be short and sweet because iām on the wrong side of 8 pints of sweet sweet Ale. I started the day by continuing to faff around with Context and have created a new Drawer component to expose additional functionality if you sign in / up
Iāll leave it there. šŗ
Day 73
Mar 13 2020
Mooooore context š«
Yeap, today iāve continued to refactor www.skin-ui.com and now, wherever possible state is managed by React context. Thereās only two very small parts that remain as local state and the app seems so much more responsive. Prop drilling can really mess with an application and passing props through components to reach deeply nested components does really affect performance. Itās been a bit of a chore shifting everything over to use Context but i think itās worth it as the the end result is an application that feels alive rather than an application that is noticeably putting the browser under a lot of pressure.
Iāve also found thereās other positives to undertaking work like this and because itās quite repetitive and mind numbingly boring my mind has wanderedā¦ which is good because iāve continued to think about what else skin-ui could be.
Todayās thoughts have mainly been around what iām calling Cloud Themes ā¦ allow me to explain.
If i can get skin-ui to a point where user auth / teams auth are a thing you could have a designer using the browser UI and making and saving changes to the theme objectā¦ but then imagine rather than manually copying and pasting that theme object in to the code base it could just be in the code base and capture changes after theyāre saved from the web editor.
I totally think this is possible if i were to write a build time Gatsby Plugins that once configured via
gatsby-config.js
using an id or something would connect to a (yet to be developed) skin-api and pull down the theme
object and then continue to compile the app/site/blog with that theme object.
This might allow yet another abstraction of āweb designā and āweb developmentāā¦ iām thinking at the moment i could make the skin-ui UI more focussed on designers needs and provide easy to understand ways to modify the theme object along with an easy way to preview the changes then, once they hit save all that design work could be pulled in to the application and, et voilĆ youād have styles for anything and everything the engineers could need to build out application functionalityā¦ this of course is dependant on Gatsby becoming the go to tool for larger builds.. but as i stated yesterday
Gatsby is just React
..so why wouldnāt you use it to build more than just blogs?
Day 72
Mar 12 2020
Gatsby is just React!
Itās pretty clear when you start with Gatsby that itās everything React is and more but to be honest iāve never really put it to the test.
When making themes and plugins iāve not really had to do a lot of React, most of the time iāve been able to leverage the Gatsby apiās to solve my problems but www.skin-ui.com is much more of an application than it is a site or blog.
I reached a point during development where local state was getting a bit gross and as i implemented more functionality iāve had to start the dreaded prop drilling pattern which i know from experience never ends well.
Iām now pretty clear on the application architecture and itās time to rip out local state and implement React Content.
Iāve used Context before on a few projects and do prefer it to Redux. Iām not at all attempting to compare the two but in this instance Redux would be over killā¦ that of course might change if the application grows further but for now itās cool.
So today has been pretty much that. Iāve implemented Context, ripped out local state and hooked up Netlify identity through context too.
Day 71
Mar 11 2020
Today has been quite varied. Aside from trying to find a job iāve still got stuck in and made some enhancements to www.skin-ui.com
First up i worked on a little bug i noticed when using the āCopyā button. It was indeed copying the theme object you see in the code editor panel but because itās imported and rendered as a string before itās passed on to CodeMirror when you paste it in to your own code editor it was missing the default export.
By adding export default { ${codeEditorData} }
iām able to correctly copy the actual theme object including the export
default and surrounding curly braces from browser back to a code editor like VS Code.
Iāve also done some work on the string that is passed to the markdown source view. Iām importing the actual .mdx
file
thatās used in the app using the raw webpack loader and passing it straight on to CodeMirror but because it contains a
bit of React and a Theme UI Box component they too appear in the āsource viewā. I tried to write my own regex to strip
out the various parts of the string but ran out of skill so posted this on stackoverflow:
Advanced RegEx Help required
I used a similar regex approach to remove the first 3 lines in that .mdx
file which are importing the layout
component. Now when you view the source using the little āeyeā icon button you get something thatās easier to copy and
past into your own project.
Iāve also added a new section to my commercial portfolio: side hussle which is an area for me to showcase things iāve built that arenāt GitHub repos or Gatsby pluginsā¦ naturally www.skin-ui.com is on there.
It was really easy to add another source directory to the project and render the data as UI elements, and it kinda proves how flexible gatsby-theme-terminal is which makes all the hard work i did on the theme worth it!
I then came back www.skin-ui.com and have implemented Netlify Identity which, if you havenāt used is absolutely super brill brills!
Iāve done auth a few times before in React applications and itās always such a ball ache, with Netlify Identity it was so easy and it even comes with third party auth providers like login with GitHub and Google etcā¦ itās mad how good Netlify areā¦ i love it!
Thatās all for today, tomorrow i need to investigate Fauna and see if i can spin up a database, this is all you can save your theme on www.skin-ui.com and is the first step towards building out a kind of user showcase gallery kinda feature.
Catch ya later!
Day 70
Mar 10 2020
Nude UI has been re-branded and ā¦ Skin UI is now live š www.skin-ui.com/
Iāve had a quite a few DMās reporting firewalls blocking nude-ui.com on work internet connections, i also had a polite request to just change the name.
I always like to hear feedback and when suggestions are made that stand to improve matters, iām happy to accommodate. š
Iāve also been working on accessibility, UX and performance improvements today all of which are going pretty well.
I get a bit hammered on the Lighthouse scores when on /editor but not because the application performs poorly but because thereās a YouTube Embed as part of the Theme UI components exampleā¦ what can ya do ay!
But by far the biggest ball ache iāve had today was trying to change the GoDaddy nameservers and adding forwarding from nude-ui.com to skin-ui.com.
Iām reliably informed by the GoDaddy tech support team that url forwarding is working although iām still yet to see it myself, but it could take up to 48hrs so iāll check it again over the next few days.
In the meantime my Netlify build pipeline which builds from pushes to the master branch deploys to both nude-ui and skin-ui which iām sure will kill my build minutes but hopefully itās only a short term thing.
Thereās one or two things that i need to sort out so will probably set up a GitHub project and raise some issues, i think i need some help on one of the issues as it involves splitting MDX content and whilst have it working now it results in a shitty copy and past experience for users if they copy the source.
Also been thinking about next steps for this project and what i really wanna do is get the user auth and gallery parts set up but i donāt really know what to start. Kinda need a cheap / free db solution so if you know of one please let me know.
Day 69
Mar 9 2020
Nude UI is live! š
Itās taken me a little over a week but today i launched an MVP of www.nude-ui.com/
In a nutshell Nude UI is a starting point for a few things.
- Example markdown
- Example Theme UI MDX Components
- Example Theme UI theme object
Iāve found that when developing Gatsby themes i always have to hunt around to find complete examples of the above, and Nude UI amis to solve this problem.
Iām not sure at this point how people will use it but iāve tried to include all the things that iāve needed in the past.
Using the Editor you can quickly see the full Theme UI object and by editing it you can immediately see the effect of the changes on both markdown and Theme UI components.
You can if you like copy the theme object straight from Nude UI and paste it in to your own project and similarly if you want example markdown and Theme UI components you can access the source from the editor too.
I know thereās a few things missing and i plan on completing the theme object over the next few days but itās a boring task and precisely why i built Nude UI but it does need to be finished.
Iām not sure what the next steps will be, iām hoping itāll be well received and via user feedback on twitter or through GitHub issues will probably help steer me towards phase 2.
If all goes to plan what iād really like to do is create a gallery where users can save and showcase their āthemesā and this in turn will provide yet more options for good starting pointsā¦ but for that i need to go Full Stack and, well, i like the Front End of the Front End too much so i might hold off tackling that until i know if Nude UI is of use or not.
Stay nude guys and do let me know what you think @PaulieScanlon
Day 68
Mar 8 2020
Today was a game of two halves.
Iām still working on nude-ui and spent the morning implementing the remaining app functionality, this includes the ability to collapse the code editor panel and switch between the markdown / mdx preview and the source.
I did have a bit of a ball ache with parsing the mdx directly onto CodeMirror but chrisbiscardi came to my aid with a sweet little webpack trick.
import mdxString from '!!raw-loader!./thing.mdx';
This allows me to bypass the webpack loader for .mdx
file types and import it as ārawā then finally pass it on to
CodeMirror as a string.
Iāve also done a bit of design work today. Iāve got what i think is a pretty sweet looking hero / open graph image and have chosen a ātastefulā nude background image for the landing page.
Tomorrow iāll focus on building out the landing page and iāll probably start thinking about re-pointing the DNS records and setting up a Netlify build pipeline.
At some point this week i think iāll be ready to launch!
Day 67
Mar 7 2020
For a Saturday iāve been pretty productive.
I tackled the sidebar nav today on nude-ui. The main HTML elements were pretty easy to implement but nude-ui is becoming
a bit of a position fixed-fest. The tricky part though was to extract the anchor hashes from the child .mdx
and use
them to populate the nav list, and then i needed to modify my scroll to anchor script in gatsby-browser.js
so it only
attempts to scroll to the position of the anchors in the page and not the anchors in nav.
Tomorrow i might tackle the landing page which might be a bit of a ball ache and iāll need different layouts for the
index and the editor pages. I think iām gonna put the editor page in a folder outside pages
and source it seperately
then i can apply a default layout it.
i need to do some SEO and perhaps a bit of design for the open graph image then itās time to deploy!
Day 66
Mar 6 2020
Another remarkably good day today.
nude-ui is coming together really well. Iāve finished off adding all styles for markdown and theme-ui/components and then set about adding application functionality.
Iāve added two checkboxes so the user can show / hide both the markdown and or components from the preview pain. Iāve also implemented the all important copy button, this will allow users to copy and paste the theme object in to their own projects.
I darenāt speak too soon but i think iāve pretty much completed everything i wanted to in order to launch an MVP!
Iāll continue to tinker over the weekend and maybe try and release something Monday.
Day 65
Mar 5 2020
What a cracking day! Woke up annoyingly early but have now cracked the CodeMirror string issue.
CodeMirror if youāre not aware is the code editor that drives the likes of CodePen et al and provides code editor like behavior in the browser.
Iām using the React version by Jed Watson but it works in the same way.
The issue was for CodeMirror to display anything it needs to be provided as a string via the value
propā¦
JSON.stringify()
to the rescue!
<CodeMirror value={JSON.stringify(themeObject, null, 2)} />
butā¦
This will pass the the object as JSON meaning the output in CodeMirror will also be JSON and look something like this.
"colors": {
"primary": "#E91E63",
"secondary": "#2196F3",
"success": "#8BC34A",
"warning": "#FF9800",
"error": "#F44336",
}
and yeap youāre right, thatās not JavaScript, and doesnāt actually represent what a theme-ui object would look like.
So we need to strip out the ""
marksā¦ but only on keys not on values and only if those values are not of type
string
ā¦ regex to the rescue!
JSON.stringify(themeObject, null, 2).replace(/"(\w+)"\s*:/g, '$1:');
butā¦
Now we need to pass the object back up to useState
whenever a change occurs so it can be passed onto the
<ThemeProvider />
I know what youāre thinkingā¦ JSON.parse()
to the rescue!
ā¦ wrong!
<CodeMirror
value={JSON.stringify(themeObject, null, 2).replace(/"(\w+)"\s*:/g, '$1:')}
onChange={(event) => JSON.parse(event)}
/>
The above wonāt work because JSON.parse()
canāt parse this as JSON because weāve removed the quote marks and these
are what make it valid JSONā¦
Ahh!! so now we need to put the quotes back in ā¦ regex to the rescue again!
<CodeMirror
value={JSON.stringify(themeObject, null, 2).replace(/"(\w+)"\s*:/g, '$1:')}
onChange={(event) => JSON.parse(event.replace(/(['"])?([a-zA-Z0-9_]+)(['"])?:/g, '"$2": '))}
/>
Only now can we display a JavaScript object in the āeditorā view and parse an object back to useState
to drive
theme-ui.
It was a challenge but regex really is a wonderful thing ā¦ top tip https://regex101.com/ is awesome!
Now that the editor / theme object / parsing issues are resolved iāve set about adding the relevant key value pairs to the āexampleā object and have added the full list of markdown and theme-ui components to the preview so you can see the changes youāre making in the editor in real time.
Pretty sweet day iād say!
Day 64
Mar 4 2020
Today was a jet lag day. I did a bit of work on nude-ui and am still wrestling with how to pass a JavaScript object as
string to CodeMirror and then part it back out as an object to useState
i think with a lot more Googleing iāll find a
solution but other than that no real updates.
Day 63
Mar 3 2020
Today was a mega travel day and iām now back in the UK!
I did manage to sneak in a couple hours work on the flight but without internet access didnāt get very far. I continued
to work on nude-ui and am trying to work out how to parse a JavaScript object in to CodeMirror as a string but i need to
remove the ""
using regex but i then need to pass it back out to useState
as a JavaScript objectā¦ JSON.stringify
and JSON.parse
seem to be the obvious choices but the regex is tripping me up.
This is definitely a job for tomorrow now. Itās late and iām tired.
Day 62
Mar 2 2020
Today iāve started something new!
Iāve been thinking about it for a while but didnāt know if thereād be a requirement for it or if i could even do it.
So i started anyway š
Iāll start by explaining āthe problemā, or at least some problems iāve had in the past.
If youāre using markdown and .mdx
in your blog or site and chose to expose the
theme-ui/components as i have with
gatsby-theme-terminal you need a pretty solid theme
file. Itās got to
cover everything in styles
for the markdown spec, it should probably have every theme key described in the
theme-ui/theme-spec and have all the relevant variants for
theme-ui/components. Whilst the first two are well documented the components were a
bit of a ball ache for me.
ā¦ and hereās my proposed āsolutionā Iāve decided to create a kind of object editor with live preview split screen so you can play around with the values for theme-ui and see the markdown and or components update right before your eyes.
The plan is to use local storage to āsaveā your work as you go and to provide a ācopyā button so when youāre finished you can just copy the code and drop it straight into you project. Other ideas iāve had are to be able to save your theme so others can use it. It could become a kind of marketplace resource for theme-ui but i gotta walk before i can run ay!
Iām not sure if this already exists somewhere and even if it does iām gonna crack on anyway as itās a good opportunity to learn something new.
As always iāll be developing this in public and no doubt itāll appear in a number of blog updates after today.
Iād welcome contribution on this one so if youād like to get involved please find me on Twitter
Cheerio!
Day 61
Mar 1 2020
Today was a travel day ā¦ again! ā¦ but i have still managed to get a few hours work done.
Iāve released a new component in gatsby-theme-terminal called SourceMonths and refactored SourceDays so they both work the same way.
These components group any sourced .mdx
file by year and provide some nice data points for days of the week and months
of the year which could be used to create data visualizations.
With these two complete and released i think i could pretty much re-create the data viz on the dashboard of gatsby-theme-gatstats which is making me think, perhaps i should shift this blog over to using gatsby-theme-terminal?
Iāve also done a bit of house keeping and have turned off my Digital Ocean servers and deleted all but one app hosted on Heroku. Pretty much everything is now serverless and hosted on Netlify!
All in all a pretty productive Sunday!
Day 60
Feb 29 2020
Iāve had a very functional day today. Iāve been looking over the data visualization i created for
gatsby-theme-gatstats and if i strip back the presentation
layer (the bar charts, donut charts, etc) what i was actually doing was abstracting the data from all .mdx
that had
been sourced and working out some nice percentage and count values for the date fields pulled from frontmatter
.
Iād like to expose this same data in gatsby-theme-terminal
and have got a new component called <SourceDays />
which you can see
here itās like the other data components in this
theme whereby it just returns values which you can then use to create the visual presentation layer. In the demo iām
just using the theme-ui components but i guess if you wanted to install your own charting library you could use these
same values to drive any kind of data visualization.
Iāve released <SourceDays />
in 0.0.7
and started working on a component called <SourceMonths />
but realized it
would probably be really handy if you could not only filter
by source eg: āpostsā, āprojectsā but also by year, this
would then allow you to create a comparison chart for posts, tags used etc for multiple years.
Iām gonna keep working on this as itās a good opportunity to work on some functional programming skills which i always struggle with but do quite enjoy!
Day 59
Feb 28 2020
What a great way to finish off the month. Iāve been overwhelmed by the likes and retweets from yesterdays announcement regarding the launch of my new theme, one chap by the name of Rob Blake even saidā¦
Hey Paul, great work. Iāve been reading your blog and looking at some of the stuff youāve been working on and Iām impressed. Inspired might be a better word. Thanks for sharing.
ā¦ which is music to my ears! Iāve been digging deep in to Gatsby related stuff for over a year now and itās so good to know others find it as interesting as i do.
If youāre reading this post Robā¦ thanks so much for your tweet!
Other bits iāve been up to today are writing a post about the new theme, why i built it and some methodologies i feel are important in theme development. You can read that post here
And then finally i got round to catching up on the #100DaysOfGatsby challenges and have now completed challenge, 7, 8 and 9 on my new commercial portfolio pauliescanlon.io/ ā¦ iāve got ot be honest making it a PWA doesnāt seem that important but itās good to know how to do it and it does impact the overall Lighthouse score which is now pretty decent! š
So thatās it for today. Iāve got to have a bit of a think about what to do next and with only a few days left of my trip i should think itāll be just tying up loose endsā¦ oh and i still need to find my next job so if you know of anything please let me know!
Day 58
Feb 27 2020
Stick a fork in me iām done!
Yeap, today i finished off and launched my new theme gatsby-theme-terminal.
Itās been a bit of a labour of love this one, mostly because i wanted to update my commercial portfolio and turn the Digital Ocean servers off.
With Gatsby and locally sourced files and the theme sorted iāve updated my portfolio which uses the themeā¦ and here it is š pauliescanlon.io
Now finally i get some decent SEO for my portfolioā¦ and hopefully more work!
I think the theme can pretty much do whatever you need and i think iām on to something with the data components and plan on creating a few more for different data sources in the next releases.
ā¦ thatās me for today!
Day 57
Feb 26 2020
Today has been brilliant and for two very good reasons.
- Iāve completed my new theme š but.. iām not quite ready to announce it.
- To test drive new theme iāve updated my current commercial portfolio which now uses it as a theme and it works in every possible way i hoped it would!
I feel like my new theme offers a lot more than just being a theme, and perhaps thatās what John Otander, Chris Biscardi and the rest of the Gatsby team who developed āthemesā envisaged all along but for me itās been a journey but i think i finally get what a theme really is.
When you think of themes and compare them to e.g: Wordpress Themes i think of a code base that does everything in a predetermined way and all you have to do is provide the content.
I feel that after a lot of experimentation with Gatsby themes they can be and are so much more.
My first theme gatsby-theme-gatstats took on a very motherly role and formatted everything for you and provided functionality in a certain way so long as you provided it with what it needs.
My new theme is the total opposite, yes, it does require frontmatter
in a particular way but quite literally
everything else is up to you.
If you want multiple pages, no problem, if you want multiple data sources, again no worries, if you want your posts list displayed in any given page / route, my new theme can do this, if you want a button a card or an avatar my new theme has got your back!
Itās yet to be decided if theme users will get on board with this approach and iām hoping itās not too far away from the current theme expectations and where it differs is for a good reason.
My new theme styles everything you could possibly need and provides data in nicely formatter ways to help drive your UI but the composition of the theme is entirely up to you.
I have high hopes for this one but fear my lack of followers on Twitter will be a bit of a problem.
If youāre reading this and arenāt following me please do and please share my work
Thanks in advance! š
Day 56
Feb 25 2020
Absolute whirlwind of a day. Not sure if youāve ever used BrowserStack ā¦ and itās slow at the best times but try doing it tethered on a 4G iPhoneā¦ š”
The reason i needed to use BrowserStack today was to test some ārecipesā iām providing in my theme docs and one is for a
āCardā based flex grid arrangement which heavily relies on flex
ā¦ i know from previous painful experiences that IE11
has trouble with flex
so just out of curiosity i decided to see how my recipes were looking in IE11 on Win 10
ā¦ needless to say on the first attempt they looked terrible so iāve spent about 4 hours painfully trying to debug IE11ā¦ but, pleased to report itās now all sorted!
With that ball ache completed iām getting very close to completing my new theme.
Iāve double checked the theme-ui/components styles and have full styles for the standard markdown specā¦ but,
does theme-ui work in IE11?
Day 55
Feb 24 2020
A much better day today! Iāve finished off the list of style objects for all
theme-ui/components ā¦ there are one or two that i canāt seem to style but iām not
sure if itās me or a problem with theme-ui
.
I know the package is undergoing some changes so iāll wait for a bit in case the issue is raised and or fixed with the next release.
Other bits iāve done today include tidying up styles, further reducing the amount of theme specific components, iām basically down to six now.
These are;
- Logo: Which is just a placeholder ready to be āshadowed
- Main: Which is really just a place to compose the
<Nav />
and the<MDXProvider />
+ all the theme-ui/components supplied via thecomponents
prop - Navā¦ you know what a Nav is rightā¦ it does compose one of my other plugins though gatsby-mdx-routes
- Seo: Just handles the meta stuffā¦
- And x2 layouts, one for Pages and one for content sourced from outside
src/pages
, the latter requires the<MDXRenderer />
And thats pretty much it!
The footprint of my new theme is gonna be really small and thereās very little to maintain and i really hope, very little that can go wrong.
Iāve also implemented anchor scroll to hash and theme-ui.com/packages/prism/ which if you havenāt used ā¦ is absolutely brilliant
Iām aware iām a little behind the actual challenges so i think the next few days ought to be investigating what they are and if i can shoe horn them into my new theme.
If all goes to plan i might be able to release the theme this week!
Day 54
Feb 23 2020
Today was a travel day so i havenāt been able to do much. However, i have made a start on creating a list of all the theme-ui/components and their default objects. I think once again iām either doing something wrong on thereās some issues with theme-ui/components.
For example in the docs for the <Badge />
component it says itās
default variant is primary
, so you should be able to use it like the below and it āshouldā pick up styles defined in
theme.badges.primary
<Badge>I'm a badge</Badge>
ā¦ but no matter what i do if i attempt to use a badge without the variant prop it defaults to the theme-ui built in styles
Iāll continue to work on this as itās a big part of my new theme and i want the full set of components to work so i need to work out where all these components grab their styles from to make sure iāve got them all covered.
Tomorrow is a new day, iāll tackle it then!
Day 53
Feb 22 2020
Today has been spent mainly thinking about stuff. I did some code and have now sorted the navigation problem i was having with my new theme.
The things iāve been thinking about today may only apply to very specific circumstances but perhaps if youāre reading this and have had similar thoughts, lets talk; @pauliescanlon
To give some context what iām attempting to do with my new theme is provide everything that
theme-ui/components offers via the use of the components
prop on the MDXProvider
Hereās the setup in .js
// some-layout.js
import { MDXProvider } from "@mdx-js/react"
import * as themeUiComponents from "@theme-ui/components"
<MDXProvider components={....themeUiComponents}>
{children}
</MDXProvider>
ā¦ and hereās the usage in .mdx
(where <Button />
is the theme-ui/components button component)
// some-page.mdx
# Hey
Some other text
<Button>Click me</Button>
Using this approach i hope to make a zero component theme and what i mean by that is there wonāt be any presentational components provided by the theme itself.
The theme will only expose what iām currently calling data components which allow for a better level of abstraction in a theme as they only concern themselves with data, not UI.
Using this approach the theme author only needs to provide a set of data components that manage querying GraphQL and itās up to the theme user to decide what to render.
The cool bit is all of the āthingsā theyāll likely want to render will have already been styled either by theme.styles
or components
Hereās any example usage of a <SourceTags />
data component iām developing.
<SourceTags>
{source => (
<Flex
sx={{flexDirection: 'column'}}
>
{
source.map((tag, index) => {
const { name, count, percent } = tag
return (<Box key={index}>
{`${name} x${count}`}
<Box mb={2} bg='muted' sx={{width: `${percent}%`, height: 4}} />
</Box>
})
}
</Flex>
)}
</SourceTags>
Inside the <SourceTags />
component is where i query GraphQL and also do a bit of functional JavaScript to return a
nice object with the name
, count
and percent
of each tag found in frontmatter
In this example iām rendering a <Box />
which displays the name
the count
and a kind of horizontal bar chart. All
of this is made using theme-ui/components
NOT components provided by the theme.
The bit iām currently stuck on is how to manage the styles for both markdown
nodes and theme-ui/components
eg:
In your theme you probably have something like this
// gatsby-plugin-theme-ui/index.js
export default {
...
styles: {
p : {
...
}
}
}
But when you introduce theme-ui/components
you also need
// gatsby-plugin-theme-ui/index.js
export default {
...
styles: {
p : {
...
}
},
...
text : {
...
}
}
The reason behind this is if your theme users write body copy itāll get translated to a p
and if they choose to use
the <Text />
component it will by default render as a div
unless they tell it itās a p
, <Text as='p'/>
ā¦ in
either case the <Text />
component will look only to the text
object in the theme, not the p
object in the
theme.styles
.
So if like me youāre planning on supporting both markdown
and theme-ui/components
in .mdx
youāll need to spread
styles across both of those objects.
eg:
// gatsby-plugin-theme-ui/index.js
const textStyles = {
fontFamily: 'body'
fontSize: 1
}
export default {
...
styles: {
p : {
...textStyles
}
},
...
text : {
...textStyles
}
}
So step one of this process for me will be to comb over the theme-ui/components docs
and construct a list of all the theme object keys the theme-ui/components
look for by default.
This above <Text />
example refers to this or more specically this
Text style variants can be defined in the
theme.text
object.
Once i have the full list of default object keys i can begin to work out where the spreads need to occour to ensure
consistent styling between markdown
and .mdx
Wish me luck!
Day 52
Feb 21 2020
Felt like a bit of a slog today.
Iāve carried on themeing my new theme, got the nav / responsive nav working and got SEO hooked up and sorted the Netlify build pipeline.
Iām a little disappointed with how the nav is looking, and to be fair i didnāt actually design it i just riffed it and coded it but itās not pretty! š© ā¦might have to go back to design briefly and smarten it up.
No real problems today apart from something thatās been bothering me for a while but iāve been working around it and
thatās how VS Code / Prettier donāt ever seem to format my .mdx
properly.
Might ask for help on Twitter about this one as itās got me stumped.
Day 51
Feb 20 2020
Today iāve continued to work on my new theme and itās getting a bit tricky in places.
Iām using theme-ui
but i have encountered a couple of things that are causing me to scratch my head.
If youāre using markdown
thatās all well and good, you can use the styles.h1
object in theme-ui
and set your
styles for eg; h1
But..
In .mdx
where of course i can still use # h1
but only if itās not wrapped by a theme-ui
component.
Let me explainā¦
In markdown i might wanna do this.
# Hey, i'm a header
and i'm some body copy.

But with .mdx
youāre able to position the body copy and the image next to each other using a combination of <Flex />
and <Box />
For example
<Flex sx={{ flexWrap: "wrap" }}>
<Box
sx={{width: ['100%', '50%']}}
>
# Hey, i'm a header and i'm some body copy.
<Box>
<Box
sx={{width: ['100%', '50%']}}
>

<Box>
</Flex>
But because both <Flex />
and <Box />
are technically React components the markdown
contained within them wonāt
render.
Instead of using # h1
you can use the theme-ui/components
<Heading as='h1' />
componentā¦
butā¦ it wonāt by default pick up the same styles from styles.h1
, instead itās default place to look for styles is
styles.headings
Which means to get the same styles applied you have to define a const
elsewhere in the theme, then spread the styles
into both of the style objects.
Furthermoreā¦ and i might be wrong on this but styles.headings.h1
doesnāt do anything.
One way round this is to use the heading component like this <Heading as='h1' variant='styles.h1' />
which is a bit of
a faff when what you really want from a writing experience is to quickly style your content.
If iām barking up the wrong tree on this, my apologies or if you know where iāve gone wrong please do tweet me @pauliescanlon
Day 50
š Feb 19 2020
Day 50 and iāve posted something every day for 50 days and iām feeling pretty pleased about that š„³
Today started a little differently and iāve gone a bit off piste, the reason for this side step is because during my alpha tests for gatsby-plugin-prop-shop i created a PR for theme-ui and got a lovely response from jxnblk who thought PropShop was a really cool ideaā¦ which made me very happy!
Thanks! This is a really cool idea!
He then went on to explain that @theme-ui
and all of itās packages are undergoing some changes, namely the whole
repository is being Converted to TypeScript and that it might be
best to wait until that work has been completed before the PR is merged in.
Upon reviewing the issue i noted that the maintainers are actively looking for help and if you wanna get involved just name the package youād like to convert and you can claim it.
So thatās what iāve done, i spent this morning converting theme-ui/packages/components/ to TypeScript and hereās the PR
I did get a bit stuck so have pushed my changes and asked for help then i jumped back on developing my new theme and have been continuing to work towards this minimal set components idea.
This approach seems to be working really well and iāve even tried using render children from .mdx
which works a treat!
I might write a follow up post about why i think this approach is so powerful for theme development but maybe i should finish developing the theme first ay!
Day 49
Feb 18 2020
Iāve had a very productive day today!
I really wanted to nail down how to manage dynamic .mdx
sources.
To explain this a bit more, i need to reference GatStats again. In GatStats to use the theme you have to put all of your
.mdx
blog posts in a directory located at src/posts
and the resulting URL will be
http://www.your-site/posts/some-post
However, what if you wanted to have a posts and a projects or maybe a talks directory as well?
You canāt because in the theme iāve configured it to only source .mdx
from src/posts
ā¦ which sucks!
In my new theme iām exposing a tiny bit of config in gatsby-config.js
and allowing the user to tell the theme where to
source
.mdx
from, in the example below the theme will look in both posts
and projects
module.exports = {
...
plugins: [
{
resolve: `@pauliescanlon/my-new-theme`,
options: {
source: [`posts`, `projects`],
},
},
],
}
This can be passed onto the gatsby-config.js
in the theme.
module.exports = themeOptions => {
const { source } = themeOptions
...
}
From here iām able to first test if source
is a single string or an array of strings, and then dynamically create a
gatsby-source-filesystem
config in the plugins
array.
module.exports = themeOptions => {
const { source } = themeOptions
let filesystemSources = []
const sourceFilesystemOption = name => {
if (source) {
return {
resolve: `gatsby-source-filesystem`,
options: {
name: `${name}`,
path: path.resolve(`src/${name}`),
},
}
}
}
if (Array.isArray(source)) {
for (let item of source) {
filesystemSources.push(sourceFilesystemOption(item))
}
} else {
filesystemSources.push(sourceFilesystemOption(source))
}
return {
...
plugins: [
...
...filesystemSources,
],
}
}
Thatās the first part sorted, the next part was a bit tricky.
Because youāre sourcing .mdx
from outside src/pages
you have to use the createPage
api to create a page for each
file found.
Over to gatsby-node.js
First i needed to create a slug
on each node, this can be done using createNodeField
butā¦ i only need to this on
nodes that donāt already have a slug
Any .mdx
sourced from src/pages
will have this field already because iām using my plugin
gatsby-mdx-routes š
so in the if
i check both that the node.internal.type
is in fact .mdx
and then i check that itās not a file thatās
already had a slug created.
exports.onCreateNode = ({ node, actions, getNode }, themeOptions) => {
const { source } = themeOptions;
const { createNodeField } = actions;
if (node.internal.type === 'Mdx' && !node.internal.fieldOwners) {
let path = source;
const value = createFilePath({ node, getNode });
if (Array.isArray(source)) {
path = node.fileAbsolutePath
.split('/')
.filter((str) => source.includes(str))
.toString();
}
createNodeField({
node,
name: `slug`,
value: `/${path}${value}`,
});
}
};
The bit that took me a while to work out was what to use as the path
.
What i want to happen is any .mdx
sourced from src/posts
would have a URL of,
http://www.your-site/posts/some-post
and if itās sourced from src/projects
it would have a URL of,
http://www.your-site/projects/some-project
If the source
is single string thatās pretty straight forward and i can use it as is.
let path = source
...
createNodeField({
node,
name: `slug`,
value: `/${path}${value}`,
})
ā¦ but if the source
is an array i need to be able to determine which source
the file has been sourced from so the
correct path
can be added to the createNodeField
value
field.
I did try and loop over the array and the call onCreateNodeField
from inside the loop but this wonāt work because
onCreateNode
is itself called on a loop.
The only other part of the node that is exposed which i can use to perform this kind of logic is node.fileAbsolutePath
and by using split
together with filter
iām able to get at the bit i want, the posts
or projects
which can then
be added to the createNodeField
value
field as before with the string.
Once thatās all sorted you can continue to follow the Gatsby docs creating-pages
Iām fairly confident this is a solid enough way to query .mdx
files from any directory in the src
directory, i
might need to re-think this if i want to allow the theme to be able to source
from anywhereā¦
Using a content directory seems to be a popular patternā¦ but itās my theme and i think src
is a good enough
place to put your source files
Day 48
Feb 17 2020
Today was a travel day so iāve only done a few hours workā¦ but. Iāve continued to work on my new theme and itās coming
together nicely. I mentioned before about learning from the mistakes i made with GatStats and this time around iāve gone
all out with the developer experience. Iām going to try and keep custom components and config to an absolute minium in
favor of exposing as much as i can though .mdx
Reason being is that with GatStats the only viable option for customization is component shadowing, and this as good as it is feels like a lot of un-doing of what a theme is supposed to provide.
I want this new theme to be as easy to customize as it is to write markdown.
To give you an example, on the home page rather than asking the user to whack a load of stuff in the config so i can
query it and pass it down to a a component in .js
to then fit it into a grid/column layout, iām going to provide
functionality through the MDXProvider
components
prop and passtheme-ui/components
straight though the.mdx
. There
will of course be a little styling to manage so when these components are used theyāll still feel like part of the theme
but the difference will be the user can decide what goes where and how itās laid out.
eg:
<Flex sx={{ flexWrap: 'wrap' }}>
<Box
sx={{
width: ['100%', '50%', '33%'],
}}
>
<Heading variant='styles.h1'>contact</Heading>
<Heading variant='styles.h2'>Email</Heading>
<Link href='mailto:pauliescanlon@gmail.com' variant='styles.a'>
pauliescanlon@gmail.com
</Link>
<Heading variant='styles.h2'>Twitter</Heading>
<Link href='https://twitter.com/PaulieScanlon/' variant='styles.a'>
https://twitter.com/PaulieScanlon/
</Link>
<Heading variant='styles.h2'>GitHub</Heading>
<Link href='https://github.com/PaulieScanlon/' variant='styles.a'>
https://github.com/PaulieScanlon/
</Link>
<Heading variant='styles.h2'>GitHub</Heading>
<Link href='https://www.linkedin.com/in/pauliescanlon/' variant='styles.a'>
https://www.linkedin.com/in/pauliescanlon/
</Link>
</Box>
</Flex>
using <Flex />
and <Box />
you can create any colum layout you like, straight from the .mdx
. š¤Æ
Youāll also see above the use of <Heading />
etc, these are theme-ui/components
and will be styled in accordance
with my theme design the same way a # heading
would if used in markdown.
I think itās gonna be a sweet theme and i will definitely be using it for my commercial portfolio once itās released!
Day 47
Feb 16 2020
Sad times today. Iāve released a first version of gatsby-plugin-prop-shop.
butā¦ i wasnāt anticipating failure, alpha tests have kind of suggested that PropShop is a bit useless š
When i built it i was thinking, this is great, i know iāve needed functionality like this on many a projectā¦ but thatās because i develop component libraries for my day job.
It would appear as though if youāre developing your own blog or a theme, PropTypes and descriptions arenāt really a priority. I somewhat disagree if youāre building a theme because youāll want your users to know what each prop does in case they shadow the component and documenting the props would be very helpful in this instance.
However, what i learned from my alpha tests are that not a lot of people bother with PropTypes š„.
I can understand this if youāre developing in .js
because thereās no real reason other than for completeness to spend
time writing out types and descriptions, if youāre using TypeScript itās of course an absolute must! But from the alpha
test PRās i created i noticed not a lot of Gatsby developers are using TypeScript.
All of this was an oversight to be honest because i was just thinking about what stack i use and how i approach a build.
Iām a huge proponent of Component Drive Development and correctly typing props and documenting them is a big part of this.
The other oversight is a pattern iāve noticed where GraphQL queries are created using the new-ish Gatsby
useStaticQuery
method and then abstracted away in to a utils directory and then imported as and when required.
When using this method passing props around neednāt occur at all as everything is returned within the component where
the query is being used. I suspect in TypeScript youād need to type the return but in .js
land everything is so loose
you can leave this un-typed and pass the data straight onto the .jsx
no questions asked. š¬
So with that said, iāve released a 0.0.1
anyway and tweeted a few people and rather than get hung up on failures, iāve
dusted myself off and come up with a new plan.
Iām going to build another theme!
I have for a while now been wanting to transfer my current commercial portfolio from Next.js to Gatsby and i had intended to use gatsby-theme-gatstats for both my portfolio and my blog.
butā¦
After digging around in the GatStats repo after having learned a lot from my recent work i think GatsStats needs a bit of an overhaulā¦ iām not sure when iāll tackle this and attempting to rip out methods to replace them with some new things iāve learned might become messy so iāve decided to start a new theme and iāll try out some new methods and see how it goes.
Iām not too disappointed with GatStats, it was my first theme after all, in fact it was the first thing i built using Gatsby so there were mistakes made along the way but thatās just all part of the process ay!
Day 46
Feb 15 2020
Itās been an absolute belter of a day today!
I started the day by refactoring some quite gross š¤¢ functional programming iād done in gatsby-plugin-prop-shop and i think second time round iāve made the ātotalsā object a lot neater and streamlined the amount of times the GraphQL array needs to be looped over.
Thereās probably a significant performance increase but iām not too worried about testing that at the moment.
Next up was to find a Gatsby theme written in TypeScript so i could test PropShop with .tsx
files or more specifically
interfaces
Fellow Gatsby fan Rich Haines has a sweet little theme for adding SEO
data to
your blog, you can see it here.
So i forked that repo and added gatsby-plugin-prop-shop ā”
There are some differences between the way react-docgen returns data from PropTypes vs interface(s), the main one being
with PropTypes all data objects are returned with an id
, with TypeScript they donāt, which was throwing my table
header out of whack.
To resolve this instead of trying to get data from the GraphQL query i just manually created an array of table headers and that was that sorted.
I then forked theme-ui š¬ and set about adding PropShop to the docs so that i could see PropTypes for theme-ui/components. It worked first time amazingly even though the components directory is in a different package!
I did have an odd style issue so iāve set the prop-shop
page wrapper to be position: absolute
so no matter which
site or blog PropShop is used in that page will always cover anything else defined in a default layout.
Iāve also written a blog post about PropShop, which if youāre interested you can read here
Iām still very much in the alpha phases of development and i now have x4 PRās open and am looking for feedback before iām ready to release a first version of the plugin so if youāre reading this and want to help, tweet me @pauliescanlon
Day 45
š Feb 14 2020
Today was a bit of a maintenance day on gatsby-plugin-prop-shop there were a few things iād noticed as part of my alpha tests as mentioned yesterday with styles not quite being as i wanted them so iāve been over all the css and tidied up a few things.
I also took another stab at the filtering, previously if you start to type you can filter out files containing props that donāt match your search term but props from the same file still appeared in the list. The new filtering system hides any prop from any file that doesnāt match the search term which makes using PropShop a much better experience.
Iāve still got a bit more tidying up to do before i release 0.0.1
and ideally iād be able to see both PRās i created
for Scott Spence and Eric Howey on their respective
projects completed.
On that note however, when speaking with Eric he mentioned heād not really used PropTypes that much and looking at his project i can see why. A lot of the components use a hook to grab the data so in theory all the components manage their own data source internally and donāt require props to be passed in. Itās a pretty sweet approach and i might āborrowā it for gatsby-theme-gatstats
Iāve also been thinking about how PropTypes are kinda optional in straight .js
world so perhaps a lot of people
developing Gatsby projects wonāt even use them if theyāre developing in .js
ā¦ .ts
however you have to have props
defined in an āinterfaceā or as a ātypeā or you get IDE errors left, right and center so i think this will be the next
thing i investigate.
It looks like from the react-docgen docs that TypeScript ātypeā is supportedā¦ not sure about āinterfaceā so iāll look into it and see whatās what.
Day 44
Feb 13 2020
šØ We have an alpha release of PropShop!
Yup, Yup, Yup. Today iāve released an alpha version of my new Gatsby Plugins gatsby-plugin-prop-shop
As you can see itās been renamed from āProject Propā to PropShopā¦ watta you think?
I started the day by forking a couple of repos from trusted Gatsby dev pals,
Scott Spence and Eric Howey and added the plugin to their
gatsby-config
.
Problem No.1 was iād set the path for gatsby-source-file-system
to path.resolve(
src/ā¦)
which will resolve to the
demo running the plugin not the theme where itās installed. In most cases theme developers will want to inspect
PropTypes for components in their theme not in the demo running the theme.
Thatās now changed to accept a full path of where to find .js
files containing PropTypes.
Problem No.2 was as suspected style related. I was using gatsby-plugin-theme-ui
for the styles but when running the
plugin in a theme or, demo of a theme that was also using gatsby-plugin-theme-ui
the theme object was getting
overwriting and i was losing all my styles because the theme or demo theme wins out.
The resolve was to still use theme-ui
but not the Gatsby Plugins version which meant i needed to wrap the
prop-shop.js
page in a <ThemeProvider />
ā¦ no big deal there.
I have seen a few other issues where some of my styles pick up the theme styles, things like global typography rules do affect my typography rules if iāve left a css value unset. I think i can sort this by being more strict with my styles and setting things that i was taking for granted as browser defaults.
So there ya go. If youāre into PropTypes and documenting components PropShop can give you a helping hand!
Enjoy!
Day 43
Feb 12 2020
So close now!
Happy to report iāve very nearly completed all work required to launch an MVP of āProject Propā looking at what iāve done iām confused as to why itās taken me so long š. That said, iām finished with all design, all styles and i only have a tiny bit more functionality to add to allow for table sorting but i might not launch with this as iām keen to get testing.
Iāve got the build pipeline sorted in Netlify, prepared a nice open graph image but will need to add some SEO back in to the demo but that shouldnāt be too much work and iāve started writing the README.
After that all i really need to do is publish it on npmā¦ then let the world knowā¦ and by world i mean my 400 followers on Twitter š
Thatās it for today, no real problems or headaches i just cracked on and all is well!
Day 42
Feb 11 2020
Another good day today! š I did have to do some traveling so lost a few hours of sweet sweet developing time but i think iāve made up for it.
After yesterdays progress with the design of āProject Propā iāve moved back to development and have implemented all the new styles apart from some small data visualizations which i might try and attempt in pure css. I think i have a repo for an old project where i created some css only donut charts.
One thing that continues to play on mind is that iāll really need to test this plugin against multiple Gatsby projects.
Whilst iām confident in the JavaScript because iām using react-docgen to generate all the data to fill the prop table iām unsure how the styles will cascade.
To give you some context āProject Propā is a plugin but it works like a theme.
When you install it and spin up your project you can visit eg http://localhost:8000/project-prop/
. The plugin has
already generated the page that will appear on this route as it gets bundled in with your site or blog build as part of
the gatsby build
/ gatsby develop
processā¦ but so too will the styles.
Iāve tried to name tokens in my theme-ui config in such a way they wonāt clash by pre-fixing them with a āppā but given that this is a theme and themes are supposed to provide all the styles for the project iām worried theyāll try and overwrite styles defined by the user.
Iām currently developing a demo using the Gatsby starter blog and it works ok but it doesnāt implement theme-ui so iām not sure if theyāll be complications further down the line.
I think what iāll do is release a 0.0.0-alpha
version then perhaps pull some ready made Gatsby themes and see what
happens when i add this pluginā¦ should be fun!
Day 41
Feb 10 2020
Feeling good today. š
I put in a proper shift and made some excellent design progress on āProject Propā i have the MVP design and interactions all sorted and i even managed to design a logo.
I know from previous experiences that for me, attempting to design in browser doesnāt really do me any favours but i do like to kind of prototype stuff and then if i think itās a good idea i go ahead and design that functionality. Better that then designing a load of things and then finding out they canāt be doneā¦ seems like a waste of time!
Having said that, thereās a few extra things iāve put in the design which i havenāt prototyped, namely the sorting behavior of the āPropTableā. It will probably be fairly straight forward so for now iām happy for this to be part of the MVP.
Thatās kind of it for today. Design is not my favorite thing to do but it is always good to dust of the old skills andā¦
i reckon i still go it! š
Day 40
Feb 9 2020
Sometimes i really struggle with functional programming, i think itās because iām not from that background or maybe my
brain is wired differently but the problem i was having yesterday with filtering a deeply nested array was quite easily
solved with the array.some
method.
some
tests whether at least one element in the array passes the test implemented, this was all i was really looking to
do, but instead i was messing around with map
which of course returns a new array, and then trying to filter that.
some
returns a boolean value and from there you can do what you want with the return value.
Anyway, i now have all the search and filtering functionality implemented to create what i suppose will be an MVP of āProject Propā.
I have an overall plan of what iād like āProject Propā to do but thereās a little part of me that is working it out as i go and as iām using it iām thinking about what other functionality should be added in.
I have to go back to the design phase a little bit now as iām not sure where some of the filter options should be positioned and this is way easier in design that it is to move styles and props around in code.
So tomorrowās plan will be just that, a little bit of pixel pushing then back on to the code. I also need a logo at some point so thatāll be fun!
Day 39
Feb 8 2020
Not an overly productive day today. The good news is iām finally back on āProject Propā, the bad news is Iāve run out of JavaScript talent.
Maybe Iām just having one of those days because I canāt for the life of me work out how to do some pretty tricky filtering. I have a search input kinda thing working but I need to filter a nested array from an array of objects and still return the parent object if itās child array contains what Iām filtering against, but filter out anything thatās not a match for the filter.
Hopefully a good nights sleep and an offline re-think will get me over this small hurdle.
Itās probably the most important part of āProject Propā as itās what will be used to aid users in inspecting prop details / descriptions.
Part of me wants to move on to another part of the project, the bit that Iāll probably enjoy more as itāll be mostly data visualization but I feel giving up on this first issue before Iāve solved the filtering problem wonāt help me in the long run.
Day 38
Feb 7 2020
Challenge accepted and challenge completed.
gatsby-theme-gatstats now has a contact page hooked up with Netlifyās built-in form service ā¦ which is truly magical!
Iāve completed the work on the components used in the demo contact page and set it up using Formik and iāve also thrown in some sweet validation using Yup
All the components used in the demo are available as shortcodes so you donāt need to import anything to use them in the themeā¦ but because this is a theme iāve not included Formik or Yup.
I feel at this point itās best to leave that up to whoever is using the theme to decide how theyād like to handle forms and form validation.
I was thinking about adding a few more form components to the theme shortcodes, radio or checkboxes might be handy but until i hear from end users iāll shelve development on the theme for a while as i really want to get back to āProject Propā and i still need to update all the posts in my commercial portfolio which iām putting offā¦ butā¦ it was great to finally take part in one of the actual challenges set out by #100DaysOfGatsby
Day 37
Feb 6 2020
Today was another travel day and i think theyāll be a few more before i find my vibe. But, i have done some work today on the form elements.
Iāve created form, text input and text area components, all work via āshortcodesā because theyāve been added to the
components object of the MDXProvider
which means they can be used in any .mdx
file within the theme, no imports
required!
All of the components in gatsby-theme-gatstats are documented in Storybook and iāve tried to write a few unit tests where applicable.
This will probably be a job for tomorrow now, then i can look in to the magic that is Netlifyās built-in form service
Day 36
Feb 5 2020
i didnāt have an opportunity today to actually right any code but i did do some thinking about a conversation i had on Twitter last night.
It started because it dawned on my that iāve not actually completed any of the task set out by #100DaysOfGatsby, this is largely due to having spent some months working with Gatsby and i feel iāve already accomplished the first few tasks in one way or anotherā¦
ā¦until now!
I noticed the challenge for week 6 was to integrate a serverless form solution, this is something that might be useful in my theme gatsby-theme-gatstats as currently thereās nothing in the demo relating to a ācontact meā page.
I had a look over the links in the challenge email and thereās a mention of using Netlifyās built-in form service and at first glance it appears to be really straight forward and all you really need to do is add an attribute to a form element and Netlify does the rest.
Now, this sounds great if youāre building youāre own blog or developing a site for someone else but i did wonder how it would work within a theme.
Iāve come to the conclusion that the theme doesnāt really need to expose this functionality as it then limits the themes
users to only using Netlify forms. Instead i think what the theme can do is provide some UI components that will work if
theyāre dropped in to an .mdx
fileā¦ yesssss the joy of .mdx
So thatās what iām gonna do, iāll probably create two new components, an input field that will spread the props for an html input element and a text area component, probably the same deal with the props. The theme is written in TypeScript so i might need to explicitly define the ānetlifyā attribute and perhaps iāll create a ācontactā page in the demo site, plus add some documentation to the README and because iām using Storybook to develop the components i can also add some notes in there about how to use the form components.
I think thatās the theory done, now i just need to write some code.
Day 35
Feb 4 2020
Today was a travel dayā¦
London Heathrow - Colombo Sri Lanka. Iāll be here for the next 30 days working on various Gatsby related things.
For those of you who donāt know, iām a contractor, i have been self employed for nearly a decade now and have learnt along the way itās important to take time off to up-skill.
The main reason is when youāre a contractor, youāre the person ātheyā bring in because you know what youāre doingā¦ but i do sometimes find that whilst on contact iām not really learning anything.
Of course working is great as i need to earn a living but if i only work and never up-skill, at some point iāll be out of touch with whatās going on.
This is my main reason for taking the next 30 days off to up-skill. Granted i could have done this in England but why not travel and see the world and more importantly ā¦ be warm!
The next few days might not be that productive as iāve got a bit of admin to take care of but i did manage to do some work on the fight without internet access, and because of this i decided to finally transfer my commercial portfolio over from a site i built using next.js and a self hosted headless version of Ghost CMS on digital ocean.
It was actually the reason i built my theme in the first place. I wanted to have a commercial portfolio and a blog that shared the same look at feel. I also wanted to go serverless which lead me to discovering Gatsby.
Iāve pulled down all the content i need from my CMS and have started to create new posts in the new site which uses my theme gatsby-theme-gatstats. Itās quite a boring task if iām honest but i think iāll just chip away at it over the next few days as i can do most of the work without internet access.
Thatās all for today. Iām hungry and tired!
Day 34
Feb 3 2020
A fantastic day today! Iāve been continuing to problem solve an issue with gatsby-mdx-embed and iām pleased to report, itās fixed!
The way the plugin works is by using the wrapRootElement
api in both gatsby-browser.js
and gatsby-ssr.js
which
does what it says on the tin and wraps the root element of the site with a component from the plugin.
In the case of gatsby-mdx-embed itās wrapping the root element with
the MdxEmbedProvider
, this in turn wraps the root element with an MDXProvider
which handles the passing of React
components from .mdx
and executes them so they render in the dom.
In all cases when running gatsby develop
this āwrappingā was working as expected but users were seeing issues when
running gatsby build
.
I started my bug fixing investigation by adding a div
with a border around the MdxEmbedProvider
and found that when
i ran gatsby develop
it was there but not when running gatsby build
This explains why users were seeing the problems with the components not rendering as expected but it took a little more investigation to work out why.
It turned out in my case to be the method i was using to export the MdxEmbedProvider
in gatsby-browser.js
and
gatsby-ssr.js
see below
// gatsby-browser.js
export { wrapRootElement } from './wrapRootElement';
// gatsby-ssr.js
export { wrapRootElement } from './wrapRootElement';
and wrapRootElement looks like this
// wrapRootElement.tsx
import React, { FunctionComponent } from 'react';
import { MdxEmbedProvider } from './components/MdxEmbedProvider';
interface IWrapRootElement {
element: React.ReactNode;
}
export const wrapRootElement: FunctionComponent<IWrapRootElement> = ({ element }) => (
<MdxEmbedProvider>{element}</MdxEmbedProvider>
);
The first thing i changed was the name of this ācomponentā because wrapRootElement
is part of the gatsby api and i
wasnāt sure if something was getting screwed up due to a clash in names. Iāve since changed the name of this file to
provider.tsx
The second thing i looked at was the method of exporting the module, this now looks like the below
// gatsby-browser.js
exports.wrapRootElement = require(`./provider`);
// gatsby-ssr.js
exports.wrapRootElement = require(`./provider`);
and the provider now looks like this
// provider.tsx
import React from 'react';
import { MdxEmbedProvider } from './components/MdxEmbedProvider';
interface IProviderProps {
element: React.ReactNode;
}
module.exports = ({ element }: IProviderProps) => <MdxEmbedProvider>{element}</MdxEmbedProvider>;
Whilst i think i could have kept the ES6 method for exporting in gatsby-browser.js
it wasnāt working in
gatsby-ssr.js
.
Changing the method in both files seems to have done the trick.
At this point i released 0.0.16
but then due to some weirdness with re-naming the file from Provider.tsx to
provider.tsx i had to release 0.0.17
but either way this is now fixed! Hereās the full run down
#11
Boogy time!
Day 33
Feb 2 2020
Kind of productive day today. Iām continuing to bug fix gatsby-mdx-embed.
At this point however iām not entirely sure there is a bug with my plugin. Iāve spun up a minimal repo and installed all the relevant peer dependencies and have installed one of the āproblemā plugins that was reported to have been causing issues.
So far i canāt re-create the problem. Which leads me to believe itās not a problem with my plugin but might be related
to having a project that uses multiple MDXProviders
I did update the README and have explained
how to manually wrap the MDXRenderer
with the MdxEmbedProvider
which is effectively what the plugin is doing but it
does appeat that importing and wrapping manually solves all the problems.
Iāll continue to investigate this tomorrow.
Day 32
Feb 1 2020
Today was a bug investigation day. Iāve had two issues raised now relating to gatsby-mdx-embed and what appears to be a āclash of the providersā.
The plugin works by wrapping a sites MDXRenderer
with itās own MDXProvider
. This provider is called
MdxEmbedProvider
, and is handled by a gatsby-ssr
method called wrapRootElement
This is the core of the plugin and means that by adding @pauliescanlon/gatsby-mdx-embed
to your gatsby-config
the
plugin can catch any components referenced in .mdx
files and converts them into the components defined by the plugin.
Unfortunately when other gatsby plugins are used in a site that also use this method it prevents the custom
MdxEmbedProvider
from working correctly.
There is one work around that seems to do the trick and thatās to manually wrap the MDXRenderer
with the custom
MdxEmbedProvider
.
eg:
// layout.js
<MdxEmbedProvider>
<MDXRenderer>{body}</MDXRenderer>
</MdxEmbedProvider>
This indeed solves the problem but iām still not sure why when you have multiple plugins that use the wrapRootElement
you experience these bugs.
While developing the plugin i asked @chrisbiscardi if he knew of any reason why multiple providers couldnāt be used. This was his response.
Multiple providers will merge the components object. Last provider wins
Iāll continue to investigate this bug and hopefully be able to release a new version with a fix in the coming days.
Day 31
Jan 31 2020
Not a whole lot of code completed today but i have been tinkering with āProject Propā. The code i have done today has been to sense check that i can later implement what iām designing. I do find myself doing this on my side projects where iāll do half in browser and half in āa hemā Photoshop ā¦ yeah i need to switch over to Sketch!
I usually start with a prototype and build it to kinda prove that i have at least some kind of functionality in place, then i attempt to design it all in browser, then i get fed up with it not looking the way i want and then finally i give in and open Photoshop.
Thatās what iāve pretty much done today, i will say however experimenting with the CSS as i go does at least mean that i know i will be able to build what iām spending time designing.
Iāve made good progress today and have also been playing around with fonts, colors and table styles. Fortunately a lot of the CSS i will need in the prop table can be borrowed from one of my other Gatsby plugins gatsby-remark-sticky-table.
Iāve also had some other thoughts about how to extract what i feel will be some useful prop related information and display it in a kind of data visualization way. Who knows, there may be donut charts in āProject Propā
Day 30
Jan 30 2020
Itās been a slow day. I started the morning by investigating an issue with gatsby-mdx-embed reported by Scott #133 turns out it was a clash of MDXProviders and a little re-jig was all that was required to fix it. I should probably add a comment to the README explaining what to do if this happens.
Other than that iāve been wrestling with tables and css on āProject Propā itās coming together but itās taking me longer as i donāt want to put my design hat on ā¦ knowing full well i will have to eventually but itās fun to design in browser!
Day 29
Jan 29 2020
Had another pretty successful day today, i continued working on āproject propā and after chasing my tail for a bit iāve changed tact slightly.
To start off my thinking was to create a plugin, and i approached this the way i have with some of my other plugins but i realized after a few hours of banging my head against a brick wall that some of what i need to do is already supported out of the box if i take the theme approach.
So thatās what iāve done.
āProject propā name tbc is for all intents and purposes a plugin and will act like a plugin but under the hood itās actually a theme which is a little confusing but now iāve got my head round it i think itās the right way to go.
The next issue iām facing is the style stack.
My initial thought was to use theme-ui but what iām finding is that iām accidentally overwriting styles in the host site. This is the expected behavior of a theme-ui so i can understand why itās happening, the problem is of course how to stop it from happening. I need to go back over the theme-ui docs and see if thereās an escape hatch or perhaps some kind of order of specificity class name i can use somewhere.
Failing that i might be tempted to use Styled-Components or maybe even go back to CSS Modules or Scss.
In either case iām pretty happy with my progress today and if i can solve the styling problem i can carry on with some Design and UX work.
Day 28
Jan 28 2020
Today has been pretty chilled. Iāve solved a lot of the problems with the two main projects iāve been working on and iām happy for them to be used and iāll be keeping an eye out for issues or feature requests which iāll continue to work on as and when.
Iām pretty new to open source and iāve gotta say itās pretty cool that people like my stuff enough to use it and to raise issues and contact me on twitter.
Iām enjoying hearing about use cases and in quite a lot of cases i feel these issues are actually really nice improvementsā¦ so a big thanks to anyone who iāve spoken with lately!
Since thereās no fires to put out i decided to kick start another project thatās been in the back of my mind for a while. Itās quite pertinent to my day job of developing custom React component libraries - for companies that make bazillions, and for almost 4 years now my go to, must have tool has been Storybook
If youāve never used Storybook, you should, itās amazing! Itās ideal for Component Driven Development as it allows you to just focus in on the one thing youāre working on and not worry about the larger applicationā¦ well some times you might have to consider where the component will end up but usually i aim to solve for the 80% of use casesā¦ butā¦
One thing missing from Storybook is a method to provide an holistic view of all the props, not just the ones in the component youāre developing on that day. The prop tables in Storybook are second to none, donāt get me wrong! but you canāt easily see the props and prop descriptions you wrote for a similar component four sprints back. To see these i usually open another browser and put the windows / stories side by side, or use the split view option in VS Code.
But this sometimes still isnāt good enough.
One important factor in the creation of Component Libraries is api consistencyā
By that i mean itās really not good having a prop called CardHeader
on a <Card />
component and then having a
PanelHeading
prop on a <Panel />
component, or worse, a prop that is actually a header / heading but called
something completely different.
So iāve started to think about writing a plugin that will provide me with a single view of all my props and their descriptions, iād hope to be able to filter / search, sort and perhaps even spell check all in one place.
The method for delivering this i think at the moment will work very similarly to how Jest create their coverage reports.
You run a build on it and it creates a static site right there in your repo, you can of course .gitignore
the
directory or deploy it.
Given that creating static sites is kinda the Gatsby thang, it seems like an obvious choice, but i have run in to one or two problems with it today. Mainly how to create the static build bundle and move it to somewhere more usefulā¦ all from the plugin.
I think iām on the right track and iām sure iāll have more to say tomorrow.
Toodle-pip!
Day 27
Jan 27 2020
Another sweet ass day today!
Iāve refactored gatsby-mdx-routes to use TypeScript and now it follows the same yarn workspace pattern as gatsby-mdx-embed and gatsby-theme-gatstats which is important to me as iām working across multiple projects and just prefer it if they can all be developed the same way.
I did have some problems though. First off iām not using babel/preset-env
because Gatsbyās static query canāt be
compiled by in a plugin like iām doing so i have to just convert my .ts
to .js
as ES6.
I imagine when the plugin is used by another Gatsby project the compiling to browser friendly code will be taken care of by the Gatsby project using the plugin. š¤·āāļø
Once the .ts
was compiled to .js
as EE6 i started to see a āmultiple graphql queryā error, this was because i had
named the graphql query and it canāt exist in both .ts
and the compiled .js
in the same project.
Removing the name of the query cleared this errorā¦ not sure if thereās a more complicated issue iām yet to find about using unnamed graphql queries? time will tell.
I also found that using staticQuery
in gatsby dev
was fine but when i ran gatsby build
and gatsby serve
i was
seeing gatsby Loading (StaticQuery) instead of the graphql response. I did have a read of some GitHub issues but
couldnāt find anything solid so just decided iād just give the new useStaticQuery
hook a go! and boom š„ it worked.
gatsby-mdx-routes is now fully converted to TypeScript! ā¦ iāve just gotta go back over my code and correctly type everything š«
Day 26
Jan 26 2020
Pleased to report i had an excellent start to the day!
My investigation into how to solve this docz
site TypeScript prop tables issue went really well and i found that by
shadowing the Props
component from gatsby-theme-docz
i was able to re-wire the props to enable correct and full
population of the prop table.
This his how i did it.
-
Shadow the
Prop
component by creating a new one in my demo site atdemo/src/gatsby-theme-docz/Props/index.js
-
Remove the
prop
prop from the component, which i think is actually theof
prop -
Create a new prop called
name
-
import and use the
useDbQuery
hook š£ -
Filter the
useDbQuery
by the newname
prop -
Pass the result on to the
Prop
component
The new component now looks like this:
...
import { useDbQuery } from "gatsby-theme-docz/src/hooks/useDbQuery"
export const Props = ({ name, getPropType, isToggle }) => {
const db = useDbQuery()
const entries = Object.entries(
db.props.filter(
prop => prop.value.length > 0 && prop.value[0].displayName === name
)[0].value[0].props
)
return (
<div sx={styles.container} data-testid="props">
{entries.map(([key, prop]) => {
return (
<Prop
key={key}
propName={key}
prop={prop}
getPropType={getPropType}
isToggle={isToggle}
/>
)
})}
</div>
)
}
ā¦and to use it i now do this,
<Props name='Gist' />
instead of this
<Props of={Gist} />
Iām not sure if this is the best way to make this work but i think until docz
have a more solid way of creating prop
tables for TypeScript components itāll have to do.
Iāve tested this in dev
and in prod
and it works!
So, as of this morning i released 0.0.15
of gatsby-mdx-embed
which is fully working with TypeScript and populated prop tables in the
demo site
š„³
Day 25
Jan 25 2020
Had another day of wins and loses. I got docz
site prop tables finally working and itās able to read props and prop
descriptions from my TypeScript files
butā¦
only when running in dev
not prod
!
This is both mad and extremely frustrating!
When i run gatsby develop
all is well, when i run gatsby build
then gatsby serve
i get a flash of prop tables,
then nothing.
My current course of investigation is to shadow the Props
component, work out what itās doing and try and find what
feeds it the props
prop.
I think at the moment this comes from the docz
core which is not part of the theme so thereās nothing i can really do
from my end š¢
butā¦
iāve had an idea which involves using the useDbQuery
hook and passing the props on to the Props
component manually.
This so far is proving very difficult but using a lot of console logs and a number of JavaScript array methods i think i can manipulate the data into a shape that will work in both dev and prod.
Day 24
Jan 24 2020
Tricky day today.
On one had it was great, iāve got babel and tsconfig playing together nicely and the two combined take my .tsx
files
and output browser friendly .js
This is perfect, and now i feel like i have a much better understanding of both tsconfig and babel.
Iāve also learned that no matter what your āmainā key in package.json is doing Gatsby ignores it and always looks to
the root of your project forgatsby-browser
,gatsby-ssr
andgatsby-config
. I suspected this was the case but thought
that i might be able to work round it.
Iāve also got both the plugin and demo dev processes working together.
The demo site for gatsby-mdx-embed and the plugin code exist in a monorepo controlled via yarn workspaces.
In order to develop the plugin code i need babel running in --watch
mode so that when i start the Gatsby develop
process both projects hot reload when i make a code change. I was able to accomplish this with npm-run-all
.
Things were going well until i had to address the docz site propTable issue again and iāve spent nearly 8 hours trying to find a way to make it work.
docz site can read props from components in a monorepo if you tell it where to look, naturally it works with React PropTypes and using a boolean in the docz config it can also read TypeScript interfaces and populate propTablesā¦ but no matter what i did today i couldnāt get it to work.
Iād all but given up hope until i posted a message on Spectrum and watta ya know Rakan Nimer one of the maintainers has replied.
If you read along that thread heās suggested changing the repo setup. This might mean dropping Yarn workspaces but if thatās gonna make these bloody PropsTables work iām all for it!
Day 23
Jan 23 2020
Hmmmm TypeScript! I thought iād sorted this already but after my old chum Scott started using gatsby-mdx-embed he noticed that he was seeing some TypeScript related issues.
This plugin is written in TypeScript but it should transpile back down to commonjs so you can use it in both TypeScript and JavaScript projects. This wasnāt quite the case!
Iāve since been using babel and gooooood lord is it great! Iāve done TypeScript setup in the past and it was always a combination of mashing tsconfig, babel and webpack together until you had a config that worked.
This was always a ball ache because youād have to maintain both babel and webpack but now iāve discovered babel 7 itās got so much easier!
The tsconfig is still a little bit of a mystery to me but the new babel presets make sense, even if you just read them by name itās pretty clear what they do.
"@babel/preset-env"
"@babel/preset-react"
"@babel/preset-typescript"
This is pretty much all thatās needed to covert back to browser compatible JavaScript and with only a few settings in tsconfig setup itās just a case of setting the input and output for babel and it goes off and builds out .js modules!
"build:js": "babel src --out-dir lib --extensions \".js,.ts,.tsx\" --source-maps inline",
This was all going really well until i ran into that docz
site issue again with it not populating prop tablesā¦ gonna
come back to that, the more pressing issue is i think iāve got some weird Gatsby thing going on.
Babel now coverts all .ts
to .js
and moves it in to a lib directory, my package.json has a āmainā key which points
to this lib directory but for some reason gatsby-browser.js
and gatsby-ssr.js
never seem to run.
Iāve put the word out so hopefully iāll get this resolved soon!
Day 22
Jan 22 2020
Had a pretty sweet day today, i continued working on gatsby-mdx-routes and focused on the on the issue i described yesterday which was to enable a way to create a dynamic recursively created navigation object based on slugsā¦ what a mouthful!
Taking a step back for a second iāll describe the problem as i see it.
When creating navigation in any website you need to think about the depths a route might be at.
Example, a top depth navigation might look like this
|-- home
|-- services
|-- contact
ā¦which is pretty easy to display an an html list, but then you get to a navigation element that might have a second or third depth, eg;
|-- home
|-- services
|--- web design
|-- user interface
|--- web development
|-- front end
|-- mobile first
|-- backend
|-- contact
At this point you need a way to have nested navigation elements within headings, and since you donāt know how many levels deep an element might be you need a solution that will cater for an element that could be 10 levels deep.
Iāve seen a lot of examples of this problem āsolvedā by suggesting the use of a frontmatter property that can be used to group menu items together by a parent heading.
If youāre creating this in your own project then thatās fine, youāre in charge of both the frontmatter and the graphQL query that fetches the dataā¦ but what if youāre not?
You may or may not know that if you add a property to a graphQL static query and that property is not found in at least one file in your file system you get an error.
This is the problem i faced with gatsby-mdx-routes, if i add a
property called menu
for example, then at least one file in the project must contain it, but if you donāt want to
group your navigation elements youād still need to add this property to frontmatter to avoid getting an error.
There are a couple of solutions to this.
- I could create a hidden file somewhere in the plugin that can be used as a place where all frontmatter properties exist but the file is never renderedā¦ š¤¢ā¦
- Set a default using
createSchemaCustomization
in gatsby-node.js ā¦ again š¤¢
Until Gatsby resolve what i understand to be a really complicated problem of allowing the graphQL static query to fail gracefully if no frontmatter is found then iām sticking by my original thoughts.
āI donāt think the answer is to add any additional properties to frontmatterā
Which leaves me with one other option to determine how a navigation element should be groupedā¦ slugs
Youāll have probably seen when you inspect slugs
that they represent a location on your file system, the chances are
youāve put all your blog posts in a posts
directory in your project, you might have even created sub directories for
year and month.
All this information is contained within the slug
and in effect all weāre trying to do in the browser is mirror what
your file system is doing.
A visual exampleā¦
|-- home
|-- services
|--- web design
Given the above directory structure the slug
(s) would be as follows;
- home = ā/ā
- services = ā/services/ā
- web design = ā/services/web-design/ā
This is all the information we need to determine how many levels deep any given navigation element isā¦ but getting at it was a real challenge for me.
My approach was to split the slug
using slug.split("/")
['', ''];
['', 'services', ''];
['', 'services', 'web-design', ''];
Then i remove the empty strings so iām just left with what is effectively an array of parents and children.
There is a bit more to it, but the code can be see here if youāre interested.
Using this slug
approach i donāt need anyone using this plugin to do anything to their frontmatter, meaning i think
this could easily be implemented into a blog / site that had loads of pages just as easily as it could be to a brand new
project.
Thereās more info about how to use this plugin in the README and hereās a very computer science looking demo
Happy routing! š
Day 21
Jan 21 2020
Recursive functions continue once more. I mentioned yesterday about an issue that had been raised with gatsby-mdx-routes and after a little digging it think one of the problems can be solved with a recursive function.
Thereās more on the issue here
Actually let me start again. gatsby-mdx-routes is a little plugin
that creates routes and generates navigation labels from frontmatter
and originally i thought the issue was related to
how could routes be grouped into menu headings as per the work iāve been doing on the docz
site but after asking that
question it was communicated that wasnāt the issue.
While i waited for a response i cracked on with a little enhancement to the MdxRoutes
component by creating a prop to
allow re-ordering of navigation routes.
For instance if you had Home , About and Contact youād probably want Home to be the first route returned but because i wasnāt sorting the results they were just returned in alphabetical order.
I thought about having an ASC
and DESC
option but that doesnāt really solve the problem so i decided upon the
reference array option.
Suppose you had a graphQL response that looked a little bit like this š
graphQlData = ['About', 'Contact', 'Home'];
By providing a navigationOrder
you can sort like this š
const navigationOrder = ['Home', 'About', 'Contact'];
graphQlData.sort((a, b) => navigationOrder.indexOf(a.navigationLabel) - navigationOrder.indexOf(b.navigationLabel));
Unlike the default method of sorting instead of simply determining if a - b
we can use the indexOf
from the
navigationOrder
array which is provided via a prop.
Great i thought, thatās that issue closed and released 0.0.5
But nope. I was notified theyād been a new comment on the issue and it was mentioned again that a new property in
frontmatter
was required to sort the navigation itemsā¦ āWT Flipā i thought, iāve just provided a solution for that.
A few comments later and we have now established that itās both a grouping and an ordering issue.
In the first instance grouping the menus should be managed by a recursive function and not via a property in
frontmatter
and then once theyāre grouped we can order them before returning.
This does mean however, iām back in to recursive hell! This time iām planning on using the slug
to determine how many
levels deep a menu should be grouped.
My initial thoughts are to take the slug. eg some-folder/some-sub-folder/some-file
and split the string to create an
array and then remove any empty space. I hope to use the array length as the condition to stop the recursive function by
removing the slug segment on each loop before calling it againā¦ weāll see how it goes!
Day 20
Jan 20 2020
Recursive functions research has continued today and although iāve solved what i set out to do with the docz
navigation i wanted to dig a little deeper.
The main reason is i feel the method(s) iāve used to reduce the menu object feel a bit ECMAScript 5, even though theyāre not.
Iām even using let
and if you follow along on Twitter thereās a lot of conversations going on about let
vs const
etc, etc ā¦ but i do feel some of the if
statements iām using, or rather the nested if
statements iāve used to
manage what to do on each iteration of the recursive function feel a bit clunky.
Thereās some really powerful ECMAScript 6 array methods available and i donāt feel iām taking advantage of them and as a result my reduce function isnāt that readable.
I became a little frustrated today so perhaps itās best if i move on to something else and let all this sink in and try to tackle the problem again at a later date.
Ironically iāve since noticed a new issue has been raised on one of my other Gatsby plugins gatsby-mdx-routes which is actually intended to make file path and menu navigation easier. š¤¦
I think this will be a job for tomorrow now but perhaps itāll be an opportunity to test drive my new found knowledge of recursionā¦ which in itself is actually kinda recursive. š¤
Day 19
Jan 19 2020
Made good progress today on the docz
recursive submenu problem and have it pretty much sorted.
Iām now able to group menu items together by defining a submenu property in the frontmatter You can see it working in the Components menu with Twitter and Pinterest acting as submenus here
The code needs a bit of cleaning up and iāll document my approach and solution in a following blog postā¦ i just gotta make sure i fully understand what i did first š¤
If youāre interested though iāve pushed to master so you can see the amended Sidebar component, the new reduce method and the new SideNavGroup componentā¦ you gotta love Component Shadowing!!!
Day 18
Jan 18 2020
I did a bit of work this morning on recursive functions to make the navigation in docz
work for me. Luckily since
docz
is also a Gatsby theme iām able to tap in to component shadowing. I started by looking at
gatsby-theme-docz/src/components/Sidebar
and investigated the menu
object.
This, as it stands doesnāt have a way to group by submenu but the theme does allow you to add an additional fields to frontmatter.
This is quite key given i need a way to determine if a file is part of a menu as well as part of a submenu.
Iāve since added a submenu property to the .mdx
files that i will use to āgroup byā and have set to work on altering
the menu object created by docz
core and recursively looping over any object that contains a submenu.
Now i have an object that i can use to drive the UI.
Iāve also been working on another blog post explaining recursive functions and will continue to update it documenting
the problem iām facing and how i intend to solve the docz
theme submenu problem.
Day 17
Jan 17 2020
This morning i was attempting to work on a recursive submenu solution for the navigation in the docz
site used by
gatsby-mdx-embed and as i worked through a solution i thought about how this
might make a good blog post.
I then thought, if i start to write the post in this blog the next time i commit and publish that post will also get published.
This blog uses my theme gatsby-theme-gatstats which currently has no way to set a post as draft which would exclude it from being published.
I then worked on my theme and have included a new property in the frontmatter called status
. If status
is set to
draft the post wonāt be displayed and none of the charts will attempt to reference it as a data source, thus not
skewing your metrics š
That issue was raised on GitHub #26 and now felt like a good time to address it.
Iāve had another issue raised today too, this time with gatsby-remark-sticky-table A user wanted to put images in a table cell.
Thereās probably something funky going on with the way iām parsing the node to the new markup so i suggested this solution and might investigate this another time.
Rightā¦ back to the recursive submenu problem!
Day 16
Jan 16 2020
This morning i was having a google around and out the corner of my eye i saw a Pinterest logoā¦ ooh i thought, i wonder if Pinterest have embed-able widgets?
They do! so that was me for the next few hours!
gatsby-mdx-embed now has three new components for Pinterest.
The standard <Pin />
where you can embed an image from Pinterest, a <PinterestFollowButton />
and a
<PinterestBoard />
with an optional prop so you can create a board from a user or a board from a board, if that makes
sense.
The inject script works very similarly to Twitter, Instagram and Flickr but i did need to do a little digging into the
window object to find the build()
function. This needs to be called on route change so that any components on the page
will execute and display as expected.
This evening iāll be looking at how to make a change to the docz
theme so i can have nested menus. I have a feeling my
old enemy array.reduce()
may be required and it doesnāt matter how many times i read or write an array.reduce()
i
always struggle!
Day 15
Jan 15 2020
I spent some of the morning updating prop names and continued to work on two new components, <TwitterTimeline />
and
<TwitterList />
and i really like the Twitter method for handling embeds!
The method iāve used to inject the Twitter embed script remains in tact and all the other Twitter embeds work without any additional injecting, which leaves me time to just develop the components and ponder about what to name the props.
At this point i was really regretting not implementing TypeScript š. I use TypeScript for my day job and in my bigger projects like gatsby-theme-gatstats and developing without it feels like being nude and leaving the house.
As development continued iād been wondering if gatsby-mdx-embed was
going to need TypeScript because PropTypes
kinda has my back but iām now feeling really un-easy with working towards a
stable release without it so implementing it now is an absolute must.
This evening iāve refactored everything and gatsby-mdx-embed is now TypeScript enabled šš„³.
It feels like a massive relief to know thatās sorted and it wasnāt too much of pain to setup. docz
did give me a few
headaches with it not generating the prop table from the TypeScript interfaceā¦ until i discovered that i needed to set
typescript: true
in the doczrc.js
With TypeScript now in place iām more comfortable with continuing to develop components and perhaps 1.0.0
will happen
sooner than expected.
Day 14
Jan 14 2020
Last night i started to implement Twitter buttons and hit a naming issue. Iād previously named the components after the provider they represent. In the case of a Tweet this is fine because a Tweet is a unique thing but, what do i do with a hashtag button?
Twitter isnāt the only provider tha uses the hashtag paradigm and so iāve had to start thinking about prefixing names to allow for this plugin to scale. Maybe at some point iāll be able to introduce an Instagram hashtag button. š¤·āāļø
This led me on to the next problem. Props!
To give some context. I build commercial React Component Libraries for companies that make bazillions and these Component Libraries are typically used by lots of engineers on various projects so iām familiar with the trouble with naming things.
I have learnt that keeping a consistent api for prop names is key.
For instance if you have a number of components that, letās say accept some kind of render prop for a heading, this prop across the entire library should always be called heading. It makes no sense if in a Card component you name it CardHeading and then in a Modal name it Header.
I was trying with gatsby-mdx-embed to keep a consistent id
prop
across all components but iāve now hit a point where id
just doesnāt cut the mustard. š
On some components like YouTube the id
is the video id and thatās how YouTube refer to it but, in a Tweet for instance
i need the prop to be less ambiguous.
I had some interesting Tweets back and forth with @AskGatsbyJS and John Otander and the general consensus was in Open Source Libraries it pays to be clear and it doesnāt matter that names or prop names are long. I think thereās a post or Tweet out there by Dan Abramov about how the function names at Facebook are really long, but the upside is itās clear what they do.
Iāve tried to keep some consistency and the id
paradigm is still in use but now itās more explicit youTubeId
.
For components that require more than an id
, Spotify and Gist for instance where the prop includes a word tracks
or
album
AND an id
iāve gone with ...Link
. I thought about using URL
in the prop name but the prop doesnāt contain
http
etc so itās not really a URL.
Interestingly though when i have username
as a prop i donāt feel i need to change this to, e.g twitterUsername
, same
thing with the hashtag
prop. If youāre using a component that has the provider in the name like
<TwitterHashtagButton />
i hope itās clear that the hashtag
prop refers to a Twitter hashtagā¦ naming things is
hard š©
A note on sem-ver. Strictly speaking these are breaking changes but i feel i can
justify not bumping the plugin to a major release version of 1.0.0
because in my mind when a library is only at
0.x.x
to start with itās still a pre-releaseā¦ even though i have already released it. š¤Æ
Iām not sure if this is a bad thing to do but i suspect iām going to encounter a lot more of these kinds of naming
issues before iām happy to release a stable 1.0.0
.
I have started a CHANGELOG which should help you fix any issues, and of course the docz will always reflect the actual state of the library.
I hope this change hasnāt or wonāt cause anyone any serious headaches but if you are having problems feel free to Tweet me @PaulieScanlon
This morning and this evening will mostly be taken up by combing over the docs and iāll be thinking hard about sensible names for both components and props.
TTFN!
Day 13
Jan 13 2020
This morning i continued to work on the Wikipedia component and switched to using the wikipedia rest_v1 api
I had a little trouble with the response, i was parsing response.json()
but i needed response.text()
This can now be injected into the srcDoc
of an iframe
which retains all the wikipedia styles but has created a new
problem.
The iframe
now needs some dimensions. By default iām setting the width as 100% and iām allowing for a height prop
which means control of the height is handed over to whoever is using this component in their .mdx
I think this will be enough for now and should i get any issues on GitHub iāll deal with them when they arise.
oh, and hereās a demo of the finished component.
Day 12
Jan 12 2020
Today i continued to work on the Wikipedia component and have a new method in place for querying the Wiki api. The new method uses the fetch api which has cleaned things up a lot and means thereās less JavaScript being injected into the page but the thing thatās stumped me now is that the response iām getting using this approach no longer returns a stylesheet.
I can render the page data but the content picks up the styles from the docz
site rather than the usual Times New
Roman style from Wikipedia that we are familiar with.
There are alternative methods documented in the Wikipedia api for fetching the css but iām having trouble getting both the stylesheet and the data in the same response. Iāll just have to keep digging!
The other issue i encountered was that the api response returns All links as relative. For instance for my test page
iām querying āThe_Jimi_Hendrix_Experienceā and the href
ās that are returned look like this
wiki/The_Jimi_Hendrix_Experience
when in fact they need to be prefixed with https://en.wikipedia.org/
https://en.wikipedia.org/wiki/The_Jimi_Hendrix_Experience
using a regex pattern iām replacing everything that has wiki
in the href with the url.
Iāve never really understood regex but i did spend some time reading, watching and learning about the bits i needed.
Hereās the final regex pattern /<a href="\/w/g
which when used in the response looks like this
text['*'].replace(/<a href="\/w/g, '<a target="_blank" href="//en.wikipedia.org/w');
I added the target="_blank"
as it feels like a better option for users, perhaps iāll turn this into a prop but it
works like a charm so iām happy to move forward.
Day 11
Jan 11 2020
Itās Saturday and i didnāt have a lot of time today. I did continue working on the Dropbox Chooser component but i think i need to shelve development on this for the time being.
After making good progress i noticed that in order to get it hooked up iād need to inject the script for each and every Chooser button.
The repercussions are that this could really affect page load speed and that really goes against what iām trying to achieve with this plugin
Iāve pushed a branch so i can keep my work but for now iām going to move on to something else.
Iāve since gone back to my list of oEmbed providers and have decided to focus on a Wikipedia component. It feels like it could be useful and the api looks to be well documented.
Upon initial inspection i think the best course of action is to fetch
the āpageā from the Wiki api, then render the
response using dangerouslySetInnerHTML
. Iāll see how it goes and no doubt report back tomorrow.
Day 10
Jan 10 2020
Last night i decided to design a logo for gatsby-mdx-embed i was hoping it would be like the Fight Club logo in a bar of soap but it didnāt quite work out that wayā¦ i was a designer a long time ago but i think iāve now just run out of talent. š¤·āāļø
Another early start again this morning and i started to think about how users might use this plugin and itās not always
clear what iām referring to as an id
in the props. As of this morning iāve created a
Help page which documents how to extract the id
the component
needs from the providers embed code or URL.
This evening iām planning on adding Dropbox chooser to the list of components. I donāt even use Dropbox but it has an embed-able script tag so it belongs in gatsby-mdx-embed
Day 9
Jan 9 2020
I had a productive morning this morning and have added Twitch to the list of components gatsby-mdx-embed supports.
While i was digging around the Twitch embed code i noticed the skip to option which allows you to embed a video and start it a certain point. This seems like a pretty useful thing to include so i added it as an optional prop along with auto play.
The way Twitch handles the time code is by having the following parameters as part of the URL.
&t=0h21m14s
t
is the time parameter which accepts a numerical value followed by the time digit h
for hours, m
for minutes and
s
for seconds. Iāve exposed a skipTo
prop on the Twitch component so you can pass through a time-code.
skipTo={{ h: 0, m: 21, s: 14 }}
I then thought iād add this to the YouTube component, but YouTube do the time-code a different way.
The URL parameter looks like this
&start=1274
After doing a bit of maths i realized the time-code is the total minutes from the start of the video. I wanted to keep the props the same as iāve done with Twitch so i needed to calculate the total time before i pass it on to the URL.
const { h, m, s } = skipTo;
const tH = h * 60;
const tM = m * 60;
const startTime = tH + tM + s;
// startTime = 1274
First i de-structure the h
, m
and s
from the skipTo
object then multiply h
by 60 and also m
by 60, you donāt
need to do it for seconds as the start
time is in minutes.
Then the last step is to add these all together to create something the YouTube URL understands.
Iāve also added the same prop to Vimeo, and luckily it handles the time-code in a very similar way to Twitch so not much work needed to be done on that.
This is now available in v0.0.6
of gatsby-mdx-embed
Day 8
Jan 8 2020
Last night i carried on working on gatsby-mdx-embed, made some
tweaks to the docz
theme and found a list of āprovidersā that are supported by oEmbed.
There where two that caught my eye, SoundCloud and Gist. SoundCloud was pretty straight forward as itās a very similar method to how YouTube and Vimeo so that was no problem.
This morning i moved on to Gists.
I Googled around to see if there were any existing React repos and found these twoā¦ so thanks Christian Korndƶrfer and Miroslav SaraÄeviÄ
The main bit that i didnāt initially understand was how to handle the JSON
callback.
If we start with a typical Gist URL and run it, youāll hit the https://gist.github.com/
https://gist.github.com/PaulieScanlon/ca0cc9239176066492cb2aba435edbf7
To provide a callback we can add on gist_callback
i Googled around for how to add the callback for ages and found no
official docs, just examples of how others have done this.
?gist_callback_ca0cc9239176066492cb2aba435edbf7
This is now the complete URL.
https://gist.github.com/PaulieScanlon/ca0cc9239176066492cb2aba435edbf7?callback=gist_callback_ca0cc9239176066492cb2aba435edbf7
The last step was to inject a <script>
tag with the URL in a useEffect
life-cycle method, create a function on the
window object called gist_callback_ca0cc9239176066492cb2aba435edbf7
then investigate the response.
The response comes back with a div
object which contains all the HTML for the Gist which can be handled by
dangerouslySetInnerHTML
<div dangerouslySetInnerHTML={{ __html: response.div }} />
ā¦and a stylesheet
object which in turn needs to be injected into the page in a <link>
tag.
With both of those things done Gists are now embed-able!
I need to do a bit more work on handling errors but iāve bundled it up and Gists are now part of v0.0.4
Happy Embedding! š§¼
Day 7
Jan 7 2020
Yesterday was a pretty good day, i was relieved to finally get docz
setup the way i wanted. I also launched a very
early version of gatsby-mdx-embed. A fellow Gatsby enthusiast who i
met at Gatsby Days London Scott Spence was keen to try it outā¦ so, here ya go Scott!
This evening iāll mostly be focussing on documenting and testing the props for each component and perhaps if thereās time iāll investigate what other providers i can include. dev.to would be sweet but i had a quick google and thereās an open issue on GitHub regarding this so i donāt know if itās possible yetā¦ stay tuned šŗ
Day 6
Jan 6 2020
Back to my day job today so i got up early and cracked on with the docz.site
ā¦ and pleased to report i got the
<Props>
component working.
I was on the right track with what i thought the problem was and docz
can be configured to source files from outside
the root directory but this hasnāt been documented anywhere.
The problem was twofold.
- What did node think the root was and what is the correct path to pass to
docz
- How the WT Flip do you pass a path to
docz
The way to do point 2 is to add docgenConfig.searchPath
to doczrc.js
which tells docz
where to look for component
props. You wonāt find this in their docs but i did find it amongst the
examples
// doczrc.js
export default {
docgenConfig: {
searchPath: directoryPath,
},
};
and directoryPath
is as follows;
const directoryPath = path.join(process.cwd(), '../@pauliescanlon/gatsby-mdx-embed/src/components');
First iām using path.join
to make sure i donāt end up with un-wanted /
ās, then iām using process.cwd
which is the
node.js method for working out the Current Working Directory then finally i go up a level "../"
and into
the yarn workspace where iām writing my MdxComponents
This now means from the .mdx
file where iām writing the documentation about the component i can also have a nicely
formatted prop table which are the real props defined by propTypes
in the component file.
You can see the example for the CodePen component here
Later tonight or early tomorrow iād like to tweak the docz
theme styles a bit then finally i can crack on with
developing the MdxComponents
Oh and dark mode now works! š
Day 5
Jan 5 2020
Today i decided to focus on the documentation site as i need a playground
setup so i can test props for each of the
MdxEmbed
components.
Usually in my professional life iād use Storybook but since part of #100DaysOfGatsby is to learn new things iāve gone for using docz.site
It was a bit tricky to setup as i wanted to use gatsby-theme-docz and i found the documentation regarding setup a bit confusing.
None the less itās all looking pretty good now. I might have found a bug with how the plugin options are passed or
perhaps in some cases they have to be set in the doczrc.js
fileā¦ who knows, maybe iāll have to live without dark
mode on ths one!
The problem iām currently experiencing is with the <Props>
component which is one of the
built-in-components which can create prop tables for my components.
Whatās not initially made clear is that itāll only work if the component in question is local to the .mdx
. For example
docz
will generate a prop table for <MyComponent>
it itās in the same things folder as MyMdx.mdx
-things;
MyComponent.js;
MyMdx.mdx;
ā¦ But it wont work if i move <MyComponent>
to somewhere else, for exampleā¦
-components;
MyComponent.js - pages;
MyMdx.mdx;
If you want to be able to source files from somewhere else you have to set the src
value in doczrc.js
export default {
...
src: './components',
}
But my docs site and my MdxEmbed project are two different repos linked together by yarn workspaces so i think i need to tell docz to jump up a level to find my components.
export default {
...
src: '../@pauliescanlon/gatsby-mdx-embed/src/components',
}
Which causes GraphQL to error, presumably because it canāt find what it needs?
Iām currently trying to work out where node thinks the root is and from there i need to refresh my memory about
path.resolve
and process.cwd
but that sounds like a job for tomorrow!
Day 4
Jan 4 2020
Yesterday i spent the morning getting lost in a docs
rabbit hole. I tried numerous documentation themes but all were
tripping me up one way or another so i decided to shelve the docs part of
gatsby-mdx-embed and just get something looking half decentā¦
which led me down another rabbit hole. I had real trouble getting theme-ui and typography treatments to works. I also
discovered that behind theme-ui typography is Kyle Mathews who has been creating a ton of them. I kept having the same
issue though and that was the body font wasnāt being set so i resorted to using emotion core global and css to just set
itā¦ again iāll come back to this.
What i was able to do on gatsby-mdx-embed though was to understand
how to create a method for the provider so that once the plugin is installed thatās all a user would have to do. Itās
similar to how gatsby-plugin-theme-ui
works by using gatsby-browser
and gatsby-ssr
and using the wrapRootElement
method to inject the MdxProvider.
This i discovered needs to be done in both files and in different ways. in gatsby-browser
es6 imports work in
gatsby-ssr
you have to use require and node modules exports.
..both are required so that in dev and prod the MdxEmbedProvider wraps the root element. With this now working iām almost ready for an early release.
ā¦
Day 3
Jan 3 2020
This post is starting to feel like an agile standup so perhaps iāll treat it like one.
Yesterday
After writing yesterdays blog post i had a think about how i was gonna handle the headings with nested <a>
styling in
gatsby-theme-gatstats which was actually pretty easy. Iām
using theme-ui for everything so a little update to my headings object to style the child <a>
was all that was needed.
const headings = {
...
a: {
fontSize: 'inherit',
fontWeight: 'inherit',
lineHeight: 'inherit',
color: 'inherit'
}
}
Next i looked at how these #
anchors should work. Typically in a site the browser jumps to where the #
starts when
clicked and if a link is shared with a #
as part of the url when the page loads itāll move the page so the #
link is
at the top.
For this to work in my theme i needed to look at the gatsby-browser
api. Thereās two methods that were required to
make this to work.
- onRouteUpdate which is called when a user changes routes,
and also called when an
<a>
is clicked (if it has a#
) - shouldUpdateScroll which allows us to influence the scroll position of the browser on load and also it would seem between route changes.
The main guts of this functionality is wrapped up in a little function i created in gatsby-browser
which is called by
both of the above methods.
const anchorScroll = (location) => {
if (location && location.hash) {
const item = document.querySelectorAll(`a[href^="${location.hash}"]`)[0].offsetTop;
const mainNavHeight = document.querySelector(`header`).offsetHeight;
setTimeout(() => {
window.scrollTo({
top: item - mainNavHeight,
behavior: 'smooth',
});
}, 50);
}
};
In short all this is doing is finding the <a>
element that contains the #
which is found from the url /
location.hash
then scrolls to it. Thereās an offset in there because
gatsby-theme-gatstats has a position fixed header so i needed
to calculate the top position minus the height of the header so the selected #
isnāt under the header when the browser
scrolls to it.
This all works well but iām a little worried about my choice to use the native window.scrollTo
method with
behavior: "smooth"
as some older browsers donāt support this so if youāre using IE 11 (š¤¢) can you let me know if it
still works as intended?
I also had a little bit of twitter activity yesterday from a new user of gatsby-theme-gatstats who tweeted to ask me how to do something. He said he didnāt want to raise as issue as he thought it was more him not knowing how to do it rather than it being a bug. My reply āwas go ahead, raise an issueā Reason being is that if heās had problems with this particular āthingā then perhaps i should explain it better in the README so others donāt have the same question.
Iām pretty new to open source and gatsby-theme-gatstats is the first real project iāve had where users are getting in touch and / or rasing issues on GitHub. I think itās fantastic to be honest as itās quite difficult to know how your project will be used and whilst i tried to think of everything i know iāve missed things so having other users point this out and let me know means i get to change and improve the project ā¦ itās a win, win!
Today
Today i really want to focus on gatsby-mdx-embed and get this converted to TypeScript and perhaps get Storybook setup.. but i do like the idea of using a Gatsby project to use as a documentation site so iāll probably have google around and see whatās out there.
Iām also thinking in the background about the subjects iām covering on this āstandupā style post and which ones should
be promoted to their own blog posts. The <a>
#
thing i did yesterday could be a good one to write up but iām not
sure iāll have time. For now at least my main focus is on getting
gatsby-mdx-embed stable and released.
Day 2
Jan 2 2020
Started this morning by fixing a bug with gatsby-theme-gatstats. Part of the dashboard has a Posts chart displaying information gathered from the previous year and current years posts. I wasnāt correctly checking that the chart would render correctly if there were no posts for the current year.
Iāve also noticed another bug which is the chart max range needs to be calculated from grabbing the highest count value from both of the year data arrays. I was previously only checking the current year data and creating the range from the hightest count value in that year but itās quite possible that the previous year will have a higher count meaning the line chart will get cut off.
I also need to investigate anchor tags in the theme so that they donāt always take on the styling for <a>
if nested
within an <h1>
This will come in handy should i want to add anchor tags to each heading from this 100DaysOfGatsby
post.
Day 1
Jan 1 2020
Today is the first day of #100DaysOfGatsby and since i already have a blog in place i think iām gonna just continue working on my various Gatsby projects and try to write a bit each day and then see where iām at in 100 days.
I currently have a number of Gatsby Plugins projects all on npm but all in various states of completeness, these are:
- gatsby-theme-gatstats
- gatsby-remark-sticky-table
- gatsby-remark-grid-system
- gatsby-mdx-routes
- gatsby-mdx-embed
Today iāve pushed a fairly stable commit to gatsby-mdx-embed which i think pretty much solves the twitter, instagram and flickr embed problem. Iām not gonna release this to npm just yet as iāve done with previous plugins because iād like to get this properly tested and converted to TypeScript first.
gatsby-mdx-embed is first and foremost an MdxProvider
and when
used in a Gatsby project along side .mdx
itāll allow for media embed codes to be used without import by using the
MdxProvider
and itās component
prop.
So far so good and i have the first 8 components in and working, although the instagram one seems a little flakey on iOS.
Lessons learned today are similar to issues iāve experienced before regarding gatsby-ssr
and gatsby-browser
. Or more
specifically the problems when attempting to use both together to accomplish similar things.
ie. if gatsby-browser
relies on something that gatsby-ssr
is gonna do which might not run synchronously. I made a
change and it was worth the effort now gatsby-browser
does all the heavy lifting things are working much better.
In the case of gatsby-mdx-embed it was because i was using
gatsby-ssr
to inject the relevant provider scripts into the <head>
and then using gatsby-browser
to invoke them.
This wasnāt really working so making the <script>
tags and appending them to <head>
AND then invoking them from
gatsby-browser
proved more solid.
Iād really like to solve the instagram problem before moving on much further but getting the project converted to Typescript and then installing Storybook sounds like itās gonna be way more fun.
I plan to open up and document a load of props to make each component more useful, adding widths to videos and or aspect ratios etc would be really cool. I havenāt set up a TypeScript / Storybook repo for a while so this should be fun!
Once iām happy gatsby-mdx-embed is pretty solid iām gonna switch
out the temporary Tweet
and YouTube
components i hacked together in my theme
gatsby-theme-gatstats
Iām gonna use this post as a kind of journal and update each day with things iām working on of things iām thinking about and would like to create more in depth posts for topics that warrant further explanation.