Skip to content

Commit

Permalink
Add experimental react support within markdown
Browse files Browse the repository at this point in the history
Also added fronmatter for interactive: true to turn on React on a file by file
basis and prevent slowing down the builds for non-react files
  • Loading branch information
chiedo committed Oct 18, 2020
1 parent 0547215 commit b781e43
Show file tree
Hide file tree
Showing 16 changed files with 31,834 additions and 581 deletions.
25,147 changes: 25,147 additions & 0 deletions assets/js/react-dom.development.js

Large diffs are not rendered by default.

3,318 changes: 3,318 additions & 0 deletions assets/js/react.development.js

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ intro: 'While you can grant read/write access to collaborators on a personal rep
versions:
free-pro-team: '*'
enterprise-server: '*'
interactive: true
---

### Personal user accounts
Expand All @@ -17,16 +18,33 @@ A repository owned by a user account has two permission levels: the *repository

### Organization accounts

#### Code Block Component
<!--react-->
<CodeBlock language="javascript">
{`var i;
for (i = 0; i < cars.length; i++) {
text += cars[i] + "<br>";
}`}
</CodeBlock>
<!--end-react-->

Organization members can have *owner*{% if currentVersion == "free-pro-team@latest" %}, *billing manager*,{% endif %} or *member* roles. Owners have complete administrative access to your organization{% if currentVersion == "free-pro-team@latest" %}, while billing managers can manage billing settings{% endif %}. Member is the default role for everyone else. You can manage access permissions for multiple members at a time with teams. For more information, see:
- "[Permission levels for an organization](/articles/permission-levels-for-an-organization)"
- "[Project board permissions for an organization](/articles/project-board-permissions-for-an-organization)"
- "[Repository permission levels for an organization](/articles/repository-permission-levels-for-an-organization)"
- "[About teams](/articles/about-teams)"

#### Code Editor Component

<!--react--><RedContent>Red content!</RedContent><!--end-react-->

<!--react--><Timer /><!--end-react-->

{% if currentVersion == "free-pro-team@latest" %}

### Enterprise accounts


*Enterprise owners* have ultimate power over the enterprise account and can take every action in the enterprise account. *Billing managers* can manage your enterprise account's billing settings. Members and outside collaborators of organizations owned by your enterprise account are automatically members of the enterprise account, although they have no access to the enterprise account itself or its settings. For more information, see "[Roles for an enterprise account](/articles/roles-for-an-enterprise-account)."

{% data reusables.gated-features.enterprise-accounts %}
Expand Down
2 changes: 2 additions & 0 deletions includes/head.html
Original file line number Diff line number Diff line change
Expand Up @@ -26,4 +26,6 @@
<link rel="stylesheet" href="/dist/index.css">
<link rel="alternate icon" type="image/png" href="/assets/images/site/favicon.png">
<link rel="icon" type="image/svg+xml" href="/assets/images/site/favicon.svg">
<script src="/assets/js/react.development.js" crossorigin></script>
<script src="/assets/js/react-dom.development.js" crossorigin></script>
</head>
9 changes: 9 additions & 0 deletions javascripts/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,9 @@ import experiment from './experiment'
import copyCode from './copy-code'
import { fillCsrf } from './get-csrf'
import initializeEvents from './events'
import CodeBlock from '../react/CodeBlock'
import RedContent from '../react/RedContent'
import Timer from '../react/Timer'

document.addEventListener('DOMContentLoaded', async () => {
displayPlatformSpecificContent()
Expand All @@ -35,3 +38,9 @@ document.addEventListener('DOMContentLoaded', async () => {
copyCode()
initializeEvents()
})

module.export = {
CodeBlock,
RedContent,
Timer
}
3 changes: 3 additions & 0 deletions lib/frontmatter.js
Original file line number Diff line number Diff line change
Expand Up @@ -70,6 +70,9 @@ const schema = {
quickstart: { type: 'string' },
learn: { type: 'string' }
}
},
interactive: {
type: 'boolean'
}
}
}
Expand Down
27 changes: 26 additions & 1 deletion lib/page.js
Original file line number Diff line number Diff line change
Expand Up @@ -15,10 +15,12 @@ const pathUtils = require('./path-utils')
const Permalink = require('./permalink')
const languages = require('./languages')
const renderContent = require('./render-content')
const { renderReact } = require('./react/engine')
const frontmatter = require('./frontmatter')
const products = require('./all-products')
const slash = require('slash')


class Page {
constructor (opts) {
assert(opts.relativePath, 'relativePath is required')
Expand Down Expand Up @@ -119,9 +121,30 @@ class Page {
this.title = await renderContent(this.rawTitle, context, { textOnly: true, encodeEntities: true })
this.shortTitle = await renderContent(this.shortTitle, context, { textOnly: true, encodeEntities: true })

const markdown = this.mapTopic
let markdown = this.mapTopic
? getMapTopicContent(this, context.pages, context.redirects)
: this.markdown

// If the article is interactive parse the React!
if (this.interactive) {
// Search for the react code comments to find the react components
const reactComponents = markdown.match(/<!--react-->(.*?)<!--end-react-->/gms)

// Render each of the react components in the markdown
for (const index in reactComponents) {
let componentStr = reactComponents[index]

// Remove the React comment indicators
componentStr = componentStr.replace('<!--react-->\n', '').replace('<!--react-->', '')
componentStr = componentStr.replace('\n<!--end-react-->', '').replace('<!--end-react-->', '')

// Get the rendered component
const renderedComponent = await renderReact(componentStr)

// Replace the react component with the rendered markdown
markdown = markdown.replace(reactComponents[index], renderedComponent)
}
}

const html = await renderContent(markdown, context)

Expand Down Expand Up @@ -163,6 +186,7 @@ class Page {
return cleanedHTML
}


// Allow other modules (like custom liquid tags) to make one-off requests
// for a page's rendered properties like `title` and `intro`
async renderProp (propName, context, opts = { unwrap: false }) {
Expand All @@ -179,6 +203,7 @@ class Page {

const html = await renderContent(prop, context, opts)


if (!opts.unwrap) return html

// The unwrap option removes surrounding tags from a string, preserving any inner HTML
Expand Down
23 changes: 23 additions & 0 deletions lib/react/engine.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
const { renderToString } = require('react-dom/server')
const transform = require('./transform')

// These all need to be here even though eslint doesn't think so
/* eslint-disable */
const React = require('react')
const CodeBlock = require('../../dist/react/CodeBlock')
const RedContent = require('../../dist/react/RedContent')
const Timer = require('../../dist/react/Timer')
/* eslint-enable */

const renderReact = async componentStr => {
const componentName = componentStr.match(/[^<]([a-zA-Z])*/gm)[0]
const jsx = `<div className="react-component-${componentName}">\n${componentStr}\n</div>`
const component = transform(jsx)

// eslint-disable-next-line
return renderToString(eval(component))
}

module.exports = {
renderReact
}
13 changes: 13 additions & 0 deletions lib/react/transform.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
const babel = require('@babel/core')
const transform = code =>
babel.transform(code, {
presets: ['@babel/preset-env'],
plugins: [
'@babel/plugin-transform-react-jsx',
'@babel/plugin-proposal-object-rest-spread',
'@babel/plugin-transform-modules-commonjs',
'@babel/plugin-proposal-class-properties'
]
}).code

module.exports = transform
Loading

0 comments on commit b781e43

Please sign in to comment.