veyron-browser: Preliminary Help Content
I am adding some preliminary topics for help content.
We need to plan how to organize this help page.
- Basic help page with a tabbed layout
- Modification of the help page to properly style parsed markdown
- Modification of our custom browserify transform system
- Filler content for each help page
Change-Id: I52a95b48651885d5bb578d27b4f1754a0a77a138
diff --git a/Makefile b/Makefile
index 068f68f..a57e053 100644
--- a/Makefile
+++ b/Makefile
@@ -22,7 +22,7 @@
# All JS and CSS files except build.js and third party.
BROWSERIFY_FILES = $(shell find src -name "*.js" -o -name "*.css")
-BROWSERIFY_OPTIONS = --transform ./css-transform --debug
+BROWSERIFY_OPTIONS = --transform ./main-transform --debug
# All Go and VDL files.
GO_FILES = $(shell find go -name "*.go")
diff --git a/css-transform.js b/css-transform.js
index f94c0e2..ae2ad44 100644
--- a/css-transform.js
+++ b/css-transform.js
@@ -4,23 +4,27 @@
var reworkVars = require('rework-vars');
var reworkImport = require('rework-import');
-module.exports = transform;
+module.exports = {
+ canTransform: isCss,
+ transform: transform
+};
+/*
+ * Transform the given css file by compiling it with rework.
+ */
function transform(file) {
- if (path.extname(file) !== '.css') {
- return through2();
- }
-
var contents = [];
return through2(write, flush);
+ // Simply collect string fragments of the css file.
function write(data, encoding, callback) {
var string = data.toString();
contents.push(string);
callback();
}
+ // Reconstruct the css and then compile it.
function flush(callback) {
var string = contents.join('');
var css = compile(string);
@@ -29,7 +33,7 @@
}
}
-/* Compiles the given CSS string using rework */
+/* Compiles the given CSS string using rework. */
function compile(string) {
var css = rework(string)
.use(reworkImport({
@@ -40,4 +44,9 @@
compress: true
});
return css;
+}
+
+/* Determines if the filetype is css. */
+function isCss(file) {
+ return path.extname(file) === '.css';
}
\ No newline at end of file
diff --git a/main-transform.js b/main-transform.js
new file mode 100644
index 0000000..8e03ea1
--- /dev/null
+++ b/main-transform.js
@@ -0,0 +1,27 @@
+var through2 = require('through2');
+
+module.exports = transform;
+
+/*
+ * A list of specialist transformers targetting specific filetypes.
+ */
+var transformers = Object.freeze([
+ require('./css-transform'),
+ require('./md-transform')
+]);
+
+/*
+ * Apply a transform to any given file. Most files are simply passed through,
+ * but a matching specialist transformer applies its transform function instead.
+ */
+function transform(file) {
+ // Attempt to find a specialist transformer to apply their transformation.
+ for (var i = 0; i < transformers.length; i++) {
+ if (transformers[i].canTransform(file)) {
+ return transformers[i].transform(file);
+ }
+ }
+
+ // If there's no result, pass the file through normally.
+ return through2();
+}
\ No newline at end of file
diff --git a/md-transform.js b/md-transform.js
new file mode 100644
index 0000000..0dba111
--- /dev/null
+++ b/md-transform.js
@@ -0,0 +1,36 @@
+var parseMarkdown = require('marked');
+var path = require('path');
+var through2 = require('through2');
+
+module.exports = {
+ canTransform: isMarkdown,
+ transform: transform
+};
+
+/*
+ * Transform the given markdown file by parsing the markdown file.
+ */
+function transform(file) {
+ var contents = [];
+
+ return through2(write, flush);
+
+ // Simply collect string fragments of the markdown file.
+ function write(data, encoding, callback) {
+ var string = data.toString();
+ contents.push(string);
+ callback();
+ }
+
+ // Reconstruct the markdown and then parse it.
+ function flush(callback) {
+ var string = contents.join('');
+ this.push('module.exports = ' + JSON.stringify(parseMarkdown(string)));
+ callback();
+ }
+}
+
+/* Determines if the filetype is markdown. */
+function isMarkdown(file) {
+ return path.extname(file) === '.md';
+}
\ No newline at end of file
diff --git a/package.json b/package.json
index 5174990..5b0691e 100644
--- a/package.json
+++ b/package.json
@@ -26,6 +26,7 @@
"insert-css": "^0.2.0",
"json-stable-stringify": "^1.0.0",
"localforage": "^1.0.2",
+ "marked": "^0.3.2",
"mercury": "^6.0.1",
"routes": "^1.2.0",
"vis": "^3.1.0",
diff --git a/src/app.js b/src/app.js
index b06e77a..811beab 100644
--- a/src/app.js
+++ b/src/app.js
@@ -1,19 +1,21 @@
var guid = require('guid');
var mercury = require('mercury');
var onDocumentReady = require('./lib/document-ready');
-var viewport = require('./components/viewport/index');
var router = require('./router');
var debug = require('./components/debug/index');
var browse = require('./components/browse/index');
var error = require('./components/error/index');
+var help = require('./components/help/index');
+var viewport = require('./components/viewport/index');
var errorRoute = require('./routes/error');
onDocumentReady(function startApp() {
var browseComponent = browse();
- var viewportComponent = viewport();
var errorComponent = error();
var debugComponent = debug();
+ var helpComponent = help();
+ var viewportComponent = viewport();
// Top level state
var state = mercury.struct({
@@ -28,12 +30,18 @@
*/
pageKey: mercury.value('')
}),
+
/*
- * Veyron Namespace Browsing related states
+ * Veyron Namespace Browsing related state
*/
browse: browseComponent.state,
/*
+ * Veyron Namespace Help related state
+ */
+ help: helpComponent.state,
+
+ /*
* State of the viewport component
*/
viewport: viewportComponent.state,
@@ -62,6 +70,11 @@
'browse',
/*
+ * Veyron Namespace Help related events
+ */
+ 'help',
+
+ /*
* Events of the viewport component
*/
'viewport'
@@ -78,6 +91,7 @@
'navigate'
]);
events.browse = browseComponent.events;
+ events.help = helpComponent.events;
events.viewport = viewportComponent.events;
// Wire Events
@@ -96,8 +110,14 @@
mercury.app(document.body, state, render);
function wireEvents() {
+ // TODO(aghassemi): Make these events global.
+ // Hook up external browse events.
events.browse.error(onError);
events.browse.toast(onToast);
+
+ // Hook up external help events.
+ events.help.navigate = events.navigation.navigate;
+ events.help.error(onError);
}
/*
diff --git a/src/components/browse/index.css b/src/components/browse/index.css
index 6e9b2c1..51bc650 100644
--- a/src/components/browse/index.css
+++ b/src/components/browse/index.css
@@ -25,6 +25,9 @@
background-color: var(--color-grey-light);
box-shadow: none;
}
+.browse-details-sidebar::shadow #dropShadow {
+ display: none;
+}
.browse-details-sidebar::shadow #mainContainer {
background-color: var(--color-grey-light);
border-left: solid 1px var(--color-divider);
diff --git a/src/components/browse/item-details/index.css b/src/components/browse/item-details/index.css
index e34b865..0f74f2e 100644
--- a/src/components/browse/item-details/index.css
+++ b/src/components/browse/item-details/index.css
@@ -1,39 +1,10 @@
@import "common-style/sizes.css";
@import "common-style/theme.css";
-
-.tabs {
- box-shadow: var(--shadow-bottom);
-}
-
-.tabs::shadow #selectionBar {
- background-color: var(--color-bright);
-}
-
-.tab {
- background-color: var(--color-main);
- color: var(--color-text-primary-invert);
-}
-
-.tab::shadow #ink {
- color: var(--color-bright);
-}
-
-.browse-details-sidebar::shadow #dropShadow {
- display: none;
-}
-
-.tab-content {
- padding: 0.5em;
-}
-
-:not(.core-selected).tab-content {
- display: none;
-}
+@import "common-style/tabs.css";
.field {
font-size: 0.9em;
margin-bottom: 0.75em;
-
}
.field h4 {
diff --git a/src/components/common-style/tabs.css b/src/components/common-style/tabs.css
new file mode 100644
index 0000000..931328f
--- /dev/null
+++ b/src/components/common-style/tabs.css
@@ -0,0 +1,26 @@
+@import "./theme.css";
+
+.tabs {
+ box-shadow: var(--shadow-bottom);
+}
+
+.tabs::shadow #selectionBar {
+ background-color: var(--color-bright);
+}
+
+.tab {
+ background-color: var(--color-main);
+ color: var(--color-text-primary-invert);
+}
+
+.tab::shadow #ink {
+ color: var(--color-bright);
+}
+
+.tab-content {
+ padding: 0.5em;
+}
+
+:not(.core-selected).tab-content {
+ display: none;
+}
\ No newline at end of file
diff --git a/src/components/help/constants.js b/src/components/help/constants.js
new file mode 100644
index 0000000..1c8b4a7
--- /dev/null
+++ b/src/components/help/constants.js
@@ -0,0 +1,47 @@
+var helpRoute = require('../../routes/help');
+
+var tabKeys = Object.freeze({
+ MAIN: 'main', // Describes the Veyron Browser to new users.
+ BROWSE: 'browse', // Introduces how to browse the namespace.
+ DETAILS: 'details', // Defines service information and icons.
+ METHODS: 'methods', // Explains how to make RPCs.
+ FAQ: 'faq' // Frequently asked questions and contact information.
+});
+
+var sections = Object.freeze(new Map([
+ [tabKeys.MAIN, {
+ index: 0,
+ header: 'Main Help',
+ markdownContent: require('./content/main.md'),
+ path: helpRoute.createUrl(tabKeys.MAIN)
+ }],
+ [tabKeys.BROWSE, {
+ index: 1,
+ header: 'How to Browse',
+ markdownContent: require('./content/browse.md'),
+ path: helpRoute.createUrl(tabKeys.BROWSE)
+ }],
+ [tabKeys.DETAILS, {
+ index: 2,
+ header: 'What You See',
+ markdownContent: require('./content/details.md'),
+ path: helpRoute.createUrl(tabKeys.DETAILS)
+ }],
+ [tabKeys.METHODS, {
+ index: 3,
+ header: 'Talk to Services',
+ markdownContent: require('./content/methods.md'),
+ path: helpRoute.createUrl(tabKeys.METHODS)
+ }],
+ [tabKeys.FAQ, {
+ index: 4,
+ header: 'FAQ',
+ markdownContent: require('./content/faq.md'),
+ path: helpRoute.createUrl(tabKeys.FAQ)
+ }]
+]));
+
+module.exports = {
+ tabKeys: tabKeys,
+ sections: sections
+};
\ No newline at end of file
diff --git a/src/components/help/content/browse.md b/src/components/help/content/browse.md
new file mode 100644
index 0000000..73f62a4
--- /dev/null
+++ b/src/components/help/content/browse.md
@@ -0,0 +1,14 @@
+How to Browse the Namespace
+===========================
+
+This tab describes the many ways in which the user can browse the namespace. A
+few examples should be included.
+
+Topics
+------
+* namespace items
+* namespace root search box
+* glob search box and how glob works
+* breadcrumbs
+* visualization page
+* how to setup shortcuts
\ No newline at end of file
diff --git a/src/components/help/content/details.md b/src/components/help/content/details.md
new file mode 100644
index 0000000..815e84d
--- /dev/null
+++ b/src/components/help/content/details.md
@@ -0,0 +1,12 @@
+How to Interpret What You See
+=============================
+
+This tab describes what the user will see when looking at a service item and why
+it is important.
+
+Topics
+------
+* its name
+* service types and icons
+* service methods (and output)
+* (potentially more information will be shown later)
\ No newline at end of file
diff --git a/src/components/help/content/faq.md b/src/components/help/content/faq.md
new file mode 100644
index 0000000..6f58912
--- /dev/null
+++ b/src/components/help/content/faq.md
@@ -0,0 +1,25 @@
+Frequently Asked Questions
+==========================
+
+TODO(alexfandrianto): The anchortext links don't work. We should think of an
+alternate FAQ setup or properly handle these links.
+
+1. [I can't do X Y or Z!](#/help/faq/1)
+2. [How can I contribute to the Namespace Browser?](#/help/faq/2)
+3. [How can I learn more about Veyron?](#/help/faq/3)
+
+I can't do X Y or Z!<a name="1"></a>
+--------------------
+
+That's a shame.
+
+How can I contribute to the Namespace Browser?<a name="2"></a>
+----------------------------------------------
+
+Visit us on github to make suggestions, report bugs, or contribute pull
+requests. INSERT A LINK
+
+How can I learn more about Vanadium?<a name="3"></a>
+------------------------------------
+
+Go to the Vanadium website. INSERT A LINK.
\ No newline at end of file
diff --git a/src/components/help/content/main.md b/src/components/help/content/main.md
new file mode 100644
index 0000000..a3f579d
--- /dev/null
+++ b/src/components/help/content/main.md
@@ -0,0 +1,35 @@
+Main Help Page
+==============
+
+Introduction
+------------
+
+The Namespace Browser is a tool to browse and interact with running services.
+
+How to Browse
+-------------
+
+(BRIEF DESCRIPTION ABOUT WHAT EXPLORING THE NAMESPACE IS ABOUT.)
+
+See the [help topic](#/help/browse) for more details.
+
+What You See
+------------
+
+(BRIEF DESCRIPTION ABOUT INFORMATION YOU MIGHT SEE WHILE USING THIS TOOL.)
+
+See the [help topic](#/help/details) for more details.
+
+Talk to Services
+----------------
+
+(BRIEF DESCRIPTION ABOUT WHAT INTERACTING WITH SERVICES INVOLVES.)
+
+See the [help topic](#/help/methods) for more details.
+
+Conclusion
+----------
+
+(SAY SOMETHING REASSURING.)
+
+See the [help topic](#/help/faq) to see a listing of frequently asked questions.
\ No newline at end of file
diff --git a/src/components/help/content/methods.md b/src/components/help/content/methods.md
new file mode 100644
index 0000000..665e638
--- /dev/null
+++ b/src/components/help/content/methods.md
@@ -0,0 +1,11 @@
+How to Talk to Services
+=======================
+
+This tab describes how the user can make method calls to a service item.
+
+Topics
+------
+* how to run simple RPCs
+* how to fill out an RPC form (including syntax)
+* where the output goes
+* how to save invocations
\ No newline at end of file
diff --git a/src/components/help/index.css b/src/components/help/index.css
new file mode 100644
index 0000000..27af845
--- /dev/null
+++ b/src/components/help/index.css
@@ -0,0 +1,32 @@
+@import "common-style/theme.css";
+@import "common-style/tabs.css";
+
+.markdown {
+ padding: 0.5em;
+}
+
+.markdown h1 {
+ text-align: center;
+}
+
+.markdown h2 {
+ margin-top: 0.75em;
+ margin-bottom: 0.25em;
+}
+
+/* Recover some sane version of link color. */
+.markdown a {
+ color: -webkit-link;
+ text-decoration: underline;
+}
+
+/* Recover some sane version of list bullets. */
+.markdown ul, .markdown ol {
+ padding-left: 2em;
+}
+.markdown ul li {
+ list-style: disc outside none;
+}
+.markdown ol li {
+ list-style: decimal outside none;
+}
\ No newline at end of file
diff --git a/src/components/help/index.js b/src/components/help/index.js
index b60b2f0..5e02b3f 100644
--- a/src/components/help/index.js
+++ b/src/components/help/index.js
@@ -1,5 +1,12 @@
var mercury = require('mercury');
var h = mercury.h;
+var AttributeHook = require('../../lib/mercury/attribute-hook');
+var insertCss = require('insert-css');
+var css = require('./index.css');
+var constants = require('./constants.js');
+
+var tabKeys = constants.tabKeys;
+var sections = constants.sections;
module.exports = create;
module.exports.render = render;
@@ -7,9 +14,49 @@
/*
* Help view
*/
-function create() {}
+function create() {
+ var state = mercury.varhash({
+ selectedTab: mercury.value(tabKeys.MAIN)
+ });
+ var events = mercury.input([
+ 'error', // Will be wired up by the application.
+ 'navigate' // Will be wired up by the application.
+ ]);
-function render() {
- // TODO(aghassemi)
- return h('div.empty', 'Help will come one day.');
+ return {
+ state: state,
+ events: events
+ };
+}
+
+/*
+ * Draws the help page, which consists of a tabbed layout and the selected tab's
+ * help content. Content comes from a parsed markdown file.
+ */
+function render(state, events) {
+ insertCss(css);
+
+ // Render each help tab, as defined by the sections.
+ var tabs = [];
+ sections.forEach(function(section, key) {
+ var tab = h('paper-tab.tab', {
+ 'tabKey': key,
+ 'ev-click': mercury.event(events.navigate, {
+ 'path': section.path
+ })
+ }, section.header);
+ tabs.push(tab);
+ });
+
+ // Show the tabs followed by the content of the selected help tab.
+ return [
+ h('paper-tabs.tabs', {
+ 'selectedProperty': new AttributeHook('tabKey'),
+ 'selected': new AttributeHook(sections.get(state.selectedTab).index),
+ 'noink': new AttributeHook(true)
+ }, tabs),
+ h('.tab-content.core-selected', h('.markdown', {
+ 'innerHTML': sections.get(state.selectedTab).markdownContent
+ }))
+ ];
}
\ No newline at end of file
diff --git a/src/components/help/selectTab.js b/src/components/help/selectTab.js
new file mode 100644
index 0000000..f992b7b
--- /dev/null
+++ b/src/components/help/selectTab.js
@@ -0,0 +1,19 @@
+var sections = require('./constants').sections;
+
+module.exports = selectTab;
+
+/*
+ * Exported function that sets the given state to the given tabKey.
+ * If there is an error, however, the error event is run.
+ */
+function selectTab(state, events, tabKey) {
+ // If the tab is invalid, go to the error page.
+ if (sections.get(tabKey) === undefined) {
+ // TODO(aghassemi): Add 404 error.
+ // events.error(type.404);
+ events.error(new Error('Invalid help page: ' + tabKey));
+ } else {
+ // Since the tabKey is valid, the selectedTab can be updated.
+ state.selectedTab.set(tabKey);
+ }
+}
\ No newline at end of file
diff --git a/src/components/main-content/index.js b/src/components/main-content/index.js
index 7fa8521..ebaa676 100644
--- a/src/components/main-content/index.js
+++ b/src/components/main-content/index.js
@@ -44,7 +44,7 @@
case 'browse':
return Browse.render(state.browse, events.browse, events.navigation);
case 'help':
- return Help.render();
+ return Help.render(state.help, events.help);
case 'error':
return ErrorPage.render(state.error);
case 'visualize':
diff --git a/src/routes/help.js b/src/routes/help.js
index ca2add0..7ddad1d 100644
--- a/src/routes/help.js
+++ b/src/routes/help.js
@@ -1,14 +1,26 @@
+var exists = require('../lib/exists');
+
module.exports = function(routes) {
- routes.addRoute('/help', handleHelpRoute);
+ routes.addRoute('/help/:topic?', handleHelpRoute);
};
-module.exports.createUrl = function() {
+module.exports.createUrl = function(topic) {
+ if (exists(topic)) {
+ return '#/help/' + topic;
+ }
return '#/help';
};
-function handleHelpRoute(state) {
+function handleHelpRoute(state, events, params) {
// Set the page to help
state.navigation.pageKey.set('help');
state.viewport.title.set('Help');
+
+ // If given, go to the specified help page tab.
+ if (params.topic) {
+ // Import selectTab here to avoid a cyclical dependency.
+ var selectTab = require('../components/help/selectTab');
+ selectTab(state.help, events.help, params.topic);
+ }
}
\ No newline at end of file