use definitions in docs export

This commit is contained in:
Loren Norman 2025-06-24 16:31:34 -04:00
parent d7d4ba4b67
commit e9f7940c4d
6 changed files with 73 additions and 43 deletions

View file

@ -1,6 +1,7 @@
export default {
name: 'Variables',
colour: 240,
usesBlocks: [ 'io_variables_set', 'io_variables_get' ],
// called each time the category is opened
callback: workspace => {
// TODO: make a Button abstraction to wrap this up

View file

@ -1,18 +1,9 @@
import { capitalize, find, forEach, map } from 'lodash-es'
import { cleanDir, copyDir, write, totalBytesWritten } from "./export_util.js"
import DefinitionSet from '#src/definition_set.js'
import toBlockMarkdown from "#src/docs/render_block.js"
import { allBlockDefinitionsAndPaths } from './src/importer/block_importer.js'
import importToolboxJson from './src/importer/toolbox_importer.js'
import { capitalize, filter, find, forEach, keyBy, map, mapValues } from 'lodash-es'
const
toolbox = await importToolboxJson(),
categories = filter(toolbox.contents, { kind: 'category' }),
categoryBlocksMap = mapValues(keyBy(categories, "name"), cat => map(cat.contents, "type"))
const pretty = jsObject => JSON.stringify(jsObject, null, 2) + "\n"
// import DefinitionSet from '#src/definition_set.js'
/** Begin Export Script */
@ -20,62 +11,56 @@ const startTime = Date.now()
console.log("\nStarting Documentation Export")
console.log("=======================")
// const definitionSet = await DefinitionSet.load()
cleanDir("docs/blockly")
// TODO: instead, export exactly what we need
copyDir("export", "docs/blockly")
cleanDir("docs/blocks")
const definitionSet = await DefinitionSet.load()
const categories = definitionSet.getCategories()
// INIT SIDEBAR
const blockSidebar = {
text: 'Blocks',
// items: map(definitionSet.toolbox.categories, categoryName => {
items: map(categoryBlocksMap, (blocks, categoryName) => {
return {
text: categoryName,
collapsed: true,
items: []
}
})
items: map(categories, ({ name }) => ({
text: name,
collapsed: true,
items: []
}))
}
// forEach(definitionSet.blocks, ({ definitionPath, definition }) => {
forEach(allBlockDefinitionsAndPaths, ({ path, definition }) => {
forEach(definitionSet.blocks, blockDefinition => {
// skip disabled blocks
if(definition.disabled) { return }
// set a default name if missing
// TODO: should happen in definition class
if(!definition.name) {
definition.name = capitalize(definition.type.replaceAll("_", " ").replace(/^io /, ""))
}
if(blockDefinition.disabled) { return }
// mirror the blocks/**/*.js path structure
const docPath = path.replace(/.js$/, '.md')
write(`docs/blocks/${docPath}`, toBlockMarkdown(definition)) // EXPORT MARKDOWN
const docPath = blockDefinition.definitionPath.replace(/.js$/, '.md')
write(`docs/blocks/${docPath}`, toBlockMarkdown(blockDefinition)) // EXPORT MARKDOWN
// APPEND TO SIDEBAR
const
blockSidebarPath = `/blocks/${docPath.slice(0, -3)}`,
sidebarEntry = {
text: capitalize(definition.name),
text: capitalize(blockDefinition.name),
link: blockSidebarPath
}
// definition.categories
// add block links to the appropriate sidebar
forEach(categoryBlocksMap, (categoryBlocks, categoryName) => {
// add links to each sidebar category we're a part of
forEach(blockDefinition.getCategories(), category => {
// if category contains this block, add to its sidebar
if(categoryBlocks.includes(definition.type)) {
find(blockSidebar.items, { text: categoryName }).items.push(sidebarEntry)
const sidebarCategory = find(blockSidebar.items, { text: category.name })
if(!sidebarCategory) {
throw new Error(`Block category (${ category.name }) not present in sidebar!`)
}
sidebarCategory.items.push(sidebarEntry)
})
})
// WRITE SIDEBAR
write('docs/blocks/_blocks_sidebar.json', pretty(blockSidebar))
write('docs/blocks/_blocks_sidebar.json', JSON.stringify(blockSidebar, null, 2) + "\n")
const elapsed = Date.now() - startTime
console.log("=======================")

View file

@ -1,17 +1,43 @@
import { capitalize, filter } from 'lodash-es'
import { allBlockDefinitionsAndPaths, importBlockJson } from '#src/importer/block_importer.js'
class BlockDefinition {
definitionSet = null
definitionPath = null
definitionJS = null
name = null
type = null
visualization = null
connections = null
lines = null
disabled = false
categories = []
/** @returns BlockDefinition[] */
static async loadAll() {
static async loadAll(definitionSet) {
const allDefinitions = allBlockDefinitionsAndPaths.map(({ definition, path }) => {
const blockDef = new BlockDefinition()
blockDef.definitionSet = definitionSet
blockDef.definitionJS = definition
blockDef.definitionPath = path
blockDef.type = definition.type
blockDef.disabled = !!definition.disabled
blockDef.visualization = definition.visualization
blockDef.connections = definition.connections
blockDef.lines = definition.lines
blockDef.name = definition.name ||
capitalize(definition.type.replaceAll("_", " ").replace(/^io /, ""))
return blockDef
})
@ -23,6 +49,14 @@ class BlockDefinition {
return JSON.stringify(await importBlockJson(), null, 2) + "\n"
}
getCategories() {
return (this.definitionSet
? filter(this.definitionSet.getCategories(), ({ contents=[], usesBlocks=[]}) =>
contents.includes(this.type) || usesBlocks.includes(this.type)
)
: [])
}
toBlocklyJSONString = async function() {
// return JSON.stringify(await exportBlockJSON(), null, 2) + "\n"
}

View file

@ -18,7 +18,7 @@ export class DefinitionSet {
newDefinitionSet.workspaces = WorkspaceDefinition.loadAll()
newDefinitionSet.toolboxes = await ToolboxDefinition.loadAll()
newDefinitionSet.blocks = await BlockDefinition.loadAll()
newDefinitionSet.blocks = await BlockDefinition.loadAll(newDefinitionSet)
return newDefinitionSet
}
@ -28,6 +28,10 @@ export class DefinitionSet {
toBlocklyJSON: async () => await BlockDefinition.exportAll(this.blocks)
}
}
getCategories() {
return this.toolboxes[0].getCategories()
}
}
export default DefinitionSet

View file

@ -82,7 +82,7 @@ const
}
},
EXPECTED_TOOLBOX_KEYS = [ "name", "colour", "label", "contents", "callback" ],
EXPECTED_TOOLBOX_KEYS = [ "name", "colour", "label", "contents", "callback", "usesBlocks" ],
validateCategoryDefinition = definition => {
const

View file

@ -1,3 +1,5 @@
import { filter } from 'lodash-es'
import exportToolboxJSON from '#src/importer/toolbox_importer.js'
import toolboxDefinition from '#app/toolbox/index.js'
@ -17,6 +19,10 @@ class ToolboxDefinition {
toBlocklyJSONString = async function() {
return JSON.stringify(await exportToolboxJSON(), null, 2) + "\n"
}
getCategories = () => {
return filter(toolboxDefinition, item => item.contents || item.callback)
}
}
export default ToolboxDefinition