mirror of
https://github.com/facebook/docusaurus.git
synced 2025-05-10 07:37:19 +02:00
Use information from "docSidebar.js" file instead of doc front matters for sidebars.
This commit is contained in:
parent
69ba05e4f1
commit
abc5e325cd
9 changed files with 190 additions and 35 deletions
10
examples/docSidebar.js
Normal file
10
examples/docSidebar.js
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
module.exports = {
|
||||||
|
docs: {
|
||||||
|
Docusaurus: ["doc1"],
|
||||||
|
"First Category": ["doc2"],
|
||||||
|
"Second Category": ["doc3"]
|
||||||
|
},
|
||||||
|
"docs-other": {
|
||||||
|
"First Category": ["doc4", "doc5"]
|
||||||
|
}
|
||||||
|
};
|
|
@ -15,8 +15,8 @@ const siteConfig = require(process.cwd() + "/siteConfig.js");
|
||||||
|
|
||||||
class DocsSidebar extends React.Component {
|
class DocsSidebar extends React.Component {
|
||||||
render() {
|
render() {
|
||||||
let layout = this.props.metadata.layout;
|
let sidebar = this.props.metadata.sidebar;
|
||||||
let docsCategories = require("./" + layout + "Categories.js");
|
let docsCategories = require("./" + sidebar + "Categories.js");
|
||||||
return (
|
return (
|
||||||
<Container className="docsNavContainer" id="docsNav" wrapper={false}>
|
<Container className="docsNavContainer" id="docsNav" wrapper={false}>
|
||||||
<SideNav
|
<SideNav
|
||||||
|
|
96
lib/gen-doc-sidebar.js
Normal file
96
lib/gen-doc-sidebar.js
Normal file
|
@ -0,0 +1,96 @@
|
||||||
|
#!/usr/bin/env node
|
||||||
|
|
||||||
|
/* script to generate docSidebar.js file for users who are using
|
||||||
|
doc front mattters with category, layout, previous, and next fields */
|
||||||
|
const glob = require("glob");
|
||||||
|
const fs = require("fs");
|
||||||
|
const path = require("path");
|
||||||
|
const prettier = require("prettier");
|
||||||
|
|
||||||
|
function splitHeader(content) {
|
||||||
|
const lines = content.split("\n");
|
||||||
|
let i = 1;
|
||||||
|
for (; i < lines.length - 1; ++i) {
|
||||||
|
if (lines[i] === "---") {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {
|
||||||
|
header: lines.slice(1, i + 1).join("\n"),
|
||||||
|
content: lines.slice(i + 1).join("\n")
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Extract markdown metadata header
|
||||||
|
function extractMetadata(content) {
|
||||||
|
const metadata = {};
|
||||||
|
const both = splitHeader(content);
|
||||||
|
const lines = both.header.split("\n");
|
||||||
|
for (let i = 0; i < lines.length - 1; ++i) {
|
||||||
|
const keyvalue = lines[i].split(":");
|
||||||
|
const key = keyvalue[0].trim();
|
||||||
|
let value = keyvalue.slice(1).join(":").trim();
|
||||||
|
// Handle the case where you have "Community #10"
|
||||||
|
try {
|
||||||
|
value = JSON.parse(value);
|
||||||
|
} catch (e) {}
|
||||||
|
metadata[key] = value;
|
||||||
|
}
|
||||||
|
return { metadata, rawContent: both.content };
|
||||||
|
}
|
||||||
|
|
||||||
|
const docs = {};
|
||||||
|
const sidebar = {};
|
||||||
|
|
||||||
|
const files = glob.sync(process.cwd() + "/../docs/**");
|
||||||
|
files.forEach(file => {
|
||||||
|
const extension = path.extname(file);
|
||||||
|
if (extension === ".md" || extension === ".markdown") {
|
||||||
|
const metadata = extractMetadata(fs.readFileSync(file, "utf8")).metadata;
|
||||||
|
if (!metadata.id) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const data = {};
|
||||||
|
data["category"] = metadata.category;
|
||||||
|
data["sidebar"] = metadata.layout;
|
||||||
|
if (metadata.next) {
|
||||||
|
data["next"] = metadata.next;
|
||||||
|
}
|
||||||
|
if (metadata.previous) {
|
||||||
|
data["previous"] = metadata.previous;
|
||||||
|
}
|
||||||
|
docs[metadata.id] = data;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Object.keys(docs).forEach(id => {
|
||||||
|
if (!docs[id].previous) {
|
||||||
|
sidebar[docs[id].sidebar] = {};
|
||||||
|
sidebar[docs[id].sidebar][docs[id].category] = [];
|
||||||
|
sidebar[docs[id].sidebar][docs[id].category].push(id);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Object.keys(sidebar).forEach(sb => {
|
||||||
|
const categories = sidebar[sb];
|
||||||
|
Object.keys(categories).forEach(category => {
|
||||||
|
const docIds = categories[category];
|
||||||
|
next = docIds[0];
|
||||||
|
while (next) {
|
||||||
|
id = next;
|
||||||
|
next = docs[id].next;
|
||||||
|
if (!next) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (docs[next].category === category) {
|
||||||
|
docIds.push(next);
|
||||||
|
} else {
|
||||||
|
categories[docs[next].category] = [];
|
||||||
|
categories[docs[next].category].push(next);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const str = prettier.format("module.exports = " + JSON.stringify(sidebar));
|
||||||
|
fs.writeFileSync(process.cwd() + "/docSidebar.js", str, "utf8");
|
|
@ -87,12 +87,12 @@ function execute() {
|
||||||
}
|
}
|
||||||
|
|
||||||
const readCategories = require("./readCategories.js");
|
const readCategories = require("./readCategories.js");
|
||||||
let layouts = {};
|
let sidebars = {};
|
||||||
for (let i = 0; i < Metadata.length; i++) {
|
for (let i = 0; i < Metadata.length; i++) {
|
||||||
let layout = Metadata[i].layout;
|
let sidebar = Metadata[i].sidebar;
|
||||||
if (layouts[layout] !== true) {
|
if (sidebars[sidebar] !== true) {
|
||||||
layouts[layout] = true;
|
sidebars[sidebar] = true;
|
||||||
readCategories(layout);
|
readCategories(sidebar);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -23,7 +23,7 @@ if (fs.existsSync(CWD + "/languages.js")) {
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
function readCategories(layout) {
|
function readCategories(sidebar) {
|
||||||
const enabledLanguages = [];
|
const enabledLanguages = [];
|
||||||
languages.filter(lang => lang.enabled).map(lang => {
|
languages.filter(lang => lang.enabled).map(lang => {
|
||||||
enabledLanguages.push(lang.tag);
|
enabledLanguages.push(lang.tag);
|
||||||
|
@ -35,7 +35,7 @@ function readCategories(layout) {
|
||||||
const language = enabledLanguages[k];
|
const language = enabledLanguages[k];
|
||||||
|
|
||||||
const metadatas = Metadata.filter(metadata => {
|
const metadatas = Metadata.filter(metadata => {
|
||||||
return metadata.layout === layout && metadata.language === language;
|
return metadata.sidebar === sidebar && metadata.language === language;
|
||||||
});
|
});
|
||||||
|
|
||||||
// Build a hashmap of article_id -> metadata
|
// Build a hashmap of article_id -> metadata
|
||||||
|
@ -91,7 +91,7 @@ function readCategories(layout) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fs.writeFileSync(
|
fs.writeFileSync(
|
||||||
__dirname + "/../core/" + layout + "Categories.js",
|
__dirname + "/../core/" + sidebar + "Categories.js",
|
||||||
"/**\n" +
|
"/**\n" +
|
||||||
" * @generated\n" +
|
" * @generated\n" +
|
||||||
" */\n" +
|
" */\n" +
|
||||||
|
|
|
@ -12,6 +12,8 @@ const CWD = process.cwd();
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
const fs = require("fs");
|
const fs = require("fs");
|
||||||
const os = require("os");
|
const os = require("os");
|
||||||
|
const sidebar = require(CWD + "/docSidebar.js");
|
||||||
|
const glob = require("glob");
|
||||||
let languages;
|
let languages;
|
||||||
if (fs.existsSync(CWD + "/languages.js")) {
|
if (fs.existsSync(CWD + "/languages.js")) {
|
||||||
languages = require(CWD + "/languages.js");
|
languages = require(CWD + "/languages.js");
|
||||||
|
@ -24,7 +26,37 @@ if (fs.existsSync(CWD + "/languages.js")) {
|
||||||
}
|
}
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
const glob = require("glob");
|
|
||||||
|
function readSidebar(allSidebars) {
|
||||||
|
const order = {};
|
||||||
|
|
||||||
|
Object.keys(allSidebars).forEach(sidebar => {
|
||||||
|
const categories = allSidebars[sidebar];
|
||||||
|
|
||||||
|
let ids = [];
|
||||||
|
let categoryOrder = [];
|
||||||
|
Object.keys(categories).forEach(category => {
|
||||||
|
ids = ids.concat(categories[category]);
|
||||||
|
for (let i = 0; i < categories[category].length; i++) {
|
||||||
|
categoryOrder.push(category);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
for (let i = 0; i < ids.length; i++) {
|
||||||
|
const id = ids[i];
|
||||||
|
let previous, next;
|
||||||
|
if (i > 0) previous = ids[i - 1];
|
||||||
|
if (i < ids.length - 1) next = ids[i + 1];
|
||||||
|
order[id] = {
|
||||||
|
previous: previous,
|
||||||
|
next: next,
|
||||||
|
sidebar: sidebar,
|
||||||
|
category: categoryOrder[i]
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return order;
|
||||||
|
}
|
||||||
|
|
||||||
function splitHeader(content) {
|
function splitHeader(content) {
|
||||||
const lines = content.split(os.EOL);
|
const lines = content.split(os.EOL);
|
||||||
|
@ -84,20 +116,29 @@ function processMetadata(file) {
|
||||||
// change ids previous, next
|
// change ids previous, next
|
||||||
metadata.localized_id = metadata.id;
|
metadata.localized_id = metadata.id;
|
||||||
metadata.id = language + "-" + metadata.id;
|
metadata.id = language + "-" + metadata.id;
|
||||||
if (metadata.previous) {
|
|
||||||
metadata.previous_id = metadata.previous;
|
|
||||||
metadata.previous = language + "-" + metadata.previous;
|
|
||||||
}
|
|
||||||
if (metadata.next) {
|
|
||||||
metadata.next_id = metadata.next;
|
|
||||||
metadata.next = language + "-" + metadata.next;
|
|
||||||
}
|
|
||||||
metadata.language = language;
|
metadata.language = language;
|
||||||
|
|
||||||
|
const order = readSidebar(sidebar);
|
||||||
|
const id = metadata.localized_id;
|
||||||
|
|
||||||
|
metadata.sidebar = order[id].sidebar;
|
||||||
|
metadata.category = order[id].category;
|
||||||
|
|
||||||
|
if (order[id].next) {
|
||||||
|
metadata.next_id = order[id].next;
|
||||||
|
metadata.next = language + "-" + order[id].next;
|
||||||
|
}
|
||||||
|
if (order[id].previous) {
|
||||||
|
metadata.previous_id = order[id].previous;
|
||||||
|
metadata.previous = language + "-" + order[id].previous;
|
||||||
|
}
|
||||||
|
|
||||||
return { metadata, rawContent: rawContent };
|
return { metadata, rawContent: rawContent };
|
||||||
}
|
}
|
||||||
|
|
||||||
function generateDocsMetadata() {
|
function generateDocsMetadata() {
|
||||||
|
const order = readSidebar(sidebar);
|
||||||
|
|
||||||
const regexSubFolder = /docs\/(.*)\/.*/;
|
const regexSubFolder = /docs\/(.*)\/.*/;
|
||||||
|
|
||||||
const enabledLanguages = [];
|
const enabledLanguages = [];
|
||||||
|
@ -126,7 +167,7 @@ function generateDocsMetadata() {
|
||||||
if (!res) {
|
if (!res) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const metadata = res.metadata;
|
let metadata = res.metadata;
|
||||||
metadatas.push(metadata);
|
metadatas.push(metadata);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -181,7 +222,9 @@ function generateBlogMetadata() {
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports.extractMetadata = extractMetadata;
|
module.exports = {
|
||||||
module.exports.processMetadata = processMetadata;
|
extractMetadata,
|
||||||
module.exports.generateDocsMetadata = generateDocsMetadata;
|
processMetadata,
|
||||||
module.exports.generateBlogMetadata = generateBlogMetadata;
|
generateDocsMetadata,
|
||||||
|
generateBlogMetadata
|
||||||
|
};
|
||||||
|
|
|
@ -17,7 +17,6 @@ function execute(port) {
|
||||||
const fs = require("fs-extra");
|
const fs = require("fs-extra");
|
||||||
const os = require("os");
|
const os = require("os");
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
const readMetadata = require("./readMetadata.js");
|
|
||||||
const toSlug = require("../core/toSlug.js");
|
const toSlug = require("../core/toSlug.js");
|
||||||
const mkdirp = require("mkdirp");
|
const mkdirp = require("mkdirp");
|
||||||
const glob = require("glob");
|
const glob = require("glob");
|
||||||
|
@ -70,22 +69,25 @@ function execute(port) {
|
||||||
|
|
||||||
/****************************************************************************/
|
/****************************************************************************/
|
||||||
|
|
||||||
|
let readMetadata;
|
||||||
let Metadata;
|
let Metadata;
|
||||||
let readCategories;
|
let readCategories;
|
||||||
|
|
||||||
function reloadMetadataCategories() {
|
function reloadMetadataCategories() {
|
||||||
|
purgeCache("./readMetadata.js");
|
||||||
|
readMetadata = require("./readMetadata.js");
|
||||||
readMetadata.generateDocsMetadata();
|
readMetadata.generateDocsMetadata();
|
||||||
purgeCache("../core/metadata.js");
|
purgeCache("../core/metadata.js");
|
||||||
Metadata = require("../core/metadata.js");
|
Metadata = require("../core/metadata.js");
|
||||||
purgeCache("./readCategories.js");
|
purgeCache("./readCategories.js");
|
||||||
readCategories = require("./readCategories.js");
|
readCategories = require("./readCategories.js");
|
||||||
|
|
||||||
let layouts = {};
|
let sidebars = {};
|
||||||
for (let i = 0; i < Metadata.length; i++) {
|
for (let i = 0; i < Metadata.length; i++) {
|
||||||
let layout = Metadata[i].layout;
|
let sidebar = Metadata[i].sidebar;
|
||||||
if (layouts[layout] !== true) {
|
if (sidebars[sidebar] !== true) {
|
||||||
layouts[layout] = true;
|
sidebars[sidebar] = true;
|
||||||
readCategories(layout);
|
readCategories(sidebar);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -41,11 +41,13 @@ function execute() {
|
||||||
files.forEach(file => {
|
files.forEach(file => {
|
||||||
const extension = path.extname(file);
|
const extension = path.extname(file);
|
||||||
if (extension === ".md" || extension === ".markdown") {
|
if (extension === ".md" || extension === ".markdown") {
|
||||||
const metadata = readMetadata.extractMetadata(
|
const res = readMetadata.processMetadata(file);
|
||||||
fs.readFileSync(file, "utf8")
|
if (!res) {
|
||||||
).metadata;
|
return
|
||||||
|
}
|
||||||
|
const metadata = res.metadata;
|
||||||
|
|
||||||
translations["localized-strings"][metadata.id] = metadata.title;
|
translations["localized-strings"][metadata.localized_id] = metadata.title;
|
||||||
translations["localized-strings"][metadata.category] = metadata.category;
|
translations["localized-strings"][metadata.category] = metadata.category;
|
||||||
|
|
||||||
if (metadata.sidebar_title) {
|
if (metadata.sidebar_title) {
|
||||||
|
|
|
@ -15,6 +15,7 @@
|
||||||
"express": "^4.15.3",
|
"express": "^4.15.3",
|
||||||
"fs-extra": "^3.0.1",
|
"fs-extra": "^3.0.1",
|
||||||
"glob": "^7.1.2",
|
"glob": "^7.1.2",
|
||||||
|
"prettier": "^1.5.3",
|
||||||
"react": "^15.5.4",
|
"react": "^15.5.4",
|
||||||
"react-dom": "^15.5.4",
|
"react-dom": "^15.5.4",
|
||||||
"request": "^2.81.0",
|
"request": "^2.81.0",
|
||||||
|
@ -27,6 +28,7 @@
|
||||||
"docusaurus-build": "./lib/build-files.js",
|
"docusaurus-build": "./lib/build-files.js",
|
||||||
"docusaurus-publish": "./lib/publish-gh-pages.js",
|
"docusaurus-publish": "./lib/publish-gh-pages.js",
|
||||||
"docusaurus-examples": "./lib/copy-examples.js",
|
"docusaurus-examples": "./lib/copy-examples.js",
|
||||||
"docusaurus-write-translations": "./lib/write-translations.js"
|
"docusaurus-write-translations": "./lib/write-translations.js",
|
||||||
|
"docusaurus-gen-doc-sidebar": "./lib/gen-doc-sidebar.js"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue