Lorem

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

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:

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ā€¦

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.

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

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.

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.

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.

  1. Iā€™ve completed my new theme šŸŽ‰ but.. iā€™m not quite ready to announce it.
  2. 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;

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.

![Some Image](/images/some-image.png)

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%']}}
  >
  ![Some Image](/images/some-image.png)
  <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.

  1. Shadow the Prop component by creating a new one in my demo site at demo/src/gatsby-theme-docz/Props/index.js

  2. Remove the prop prop from the component, which i think is actually the of prop

  3. Create a new prop called name

  4. import and use the useDbQuery hook šŸŽ£

  5. Filter the useDbQuery by the new name prop

  6. 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-ssrandgatsby-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.

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;

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.

  1. What did node think the root was and what is the correct path to pass to docz
  2. 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.

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:

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.