From a4e3af191c31740f844568d78c815821e494019a Mon Sep 17 00:00:00 2001
From: Nathan Letwory A Visual Studio Code extension for writing literate programs. A simple implementation to provide code completion will help authors writing
+their literate programs. Having possible tag names suggested will help
+decreasing the cognitive load of remembering all code fragment names in a
+literate project. This project itself has well over 50 fragments, and having to
+remember them by name is not easy. Until there is a good literate file type integration with Visual Studio Code
+we'll be relying on the built-in Markdown functionality. The completion item provider will generate a After setting up the necessary variables with
+ After the workspace folder has been determined we can gather all fragments in
+our project. Finally we generate the completion items into the array Completion items are going to be collected in an Determining the workspace folder for the given TextDocument is done by creating
+relative paths from each workspace folder to the document. If the path does not
+start with If no workspace folders were found, or if the TextDocument did not have a
+workspace folder we essentially end up returning an empty array from the
+completion item provider. Code completion item providers run essentially on document changes. The
+ With all fragments in the map we iterate over all the keys. For each key we
+fetch the corresponding Further the fragment code is set to be the detail of the completion item. This
+will provide a tooltip with the code fragment readable, so that it is easy to
+understand what fragment is currently highlighted in the completion list. Finally the set the completion item kind to The Literate Fragment Explorer is a The Literate Fragment Explorer needs a
+ The class The API for The current implementation simply fires the With those two in place the The As said, the On the other hand the When the workspace has no workspace folders at all there will be no children to
+return, as there are no literate documents to begin with. If we do have workspace folders, but no element is given to look for children we
+need to look at the all the fragments available in all documents across all
+workspace folders. If on the other hand an element is given then its children
+are retrieved. When no element is passed we want the root of all the branches, where each
+workspace folder is the root of its own branch. To this end the children are all essentially the workspace folder names. Since
+these are the work folders the fragments representing them have no Getting the children for a given element is a bit more involved. First we set
+up a constant From the element we already learned the workspace folder for its project, so we
+can use that directly to parse the literate content. With the There are essentially two cases we need to check for. If the given element has
+no To find the fragment information to build Still to do. Right now essentially the map structure is shown, but that isn't
+very useful. What we really need is a hierarchical form with each fragment under
+its parent fragment so that the structure of the literate program can be seen. Another improvement we could make is to show Markdown outline of chapters, with
+fragment occurance under that shown. When we have found the fragment the passed in element represents we can find the
+child fragment names, that is the fragment names used in this fragment. All
+matches against When the workspace folder is given as the element, or rather the A fragment node represents a literate project fragment in a Visual Studio
+Code tree view. The class For the visualization part we need a We further encode some more information in Each node in the tree view represents a fragment. When the tree item is used to
+denote a workspace folder the theme icon for The In addition to code completion we can provide hover information. We want to see
+the implementation of fragments when hovering of fragment usages. That way code
+inspection can be easier done. We'll create The We get the current line of text from the document. We are going to look only for
+tags that are on one line. In the future it would be nice to add support for
+cases where mentioning a fragment in explaining text is split over several lines
+due to word wrapping, but with the current implementation we'll look only at
+those that are on one line. Next we need to know the the workspace folder for the given document so that we
+can query the correct project for the fragments. If no workspace folder was
+determined return Fragments are now available so we can see if we have a fragment under our
+cursor. If we do, and the fragment is not one that defines or appends to a
+fragment we know our cursor is over either fragment usage in a code fence or a
+fragment mention in explaining text. For this we can create a If that is not the case our With the workspace folder in hand we can ask the Literate Programming
diff --git a/index.literate b/index.literate
index c4b5883..dd8837f 100644
--- a/index.literate
+++ b/index.literate
@@ -3,5 +3,8 @@
A Visual Studio Code extension for writing literate programs.
* [Literate Programming](literate/literate.html), the main extension program
+* [Fragment Explorer](literate/fragment_explorer.html)
+* [Code Completion](literate/code_completion.html)
+* [Hovers](literate/hovers.html)
* [Grabber](literate/grabber.html), plug-in code used to grab the MarkdownIt
parser state
diff --git a/literate/code_completion.html b/literate/code_completion.html
new file mode 100644
index 0000000..4e9efdf
--- /dev/null
+++ b/literate/code_completion.html
@@ -0,0 +1,95 @@
+
+
+
+
+
+ Code completion
+
+const completionItemProvider =
+ vscode.languages.registerCompletionItemProvider('markdown', {
+ <<implement provide completion items>>
+}, '<');
+context.subscriptions.push(completionItemProvider);
+
Providing completion items
+CompletionItem
for each fragment
+we currently know of. Although the provider gets passed in the TextDocument
+for which it was triggered we will present fragments from the entire project.
+async provideCompletionItems(
+ document : vscode.TextDocument,
+ ..._
+)
+{
+
<<setup variables for providing completion items>>
we figure out to which
+workspace folder the current TextDocument
. If no workspace folder can be
+determined we return an empty array. This can happen with an unsaved new file,
+or when documents were opened that are not part of the workspace.
+ <<setup variables for providing completion items>>
+ <<get workspace for TextDocument>>
+
+ <<get fragments for completion items>>
+
completionItems
that
+we return when done.
+ <<for each fragment create a completion item>>
+ return completionItems;
+}
+
Setting up variables
+Array<CompletionItem>
.
+let completionItems : Array<vscode.CompletionItem> =
+ new Array<vscode.CompletionItem>();
+
Workspace folder for TextDocument
+..
we found the workspace folder where the document is from.
+const workspaceFolder : vscode.WorkspaceFolder | undefined = determineWorkspaceFolder(document);
+if(!workspaceFolder) { return []; }
+
Retrieving fragments of project
+FragmentRepository
in most cases handles processing of literate files
+automatically, but it skips that when a change is caused by typing <
, the
+opening chevron. That means we need to ensure literate files are processed
+before getting the fragment map for our workspace folder.
+await theOneRepository.processLiterateFiles(workspaceFolder);
+let fragments = theOneRepository.getFragments(workspaceFolder).map;
+
Creating the CompletionItems
+FragmentInformation
. Now we can create the
+CompletionItem
with the fragmentName
as its content.Reference
so that we get a nice
+icon in the completion list pop-up.
+
+
+
\ No newline at end of file
diff --git a/literate/code_completion.literate b/literate/code_completion.literate
new file mode 100644
index 0000000..9e629bc
--- /dev/null
+++ b/literate/code_completion.literate
@@ -0,0 +1,123 @@
+# Code completion
+
+A simple implementation to provide code completion will help authors writing
+their literate programs. Having possible tag names suggested will help
+decreasing the cognitive load of remembering all code fragment names in a
+literate project. This project itself has well over 50 fragments, and having to
+remember them by name is not easy.
+
+Until there is a good **literate** file type integration with Visual Studio Code
+we'll be relying on the built-in **Markdown** functionality.
+
+``` ts : <for(const fragmentName of fragments.keys())
+{
+ const fragment : FragmentInformation | undefined = fragments.get(fragmentName);
+ if(!fragment) {
+ continue;
+ }
+ const fragmentCompletion = new vscode.CompletionItem(fragmentName);
+ fragmentCompletion.detail = fragment.code;
+ fragmentCompletion.kind = vscode.CompletionItemKind.Reference;
+ completionItems.push(fragmentCompletion);
+}
+
Fragment explorer
+TreeView
that uses FragmentNodeProvider
+to show fragments available in a workspace. The tree view has FragmentNode
as
+its type parameter.
+export class FragmentExplorer {
+ private fragmentView : vscode.TreeView<FragmentNode>;
+ constructor(context : vscode.ExtensionContext) {
+ const fragmentNodeProvider = new FragmentNodeProvider();
+ context.subscriptions.push(
+ vscode.window.registerTreeDataProvider(
+ 'fragmentExplorer',
+ fragmentNodeProvider
+ )
+ );
+ this.fragmentView = vscode.window.createTreeView(
+ 'fragmentExplorer',
+ {
+ treeDataProvider : fragmentNodeProvider
+ });
+
+ context.subscriptions.push(
+ vscode.commands.registerCommand(
+ 'fragmentExplorer.refreshEntry',
+ () => fragmentNodeProvider.refresh())
+ );
+ context.subscriptions.push(vscode.workspace.onDidChangeTextDocument(
+ _ => {
+ fragmentNodeProvider.refresh();
+ }
+ ));
+ context.subscriptions.push(this.fragmentView);
+ }
+}
+
Fragment tree provider
+TreeDataProvider
+implementation to present the fragment structure to Visual Studio Code so that
+the data can be visualized in the fragmentExplorer
custom view.FragmentNodeProvider
implements a TreeDataProvider
with
+FragmentNode
as the tree item.
+export class FragmentNodeProvider implements vscode.TreeDataProvider<FragmentNode>
+{
+ <<fragment node provider members>>
+ <<fragment node provider API>>
+}
+
FragmentNodeProvider
gives as method to update the tree view
+refresh(): void {
+ <<refresh fragment node provider>>
+}
+
onDidChangeTreeData
event but
+could do more work if needed. To that end there is a private member for emitting
+the event, and the actual event to which the event emitter is published.
+private _onDidChangeTreeData:
+ vscode.EventEmitter<
+ FragmentNode |
+ undefined |
+ void
+ > = new vscode.EventEmitter<FragmentNode | undefined | void>();
+readonly onDidChangeTreeData :
+ vscode.Event<
+ FragmentNode |
+ undefined |
+ void
+ > = this._onDidChangeTreeData.event;
+
refresh
function can fire the event whenever
+called.
+this._onDidChangeTreeData.fire();
+
TreeDataProvider
implementation provided by FragmentNodeProvider
is
+completed by getTreeItem
and getChildren
. The first one is simple, it just
+returns the element that is passed to it, as there is no need to find out more
+information about this. Instead, elements have been already created by the
+getChildren
function, where all FragmentNode
instances are created with all
+the data necessary.
+getTreeItem(element : FragmentNode): vscode.TreeItem {
+ <<get fragment tree item>>
+}
+
getTreeItem
implementation remains simple
+return element;
+
getChildren
function is more involved. Yet its job is
+simple: get all FragmentNode
s that represent the direct children of the
+element given.
+async getChildren(element? : FragmentNode): Promise<FragmentNode[]>
+{
+ <<get direct children>>
+}
+
+if(!vscode.workspace.workspaceFolders ||
+ (
+ vscode.workspace.workspaceFolders &&
+ vscode.workspace.workspaceFolders.length < 1
+ )) {
+ vscode.window.showInformationMessage('No fragments in empty workspace');
+ return Promise.resolve([]);
+}
+
+if(!element)
+{
+ <<get children for workspace folders>>
+}
+else
+{
+ <<get children for element>>
+}
+
parentName
+specified. As folderName
we pass on the workspace folder name. This is a
+property all its children and the rest of its offspring inherit. The
+folderName
is used to find the correct workspace folder to search for the
+given element and its offspring.
+let arr = new Array<FragmentNode>();
+for(const wsFolder of vscode.workspace.workspaceFolders)
+{
+ arr.push(
+ new FragmentNode(
+ wsFolder.name,
+ new vscode.MarkdownString('$(book) (workspace folder)', true),
+ 'Workspace folder containing a literate project',
+ vscode.TreeItemCollapsibleState.Collapsed,
+ wsFolder.name,
+ undefined,
+ wsFolder,
+ undefined));
+}
+return Promise.resolve(arr);
+
folderName
for ease of access. Then we also creat an array of
+FragmentNode
s.
+const folderName : string = element.folderName;
+const fldr : vscode.WorkspaceFolder = element.workspaceFolder;
+let arr = new Array<FragmentNode>();
+
fragments
+map of the workspace folder in hand we can iterate over the keys in the
+fragments
map.parentName
set we know it is a fragment in the document level, so a
+fragment that was created. In contrast for a fragment there are child fragments,
+meaning that in the fragment code block other fragments were used. These are
+presented in the tree view as children to that fragment.
+<<get fragment family for offspring search>>
+for(const fragmentName of fragments.keys() )
+{
+ if(!element.parentName) {
+ <<create fragment node for document level>>
+ }
+ else if (fragmentName === element.label) {
+ <<create fragment node for fragment parent>>
+ }
+}
+
+return Promise.resolve(arr);
+
Getting all fragments
+FragmentNode
s from iterate over the
+literate files in the workspace folder that we determined we need to search.
+Then build the fragment map based on the tokens generated by the iteration pass.
+As a reminder the fragments map has the fragment name as key and the
+corresponding FragmentInformation
as the value to that key.
+const fragments = theOneRepository.getFragments(fldr).map;
+
TODO: build proper fragment hierarchy from fragments map
+Fragment used in other fragment
+FRAGMENT_USE_IN_CODE_RE
are found and for each case a
+corresponding FragmentNode
is created to function as a child to our parent
+element.
+let fragmentInfo = fragments.get(fragmentName) || undefined;
+if (fragmentInfo) {
+ const casesToReplace = [...fragmentInfo.code.matchAll(FRAGMENT_USE_IN_CODE_RE)];
+ for (let match of casesToReplace) {
+ if(!match || !match.groups)
+ {
+ continue;
+ }
+ let tag = match[0];
+ let ident = match.groups.ident;
+ let tagName = match.groups.tagName;
+ let root = match.groups.root;
+ let add = match.groups.add;
+ arr.push(
+ new FragmentNode(
+ tagName,
+ new vscode.MarkdownString(`$(symbol-file) ${fragmentInfo.literateFileName}`, true),
+ fragmentName,
+ vscode.TreeItemCollapsibleState.Collapsed,
+ folderName,
+ element.label,
+ element.workspaceFolder,
+ undefined
+ )
+ );
+ }
+}
+
Fragment on document level
+parentName
+of the given element is undefined, we have a fragment on document level. There
+are two types of fragments we want to discern beetween: top level fragments, or
+fragments that also tell us what file to create, and other fragments. A
+literate document can contain multiple top level fragments. But each top
+level fragment will generate only one source code file.
+let fragmentType : vscode.MarkdownString;
+let fragmentInfo = fragments.get(fragmentName) || undefined;
+if (fragmentInfo) {
+ if(fragmentName.indexOf(".*") >= 0)
+ {
+ fragmentType = new vscode.MarkdownString(
+ `$(globe): ${fragmentInfo.literateFileName}`,
+ true);
+ }
+ else
+ {
+ fragmentType = new vscode.MarkdownString(
+ `$(code): ${fragmentInfo.literateFileName}`,
+ true);
+ }
+ arr.push(
+ new FragmentNode(
+ fragmentName,
+ fragmentType,
+ fragmentInfo.literateFileName,
+ vscode.TreeItemCollapsibleState.Collapsed,
+ folderName,
+ element.label,
+ element.workspaceFolder,
+ undefined));
+}
+
Fragment node for tree view
+FragmentNode
extends the vscode.TreeItem
. Apart
+from just showing basic information like the fragment name and the file it is
+defined in we use FragmentNode
also to keep track of the workspace folder it
+is hosted in as well as the text document if there is one. Text documents are
+documents the workspace currently has opened. We need to take these into
+account so that we can directly use these as part of the literate document
+parsing.
+class FragmentNode extends vscode.TreeItem
+{
+ constructor (
+ <<fragment node readonly members>>
+ )
+ {
+ <<fragment node initialization>>
+ }
+}
+
label
, a tooltip
, a description
and a
+collapsibleState
. These are the only pieces of information needed that show up
+in the tree view.
+public readonly label : string,
+public readonly tooltip : vscode.MarkdownString,
+public readonly description : string,
+public readonly collapsibleState : vscode.TreeItemCollapsibleState,
+
FragmentNode
so that subsequent
+parsing can be done much more efficiently.
+public readonly folderName: string,
+public readonly parentName : string | undefined,
+public readonly workspaceFolder : vscode.WorkspaceFolder,
+public readonly textDocument : vscode.TextDocument | undefined
+
'book'
is used. Actual fragments
+get the theme icon for 'code'
.
+super(label, collapsibleState);
+this.tooltip = tooltip;
+this.description = description;
+this.iconPath = this.parentName ?
+ new vscode.ThemeIcon('code')
+ : new vscode.ThemeIcon('book');
+this.contextValue = 'literate_fragment';
+
registering FragmentNodeProvider
+FragmentNodeProvide
needs to be registered with Visual Studio Code so it
+can work when literate files are found in a work space.
+
+
+
\ No newline at end of file
diff --git a/literate/fragment_explorer.literate b/literate/fragment_explorer.literate
new file mode 100644
index 0000000..2d23eb0
--- /dev/null
+++ b/literate/fragment_explorer.literate
@@ -0,0 +1,377 @@
+# Fragment explorer
+
+The Literate Fragment Explorer is a `TreeView` that uses `FragmentNodeProvider`
+to show fragments available in a workspace. The tree view has `FragmentNode` as
+its type parameter.
+
+``` ts : <new FragmentExplorer(context);
+
Hover elements
+FragmentHoverProvider
which implements HoverProvider
.
+export class FragmentHoverProvider implements vscode.HoverProvider {
+ readonly fragmentRepository : FragmentRepository;
+ constructor(repository : FragmentRepository)
+ {
+ this.fragmentRepository = repository;
+ }
+ <<hover provider method>>
+}
+
FragmentHoverProvider
implements provideHover
. This will create the
+Hover
item if under the current cursor position there is a fragment, including
+its opening and closing double chevrons.
+public async provideHover(
+ document : vscode.TextDocument,
+ position : vscode.Position,
+ _: vscode.CancellationToken
+)
+{
+ <<get current line>>
+ <<find workspace folder for hover detection>>
+ <<create hover item for fragment>>
+ return null;
+}
+
+const currentLine = document.lineAt(position.line);
+
null
, as there is no literate project associated with the
+given document.
+const workspaceFolder : vscode.WorkspaceFolder | undefined = determineWorkspaceFolder(document);
+if(!workspaceFolder) { return null; }
+
Hover
with the
+code of the fragment as a MarkdownString
in a code fence.provideHover
implementation will return null
.
+const matchesOnLine = [...currentLine.text.matchAll(FRAGMENT_USE_IN_CODE_RE)];
+for(const match of matchesOnLine)
+{
+ if(!match || !match.groups) {
+ continue;
+ }
+ const foundIndex = currentLine.text.indexOf(match[0]);
+ if(foundIndex>-1) {
+ <<get fragments for hover detection>>
+ if(foundIndex <= position.character && position.character <= foundIndex + match[0].length && fragments.has(match.groups.tagName))
+ {
+ const startPosition = new vscode.Position(currentLine.lineNumber, foundIndex);
+ const endPosition = new vscode.Position(currentLine.lineNumber, foundIndex + match[0].length);
+ let range : vscode.Range = new vscode.Range(startPosition, endPosition);
+ let fragment = fragments.get(match.groups.tagName) || undefined;
+ if (fragment && !match.groups.root) {
+ return new vscode.Hover(
+ new vscode.MarkdownString(`~~~ ${fragment.lang}\n${fragment.code}\n~~~`, true),
+ range);
+ }
+ }
+ }
+}
+
FragmentRepository
for the
+fragment map that has been generated for the workspace folder.
+
+
+
\ No newline at end of file
diff --git a/literate/hovers.literate b/literate/hovers.literate
new file mode 100644
index 0000000..680e11a
--- /dev/null
+++ b/literate/hovers.literate
@@ -0,0 +1,97 @@
+# Hover elements
+
+In addition to code completion we can provide hover information. We want to see
+the implementation of fragments when hovering of fragment usages. That way code
+inspection can be easier done.
+
+We'll create `FragmentHoverProvider` which implements `HoverProvider`.
+
+``` ts : <let fragments = this.fragmentRepository.getFragments(workspaceFolder).map;
+
Literate Programming
create multiple source files within just one literate document.
This text describes the Literate Programming extension as a literate program.
+The tools provided by the Literate Programming extension are built around +one repository of the project providing all necessary information around +fragments.
+The fragment repository handles parsing of literate documents, reacting to +changes made by users. The repository provides all fragments found in the +projects added to the current workspace. Additionally the repository will write +out source files and rendered HTML files.
+The fragment model is defined in the FragmentRepository
class, which will be
+described in detail after introducing a couple of classes that help the
+repository.
The FragmentMap
class holds a map of strings, the fragment names, and their
+associated FragmentInformation
. This map is available through the map
+property. The class provides also a clear
method and a dispose
method.
class FragmentMap {
+ map : Map<string, FragmentInformation>;
+
+ constructor()
+ {
+ this.map = new Map<string, FragmentInformation>();
+ }
+
+ clear()
+ {
+ this.map.clear();
+ }
+
+ dispose()
+ {
+ this.map.clear();
+ }
+};
+
+The class GrabbedStateList
holds an array of GrabbedState
accessible through
+the list
property. The class provides clear
and dispose
properties.
class GrabbedStateList {
+ list : Array<GrabbedState>;
+
+ constructor()
+ {
+ this.list = new Array<GrabbedState>();
+ }
+
+ clear()
+ {
+ this.list = new Array<GrabbedState>();
+ }
+
+ dispose()
+ {
+ while(this.list.length>0)
+ {
+ this.list.pop();
+ }
+ }
+};
+
+The FragmentRepository
uses several helper classes, these we introduce right
+before defining the repository class.
<<fragment map>>
+<<list of grabbed states>>
+
+export class FragmentRepository {
+ <<fragment repository member variables>>
+ <<fragment repository constructor>>
+ <<fragment generation method>>
+
+ <<method to get fragments from repository>>
+
+ dispose() {
+ for(let fragmentMap of this.fragmentsForWorkspaceFolders.values())
+ {
+ fragmentMap.dispose();
+ }
+ this.fragmentsForWorkspaceFolders.clear();
+
+ for(let grabbedState of this.grabbedStateForWorkspaceFolders.values())
+ {
+ grabbedState.dispose();
+ }
+ this.grabbedStateForWorkspaceFolders.clear();
+ }
+}
+
+Our FragmentRepository
needs a couple of member variables to function
+properly. We'll need an instance of a properly configured MarkdownIt parser.
private md : MarkdownIt;
+
+Since we work with a multi-root workspace we'll create a map of maps. The keys
+for this top-level map will be the workspace folder names. The actual
+FragmentMap
s will be the values to each workspace folder.
readonly fragmentsForWorkspaceFolders : Map<string, FragmentMap>;
+
+For our parsing functionality we need an Array<GrabbedState>
, which we have
+encapsulated in the class GrabbedStateList
and is available through the list
+property. Each GrabbedStateList
is saved to the map of workspace folder name
+and list key-value pair.
readonly grabbedStateForWorkspaceFolders : Map<string, GrabbedStateList>;
+
+Finally we need a DiagnosticCollection
to be able to keep track of detected
+problems in literate projects. TBD: this probably needs to be changed into a
+map of DiagnosticCollection
, again with the workspace folder names as keys.
readonly diagnostics : vscode.DiagnosticCollection;
+
+The constructor takes an extension context to register any disposables there.
+constructor(
+ context : vscode.ExtensionContext
+)
+{
+ <<initializing the fragment repository members>>
+
+ <<subscribe to text document changes>>
+ <<subscribe to workspace changes>>
+ context.subscriptions.push(
+ vscode.workspace.onDidChangeWorkspaceFolders(
+ async (e : vscode.WorkspaceFoldersChangeEvent) =>
+ {
+ for(const addedWorkspaceFolder of e.added) {
+ await this.processLiterateFiles(addedWorkspaceFolder);
+ }
+ for(const removedWorkspaceFolder of e.removed)
+ {
+ this.fragmentsForWorkspaceFolders.delete(removedWorkspaceFolder.name);
+ this.grabbedStateForWorkspaceFolders.delete(removedWorkspaceFolder.name);
+ }
+ }
+ )
+ );
+}
+
+this.md = createMarkdownItParserForLiterate();
+this.fragmentsForWorkspaceFolders = new Map<string, FragmentMap>();
+this.grabbedStateForWorkspaceFolders = new Map<string, GrabbedStateList>();
+this.diagnostics = vscode.languages.createDiagnosticCollection('literate');
+context.subscriptions.push(this.diagnostics);
+
+The repository subscribes to the onDidChangeTextDocument
event on the
+workspace. It could process literate files on each change, but the
+completion item provider needs to trigger itself processing of literate files.
+Since completion item provider gets called on typing a opening chevron (<
) we
+skip triggering the processing here when such a character has been typed.
context.subscriptions.push(
+ vscode.workspace.onDidChangeTextDocument(
+ async (e : vscode.TextDocumentChangeEvent) =>
+ {
+ if(!(e.contentChanges.length>0 && e.contentChanges[0].text.startsWith('<')))
+ {
+ await this.processLiterateFiles(e.document);
+ }
+ }
+ )
+);
+
+Triggering of processing literate documents is necessary when new workspace +folders have been added. Additionally we need to clean up fragment maps and +grabbed states for those workspace folders that have been removed from the +workspace folder.
+context.subscriptions.push(
+ vscode.workspace.onDidChangeWorkspaceFolders(
+ async (e : vscode.WorkspaceFoldersChangeEvent) =>
+ {
+ for(const addedWorkspaceFolder of e.added) {
+ await this.processLiterateFiles(addedWorkspaceFolder);
+ }
+ for(const removedWorkspaceFolder of e.removed)
+ {
+ this.fragmentsForWorkspaceFolders.delete(removedWorkspaceFolder.name);
+ this.grabbedStateForWorkspaceFolders.delete(removedWorkspaceFolder.name);
+ }
+ }
+ )
+);
+
+The parsing and setting up of the fragments
map is handled with the method
+processLiterateFiles
. Additionally the method will write out all specified
+source files.
Processing the literate files is started generally in one of three cases: 1) change
+in workspace due to addition or removal of a workspace folder, 2) change to a
+literate document or through triggering of the literate.process
command.
async processLiterateFiles(
+ trigger :
+ vscode.WorkspaceFolder
+ | vscode.TextDocument
+ | undefined) {
+ <<set up workspace folder array>>
+ <<iterate over workspace folders and parse>>
+}
+
+First we determine the workspace folder or workspace folders to process. In the
+case where trigger
is a workspace folder or a text document we use the given
+workspace folder or determine the one to which the text document belongs. In
+these cases we'll have an array with just the one workspace folder as element.
+When the trigger is undefined
we'll use all workspace folders registered to
+this workspace.
const workspaceFolders : Array<vscode.WorkspaceFolder> | undefined = (() => {
+ if(trigger)
+ {
+ <<get workspace if text document>>
+ <<else just use passed in workspace>>
+ if("eol" in trigger) {
+ const ws = determineWorkspaceFolder(trigger);
+ if(ws)
+ {
+ return [ws];
+ }
+ } else {
+ return [trigger];
+ }
+ }
+ if(vscode.workspace.workspaceFolders && vscode.workspace.workspaceFolders.length>0) {
+ let folders = new Array<vscode.WorkspaceFolder>();
+ for(const ws of vscode.workspace.workspaceFolders)
+ {
+ folders.push(ws);
+ }
+ return folders;
+ }
+ return undefined;
+}
+)();
+
+We can check if our trigger
is a TextDocument
to see if eol
is a property.
+Otherwise it is a Workspace
.
if("eol" in trigger) {
+ const ws = determineWorkspaceFolder(trigger);
+ if(ws)
+ {
+ return [ws];
+ }
+}
+
+else
+{
+ return [trigger];
+}
+
+if(workspaceFolders) {
+ const writeOutHtml : WriteRenderCallback =
+ (fname : string,
+ folderUri : vscode.Uri,
+ rendered : string) : Thenable<void> => {
+ const html =
+`<html>
+ <head>
+ <link rel="stylesheet" type="text/css" href="./style.css">
+ </head>
+ <body>
+ ${rendered}
+ </body>
+</html>`;
+ const encoded = Buffer.from(html, 'utf-8');
+ fname = fname.replace(".literate", ".html");
+ const fileUri = vscode.Uri.joinPath(folderUri, fname);
+ return Promise.resolve(vscode.workspace.fs.writeFile(fileUri, encoded));
+ };
+ for(const folder of workspaceFolders)
+ {
+ if(!this.fragmentsForWorkspaceFolders.has(folder.name))
+ {
+ this.fragmentsForWorkspaceFolders.set(folder.name, new FragmentMap());
+ }
+ if(!this.grabbedStateForWorkspaceFolders.has(folder.name))
+ {
+ this.grabbedStateForWorkspaceFolders.set(folder.name, new GrabbedStateList());
+ }
+ const fragments = this.fragmentsForWorkspaceFolders.get(folder.name);
+ const grabbedStateList = this.grabbedStateForWorkspaceFolders.get(folder.name);
+ if(fragments && grabbedStateList) {
+ fragments.clear();
+ grabbedStateList.clear();
+ await iterateLiterateFiles(folder,
+ writeOutHtml, /* writeHtml : WriteRenderCallback*/
+ grabbedStateList.list,
+ this.md);
+ this.diagnostics.clear();
+ fragments.map = await handleFragments(folder, grabbedStateList.list, this.diagnostics, false, undefined);
+ this.diagnostics.clear();
+ await handleFragments(folder, grabbedStateList.list, this.diagnostics, true, writeSourceFiles);
+ }
+ }
+}
+
+When we call getFragments
we assume the literate projects have all been
+process properly. In most cases that is triggered automatically, but it may be
+necessary to trigger the processing manually before calling getFragments
. When
+the projects have been properly processed, though, this function returns the
+FragmentMap
for the given workspace folder.
getFragments(workspaceFolder : vscode.WorkspaceFolder) : FragmentMap
+{
+ let fragmentMap : FragmentMap = new FragmentMap();
+ this.fragmentsForWorkspaceFolders.forEach(
+ (value, key, _) =>
+ {
+ if(key===workspaceFolder.name)
+ {
+ fragmentMap = value;
+ }
+ }
+ );
+
+ return fragmentMap;
+}
+
As mentioned in the introduction the main idea of the extension is to collect
all fragments that are created in all .literate
files. Once all fragments have
@@ -139,34 +452,19 @@
GrabberPlugin
registered, which provides a rule that helps us collecting the
states of each rendered file. The grabbed state is collected in gstate
, which
is an instance of the StateCore
, provided by MarkdownIt.
-/**
- * Interface for environment to hold the Markdown file name and the StateCore
- * grabbed by the grabberPlugin.
- * The gstate we use to access all the tokens generated by the MarkdownIt parser.
- *
- * @see StateCore
- */
-interface GrabbedState {
- /**
- * File name of the Markdown document to which the state belongs.
- */
+The interface defines literateFileName
, which is the filename of the
+literate document to which the grabbed state belongs. literateUri
is the
+full uri for this document. Finally gstate
holds the StateCore
of the
+parsing result.
+interface GrabbedState {
literateFileName: string;
- /**
- * Uri for the Markdown document.
- */
literateUri: vscode.Uri;
- /**
- * State grabbed from the MarkdownIt parser.
- */
gstate: StateCore;
}
Preparing MarkdownIt
In the iterateLiterateFiles
we start by setting up the MarkdownIt parser.
-/**
- * MarkdownIt instance with grabber_plugin in use.
- */
-const md : MarkdownIt = createMarkdownItParserForLiterate();
+const md : MarkdownIt = createMarkdownItParserForLiterate();
The function createMarkdownItParserForLiterate
does this setup so that it is
easy to get a new parser to use for different purposes, like parsing documents
@@ -215,7 +513,7 @@
Fragment use in code
of fragments in code we use FRAGMENT_USE_IN_CODE_RE
.
//let HTML_ENCODED_FRAGMENT_TAG_RE = /(<<.*?>>)/g;
let FRAGMENT_USE_IN_CODE_RE =
- /(?<indent>[ \t]*)<<(?<tagName>.*)>>(?<root>=)?(?<add>\+)?/g;
+ /(?<indent>[ \t]*)<<(?<tagName>.+)>>(?<root>=)?(?<add>\+)?/g;
The regular expression captures four groups. A match will give us 5 or more
results, the whole string matched and the captured groups. There may be some
@@ -237,7 +535,7 @@
Creating and modifying fragments
block. The actual fragment tag is placed as first option right after the colon
following the language specifier.
let FRAGMENT_RE =
- /(?<lang>.*):.*<<(?<tagName>.*)>>(?<root>=)?(?<add>\+)?\s*(?<fileName>.*)/;
+ /(?<lang>.*):.*<<(?<tagName>.+)>>(?<root>=)?(?<add>\+)?\s*(?<fileName>.*)/;
Most of the groups correspond to the ones defined by FRAGMENT_USE_IN_CODE_RE
with a few additions. Most notably there is the group catching the language
@@ -298,17 +596,10 @@
Populating the fragment map
First we build a map of all available fragments. These will go into fragments
,
which is of type Map<string, FragmentInformation>
. The name of a fragment will
function as the key, and an instance of FragmentInformation
will be the value.
-/**
- * Map of fragment names and tuples of code fragments for these. The
- * tuples contain code language identifier followed by the filename and
- * lastly followed by the actual code fragment.
- */
-const fragments = new Map<string, FragmentInformation>();
-// Now we have the state, we have access to the tokens
-// over which we can iterate to extract all the code
-// fragments and build up the map with the fragments concatenated
-// where necessary. We'll extrapolate all fragments in the second
-// pass.
+const fragments = new Map<string, FragmentInformation>();
+const overwriteAttempts = new Array<string>();
+const missingFilenames = new Array<string>();
+const addingToNonExistant = new Array<string>();
for (let env of envList) {
for (let token of env.gstate.tokens) {
<<handle fence tokens>>
@@ -348,15 +639,17 @@ Creating a new fragment
content in token.content
. Finally the new FragmentInformation
instance is
added to the fragments
map.
if (root && !add) {
- if (fragments.has(name)) {
+ if (fragments.has(name) && !overwriteAttempts.includes(name)) {
let msg = `Trying to overwrite existing fragment fragment ${name}. ${env.literateFileName}${linenumber}`;
const diag = createErrorDiagnostic(token, msg);
updateDiagnostics(env.literateUri, diagnostics, diag);
+ overwriteAttempts.push(name);
} else {
- if (!fileName && name.indexOf(".*") > -1) {
+ if (!fileName && name.indexOf(".*") > -1 && !missingFilenames.includes(name)) {
let msg = `Expected filename for star fragment ${name}`;
const diag = createErrorDiagnostic(token, msg);
updateDiagnostics(env.literateUri, diagnostics, diag);
+ missingFilenames.push(name);
} else {
let code = token.content;
let fragmentInfo: FragmentInformation = {
@@ -392,9 +685,12 @@ Modifying an exiting fragment
fragments.set(name, fragmentInfo);
}
} else {
- let msg = `Trying to add to non-existant fragment ${name}. ${env.literateFileName}:${linenumber}`;
- const diag = createErrorDiagnostic(token, msg);
- updateDiagnostics(env.literateUri, diagnostics, diag);
+ if(!addingToNonExistant.includes(name)) {
+ let msg = `Trying to add to non-existant fragment ${name}. ${env.literateFileName}:${linenumber}`;
+ const diag = createErrorDiagnostic(token, msg);
+ updateDiagnostics(env.literateUri, diagnostics, diag);
+ addingToNonExistant.push(name);
+ }
}
}
@@ -445,6 +741,9 @@ Extrapolating fragments
fragments can be combined into source code.
// for now do several passes
let pass: number = 0;
+const rootIncorrect = new Array<string>();
+const addIncorrect = new Array<string>();
+const fragmentNotFound = new Array<string>();
do {
pass++;
let fragmentReplaced = false;
@@ -464,20 +763,23 @@ Extrapolating fragments
let tagName = match.groups.tagName;
let root = match.groups.root;
let add = match.groups.add;
- if (root) {
+ if (root && !rootIncorrect.includes(tag)) {
let msg = `Found '=': incorrect fragment tag in fragment, ${tag}`;
const diag = createErrorDiagnostic(fragmentInfo.tokens[0], msg);
updateDiagnostics(fragmentInfo.env.literateUri, diagnostics, diag);
+ rootIncorrect.push(tag);
}
- if (add) {
+ if (add && !addIncorrect.includes(tag)) {
let msg = `Found '+': incorrect fragment tag in fragment: ${tag}`;
const diag = createErrorDiagnostic(fragmentInfo.tokens[0], msg);
updateDiagnostics(fragmentInfo.env.literateUri, diagnostics, diag);
+ addIncorrect.push(tag);
}
- if (!fragments.has(match.groups.tagName) && tagName !== "(?<tagName>.*)") {
+ if (!fragments.has(match.groups.tagName) && tagName !== "(?<tagName>.+)" && !fragmentNotFound.includes(tagName)) {
let msg = `Could not find fragment ${tag} (${tagName})`;
const diag = createErrorDiagnostic(fragmentInfo.tokens[0], msg);
updateDiagnostics(fragmentInfo.env.literateUri, diagnostics, diag);
+ fragmentNotFound.push(tagName);
}
let fragmentToReplaceWith = fragments.get(tagName) || undefined;
if (fragmentToReplaceWith) {
@@ -547,8 +849,7 @@ custom code fence rendering
Register the literate.process command
The command literate.process
is registered with Visual Studio Code. The
disposable that gets returned by registerCommand
is held in
-literateProcessDisposable
so that it can be used later on, for instance for
-diagnostics management.
+literateProcessDisposable
so that we can push it into context.subscriptions
.
Here we find the main program of our literate.process
command. Our
MarkdownIt is set up, .literate
files are searched and iterated. Each
.literate
file is rendered, and code fragments are harvested. Finally code
@@ -563,560 +864,10 @@
Register the literate.process command
let literateProcessDisposable = vscode.commands.registerCommand(
'literate.process',
async function () {
-
- <<set up MarkdownIt>>
-
- diagnostics.clear();
-
- if (!vscode.workspace.workspaceFolders) {
- return vscode.window.showInformationMessage("No workspace or folder opened");
- }
-
-
- const writeOutHtml : WriteRenderCallback =
- (fname : string,
- folderUri : vscode.Uri,
- rendered : string) : Thenable<void> => {
- const html =
-`<html>
- <head>
- <link rel="stylesheet" type="text/css" href="./style.css">
- </head>
- <body>
- ${rendered}
- </body>
-</html>`;
- const encoded = Buffer.from(html, 'utf-8');
- fname = fname.replace(".literate", ".html");
- const fileUri = vscode.Uri.joinPath(folderUri, fname);
- return Promise.resolve(vscode.workspace.fs.writeFile(fileUri, encoded));
- };
-
- for(const workspaceFolder of vscode.workspace.workspaceFolders) {
- const envList: Array<GrabbedState> = new Array<GrabbedState>();
- await iterateLiterateFiles(workspaceFolder, writeOutHtml, envList, md);
- let _ = await handleFragments(workspaceFolder, envList, diagnostics, true, writeSourceFiles);
- }
-
- let hasAnyDiagnostics = false;
- diagnostics.forEach(
- function(
- _: vscode.Uri,
- diags: readonly vscode.Diagnostic[],
- __: vscode.DiagnosticCollection
- ) : any {
- hasAnyDiagnostics ||= (diags.length > 0);
- }
- );
-
- if (hasAnyDiagnostics) {
- return vscode.window.setStatusBarMessage(
- (new vscode.MarkdownString(
- "$(error) Error encountered during process"
- )).value, 2000);
- }
- else {
+ theOneRepository.processLiterateFiles(undefined);
return vscode.window.setStatusBarMessage("Literate Process completed", 5000);
- }
});
-Fragment explorer
-The Literate Fragment Explorer is a TreeView
that uses FragmentNodeProvider
-to show fragments available in a workspace. The tree view has FragmentNode
as
-its type parameter.
-export class FragmentExplorer {
- private fragmentView : vscode.TreeView<FragmentNode>;
- constructor(context : vscode.ExtensionContext) {
- const fragmentNodeProvider = new FragmentNodeProvider();
- context.subscriptions.push(
- vscode.window.registerTreeDataProvider(
- 'fragmentExplorer',
- fragmentNodeProvider
- )
- );
- this.fragmentView = vscode.window.createTreeView(
- 'fragmentExplorer',
- {
- treeDataProvider : fragmentNodeProvider
- });
-
- context.subscriptions.push(
- vscode.commands.registerCommand(
- 'fragmentExplorer.refreshEntry',
- () => fragmentNodeProvider.refresh())
- );
- context.subscriptions.push(vscode.workspace.onDidChangeTextDocument(
- _ => {
- fragmentNodeProvider.refresh();
- }
- ));
- context.subscriptions.push(this.fragmentView);
- }
-}
-
-Fragment tree provider
-The Literate Fragment Explorer needs a
-TreeDataProvider
-implementation to present the fragment structure to Visual Studio Code so that
-the data can be visualized in the fragmentExplorer
custom view.
-The class FragmentNodeProvider
implements a TreeDataProvider
with
-FragmentNode
as the tree item.
-export class FragmentNodeProvider implements vscode.TreeDataProvider<FragmentNode>
-{
- <<fragment node provider members>>
- <<fragment node provider API>>
-}
-
-The constructor takes care of all necessary initialization.
-constructor()
-{
- <<initialize fragment node provider>>
-}
-
-The constructor for the FragmentNodeProvider
creates an instance of the
-MarkdownIt
module, fully configured for our literate programming needs.
-Additionally a DiagnosticCollection
is created so that it can be passed on to
-the handleFragments
function that is utilized in the FragmentNodeProvider
.
-this.md = createMarkdownItParserForLiterate();
-this.diagnostics = vscode.languages.createDiagnosticCollection('literate-treeview');
-
-This means we need two members to hold these instances.
-private md : MarkdownIt;
-private diagnostics : vscode.DiagnosticCollection;
-
-The API for FragmentNodeProvider
gives as method to update the tree view
-refresh(): void {
- <<refresh fragment node provider>>
-}
-
-The current implementation simply fires the onDidChangeTreeData
event but
-could do more work if needed. To that end there is a private member for emitting
-the event, and the actual event to which the event emitter is published.
-private _onDidChangeTreeData:
- vscode.EventEmitter<
- FragmentNode |
- undefined |
- void
- > = new vscode.EventEmitter<FragmentNode | undefined | void>();
-readonly onDidChangeTreeData :
- vscode.Event<
- FragmentNode |
- undefined |
- void
- > = this._onDidChangeTreeData.event;
-
-With those two in place the refresh
function can fire the event whenever
-called.
-this._onDidChangeTreeData.fire();
-
-The TreeDataProvider
implementation provided by FragmentNodeProvider
is
-completed by getTreeItem
and getChildren
. The first one is simple, it just
-returns the element that is passed to it, as there is no need to find out more
-information about this. Instead, elements have been already created by the
-getChildren
function, where all FragmentNode
instances are created with all
-the data necessary.
-getTreeItem(element : FragmentNode): vscode.TreeItem {
- <<get fragment tree item>>
-}
-
-As said, the getTreeItem
implementation remains simple
-return element;
-
-On the other hand the getChildren
function is more involved. Yet its job is
-simple: get all FragmentNode
s that represent the direct children of the
-element given.
-async getChildren(element? : FragmentNode): Promise<FragmentNode[]>
-{
- <<get direct children>>
-}
-
-When the workspace has no workspace folders at all there will be no children to
-return, as there are no literate documents to begin with.
-if(!vscode.workspace.workspaceFolders ||
- (
- vscode.workspace.workspaceFolders &&
- vscode.workspace.workspaceFolders.length < 1
- )) {
- vscode.window.showInformationMessage('No fragments in empty workspace');
- return Promise.resolve([]);
-}
-
-If we do have workspace folders, but no element is given to look for children we
-need to look at the all the fragments available in all documents across all
-workspace folders. If on the other hand an element is given then its children
-are retrieved.
-if(!element)
-{
- <<get children for workspace folders>>
-}
-else
-{
- <<get children for element>>
-}
-
-When no element is passed we want the root of all the branches, where each
-workspace folder is the root of its own branch.
-To this end the children are all essentially the workspace folder names. Since
-these are the work folders the fragments representing them have no parentName
-specified. As folderName
we pass on the workspace folder name. This is a
-property all its children and the rest of its offspring inherit. The
-folderName
is used to find the correct workspace folder to search for the
-given element and its offspring.
-let arr = new Array<FragmentNode>();
-for(const wsFolder of vscode.workspace.workspaceFolders)
-{
- arr.push(
- new FragmentNode(
- wsFolder.name,
- new vscode.MarkdownString('$(book) (workspace folder)', true),
- 'Workspace folder containing a literate project',
- vscode.TreeItemCollapsibleState.Collapsed,
- wsFolder.name,
- undefined,
- wsFolder,
- undefined));
-}
-return Promise.resolve(arr);
-
-Getting the children for a given element is a bit more involved. First we set
-up a constant folderName
for ease of access. Then we also creat an array of
-FragmentNode
s.
-const folderName : string = element.folderName;
-const fldr : vscode.WorkspaceFolder = element.workspaceFolder;
-let arr = new Array<FragmentNode>();
-
-From the element we already learned the workspace folder for its project, so we
-can use that directly to parse the literate content. With the fragments
-map of the workspace folder in hand we can iterate over the keys in the
-fragments
map.
-There are essentially two cases we need to check for. If the given element has
-no parentName
set we know it is a fragment in the document level, so a
-fragment that was created. In contrast for a fragment there are child fragments,
-meaning that in the fragment code block other fragments were used. These are
-presented in the tree view as children to that fragment.
-<<get fragment family for offspring search>>
-for(const fragmentName of fragments.keys() )
-{
- if(!element.parentName) {
- <<create fragment node for document level>>
- }
- else if (fragmentName === element.label) {
- <<create fragment node for fragment parent>>
- }
-}
-
-return Promise.resolve(arr);
-
-Getting all fragments
-To find the fragment information to build FragmentNode
s from iterate over the
-literate files in the workspace folder that we determined we need to search.
-Then build the fragment map based on the tokens generated by the iteration pass.
-As a reminder the fragments map has the fragment name as key and the
-corresponding FragmentInformation
as the value to that key.
-let envList: Array<GrabbedState> = new Array<GrabbedState>();
-await iterateLiterateFiles(fldr, undefined, envList, this.md);
-const fragments = await handleFragments(fldr, envList, this.diagnostics, false, undefined);
-
-TODO: build proper fragment hierarchy from fragments map
-Still to do. Right now essentially the map structure is shown, but that isn't
-very useful. What we really need is a hierarchical form with each fragment under
-its parent fragment so that the structure of the literate program can be seen.
-Another improvement we could make is to show Markdown outline of chapters, with
-fragment occurance under that shown.
-Fragment used in other fragment
-When we have found the fragment the passed in element represents we can find the
-child fragment names, that is the fragment names used in this fragment. All
-matches against FRAGMENT_USE_IN_CODE_RE
are found and for each case a
-corresponding FragmentNode
is created to function as a child to our parent
-element.
-let fragmentInfo = fragments.get(fragmentName) || undefined;
-if (fragmentInfo) {
- const casesToReplace = [...fragmentInfo.code.matchAll(FRAGMENT_USE_IN_CODE_RE)];
- for (let match of casesToReplace) {
- if(!match || !match.groups)
- {
- continue;
- }
- let tag = match[0];
- let ident = match.groups.ident;
- let tagName = match.groups.tagName;
- let root = match.groups.root;
- let add = match.groups.add;
- arr.push(
- new FragmentNode(
- tagName,
- new vscode.MarkdownString(`$(symbol-file) ${fragmentInfo.literateFileName}`, true),
- fragmentName,
- vscode.TreeItemCollapsibleState.Collapsed,
- folderName,
- element.label,
- element.workspaceFolder,
- undefined
- )
- );
- }
-}
-
-Fragment on document level
-When the workspace folder is given as the element, or rather the parentName
-of the given element is undefined, we have a fragment on document level. There
-are two types of fragments we want to discern beetween: top level fragments, or
-fragments that also tell us what file to create, and other fragments. A
-literate document can contain multiple top level fragments. But each top
-level fragment will generate only one source code file.
-let fragmentType : vscode.MarkdownString;
-let fragmentInfo = fragments.get(fragmentName) || undefined;
-if (fragmentInfo) {
- if(fragmentName.indexOf(".*") >= 0)
- {
- fragmentType = new vscode.MarkdownString(
- `$(globe): ${fragmentInfo.literateFileName}`,
- true);
- }
- else
- {
- fragmentType = new vscode.MarkdownString(
- `$(code): ${fragmentInfo.literateFileName}`,
- true);
- }
- arr.push(
- new FragmentNode(
- fragmentName,
- fragmentType,
- fragmentInfo.literateFileName,
- vscode.TreeItemCollapsibleState.Collapsed,
- folderName,
- element.label,
- element.workspaceFolder,
- undefined));
-}
-
-Fragment node for tree view
-A fragment node represents a literate project fragment in a Visual Studio
-Code tree view. The class FragmentNode
extends the vscode.TreeItem
. Apart
-from just showing basic information like the fragment name and the file it is
-defined in we use FragmentNode
also to keep track of the workspace folder it
-is hosted in as well as the text document if there is one. Text documents are
-documents the workspace currently has opened. We need to take these into
-account so that we can directly use these as part of the literate document
-parsing.
-class FragmentNode extends vscode.TreeItem
-{
- constructor (
- <<fragment node readonly members>>
- )
- {
- <<fragment node initialization>>
- }
-}
-
-For the visualization part we need a label
, a tooltip
, a description
and a
-collapsibleState
. These are the only pieces of information needed that show up
-in the tree view.
-public readonly label : string,
-public readonly tooltip : vscode.MarkdownString,
-public readonly description : string,
-public readonly collapsibleState : vscode.TreeItemCollapsibleState,
-
-We further encode some more information in FragmentNode
so that subsequent
-parsing can be done much more efficiently.
-public readonly folderName: string,
-public readonly parentName : string | undefined,
-public readonly workspaceFolder : vscode.WorkspaceFolder,
-public readonly textDocument : vscode.TextDocument | undefined
-
-Each node in the tree view represents a fragment. When the tree item is used to
-denote a workspace folder the theme icon for 'book'
is used. Actual fragments
-get the theme icon for 'code'
.
-super(label, collapsibleState);
-this.tooltip = tooltip;
-this.description = description;
-this.iconPath = this.parentName ?
- new vscode.ThemeIcon('code')
- : new vscode.ThemeIcon('book');
-this.contextValue = 'literate_fragment';
-
-registering FragmentNodeProvider
-The FragmentNodeProvide
needs to be registered with Visual Studio Code so it
-can work when literate files are found in a work space.
-new FragmentExplorer(context);
-
-Code completion
-A simple implementation to provide code completion will help authors writing
-their literate programs. Having possible tag names suggested will help
-decreasing the cognitive load of remembering all code fragment names in a
-literate project. This project itself has well over 50 fragments, and having to
-remember them by name is not easy.
-Until there is a good literate file type integration with Visual Studio Code
-we'll be relying on the built-in Markdown functionality.
-const completionItemProvider =
- vscode.languages.registerCompletionItemProvider('markdown', {
- <<implement provide completion items>>
-}, '<');
-context.subscriptions.push(completionItemProvider);
-
-Providing completion items
-The completion item provider will generate a CompletionItem
for each fragment
-we currently know of. Although the provider gets passed in the TextDocument
-for which it was triggered we will present fragments from the entire project.
-async provideCompletionItems(
- document : vscode.TextDocument,
- ..._
-)
-{
-
-After setting up the necessary variables with
-<<setup variables for providing completion items>>
we figure out to which
-workspace folder the current TextDocument
. If no workspace folder can be
-determined we return an empty array. This can happen with an unsaved new file,
-or when documents were opened that are not part of the workspace.
- <<setup variables for providing completion items>>
- <<get workspace for TextDocument>>
-
-After the workspace folder has been determined we can gather all fragments in
-our project.
- <<get fragments for completion items>>
-
-Finally we generate the completion items into the array completionItems
that
-we return when done.
- <<for each fragment create a completion item>>
- return completionItems;
-}
-
-Setting up variables
-Completion items are going to be collected in an Array<CompletionItem>
.
-Further, creating completion items for code completion needs to parse the
-entire project, so we need an Array<GrabbedState>
. Iterating and parsing
-through the project also needs a DiagnosticCollection
, although we won't be
-using it any further. Lastly we create an instance of the MarkdownIt parser to
-give to iterateLiterateFiles
.
-let completionItems : Array<vscode.CompletionItem> =
- new Array<vscode.CompletionItem>();
-let envForCompletion : Array<GrabbedState> = new Array<GrabbedState>();
- new Array<vscode.CompletionItem>();
-const diagnostics = vscode.languages.createDiagnosticCollection('literate-completionitems');
-const md : MarkdownIt = createMarkdownItParserForLiterate();
-
-Workspace folder for TextDocument
-Determining the workspace folder for the given TextDocument is done by creating
-relative paths from each workspace folder to the document. If the path does not
-start with ..
we found the workspace folder where the document is from.
-If no workspace folders were found, or if the TextDocument did not have a
-workspace folder we essentially end up returning an empty array from the
-completion item provider.
-const workspaceFolder : vscode.WorkspaceFolder | undefined = determineWorkspaceFolder(document);
-if(!workspaceFolder) { return []; }
-
-Retrieving fragments of project
-Getting the fragments for our project means we iterateLiterateFiles
with the
-envForCompletion
given, along with the workspace folder and the MarkdownIt
-parser. Once we have iterated over all files, and thus envForCompletion
now
-contains all literate documents tokenized we can pass those to handleFragments
-so we can end up with a map of all fragments. We pass in false
to the function
-to ensure fragments aren't extrapolated: we want to show the fragments as they
-are in code completion.
- await iterateLiterateFiles(workspaceFolder, undefined, envForCompletion, md);
- let fragments = await handleFragments(workspaceFolder, envForCompletion, diagnostics, false, writeSourceFiles);
-
-Creating the CompletionItems
-With all fragments in the map we iterate over all the keys. For each key we
-fetch the corresponding FragmentInformation
. Now we can create the
-CompletionItem
with the fragmentName
as its content.
-Further the fragment code is set to be the detail of the completion item. This
-will provide a tooltip with the code fragment readable, so that it is easy to
-understand what fragment is currently highlighted in the completion list.
-Finally the set the completion item kind to Reference
so that we get a nice
-icon in the completion list pop-up.
- for(const fragmentName of fragments.keys())
- {
- const fragment : FragmentInformation | undefined = fragments.get(fragmentName);
- if(!fragment) {
- continue;
- }
- const fragmentCompletion = new vscode.CompletionItem(fragmentName);
- fragmentCompletion.detail = fragment.code;
- fragmentCompletion.kind = vscode.CompletionItemKind.Reference;
- completionItems.push(fragmentCompletion);
- }
-
-Hover elements
-In addition to code completion we can provide hover information. We want to see
-the implementation of fragments when hovering of fragment usages. That way code
-inspection can be easier done.
-We'll create FragmentHoverProvider
which implements HoverProvider
.
-class FragmentHoverProvider implements vscode.HoverProvider {
- <<hover provider method>>
-}
-
-The FragmentHoverProvider
implements provideHover
. This will create the
-Hover
item if under the current cursor position there is a fragment, including
-its opening and closing double chevrons.
-public async provideHover(
- document : vscode.TextDocument,
- position : vscode.Position,
- _: vscode.CancellationToken
-)
-{
- <<get current line>>
- <<find workspace folder for hover detection>>
- <<create hover item for fragment>>
- return null;
-}
-
-We get the current line of text from the document. We are going to look only for
-tags that are on one line. In the future it would be nice to add support for
-cases where mentioning a fragment in explaining text is split over several lines
-due to word wrapping, but with the current implementation we'll look only at
-those that are on one line.
-const currentLine = document.lineAt(position.line);
-
-Next we need to know the the workspace folder for the given document so that we
-can query the correct project for the fragments. If no workspace folder was
-determined return null
, as there is no literate project associated with the
-given document.
-const workspaceFolder : vscode.WorkspaceFolder | undefined = determineWorkspaceFolder(document);
-if(!workspaceFolder) { return null; }
-
-Fragments are now available so we can see if we have a fragment under our
-cursor. If we do, and the fragment is not one that defines or appends to a
-fragment we know our cursor is over either fragment usage in a code fence or a
-fragment mention in explaining text. For this we can create a Hover
with the
-code of the fragment as a MarkdownString
in a code fence.
-If that is not the case our provideHover
implementation will return null
.
-const matchesOnLine = [...currentLine.text.matchAll(FRAGMENT_USE_IN_CODE_RE)];
-for(const match of matchesOnLine)
-{
- if(!match || !match.groups) {
- continue;
- }
- const foundIndex = currentLine.text.indexOf(match[0]);
- if(foundIndex>-1) {
- <<get fragments for hover detection>>
- if(foundIndex <= position.character && position.character <= foundIndex + match[0].length && fragments.has(match.groups.tagName))
- {
- const startPosition = new vscode.Position(currentLine.lineNumber, foundIndex);
- const endPosition = new vscode.Position(currentLine.lineNumber, foundIndex + match[0].length);
- let range : vscode.Range = new vscode.Range(startPosition, endPosition);
- let fragment = fragments.get(match.groups.tagName) || undefined;
- if (fragment && !match.groups.root) {
- return new vscode.Hover(
- new vscode.MarkdownString(`~~~ ${fragment.lang}\n${fragment.code}\n~~~`, true),
- range);
- }
- }
- }
-}
-
-With the workspace folder in hand we can iterate over all literate files in the
-workspace and get the fragments for the project. We don't want extrapolated
-fragments, we want to see them as they are with fragment usages intact.
-const diagnostics = vscode.languages.createDiagnosticCollection('literate-completionitems');
-const md : MarkdownIt = createMarkdownItParserForLiterate();
-let envForCompletion : Array<GrabbedState> = new Array<GrabbedState>();
- new Array<vscode.CompletionItem>();
-await iterateLiterateFiles(workspaceFolder, undefined, envForCompletion, md);
-let fragments = await handleFragments(workspaceFolder, envForCompletion, diagnostics, false, writeSourceFiles);
-
Diagnostics
function updateDiagnostics(
uri: vscode.Uri,
@@ -1245,6 +996,7 @@ The extension
<<render and collect state>>
<<handle fragments>>
<<write out source files>>
+<<fragment repository>>
Utility function to determine the workspace folder for a TextDocument
function determineWorkspaceFolder(document : vscode.TextDocument) : vscode.WorkspaceFolder | undefined
@@ -1311,19 +1063,24 @@ Interfaces used in Literate Programming
<<fragment information type>>
Extension activation
-export function activate(context: vscode.ExtensionContext) {
+let theOneRepository : FragmentRepository;
+export async function activate(context: vscode.ExtensionContext) {
const rootPath = (vscode.workspace.workspaceFolders && (vscode.workspace.workspaceFolders.length > 0))
? vscode.workspace.workspaceFolders[0].uri.fsPath : undefined;
console.log('Ready to do some Literate Programming');
const diagnostics = vscode.languages.createDiagnosticCollection('literate');
+ theOneRepository = new FragmentRepository(context);
+ await theOneRepository.processLiterateFiles(undefined);
+ context.subscriptions.push(theOneRepository);
+
<<register literate.process>>
<<register fragment tree view>>
<<register completion item provider>>
context.subscriptions.push(
- vscode.languages.registerHoverProvider('markdown', new FragmentHoverProvider())
+ vscode.languages.registerHoverProvider('markdown', new FragmentHoverProvider(theOneRepository))
);
if (vscode.window.activeTextEditor) {
@@ -1336,11 +1093,6 @@ Extension activation
}));
context.subscriptions.push(literateProcessDisposable);
- context.subscriptions.push(vscode.workspace.onDidChangeTextDocument(
- _ => {
- vscode.commands.executeCommand('literate.process');
- }
- ));
return {
extendMarkdownIt(md: any) {
diff --git a/literate/literate.literate b/literate/literate.literate
index 9c848c2..93c28ea 100644
--- a/literate/literate.literate
+++ b/literate/literate.literate
@@ -45,6 +45,382 @@ create multiple source files within just one **literate** document.
This text describes the **Literate Programming** extension as a **literate**
program.
+## Fragment Model
+
+The tools provided by the **Literate Programming** extension are built around
+one repository of the project providing all necessary information around
+fragments.
+
+The fragment repository handles parsing of **literate** documents, reacting to
+changes made by users. The repository provides all fragments found in the
+projects added to the current workspace. Additionally the repository will write
+out source files and rendered HTML files.
+
+The fragment model is defined in the `FragmentRepository` class, which will be
+described in detail after introducing a couple of classes that help the
+repository.
+
+### FragmentMap class
+
+The `FragmentMap` class holds a map of strings, the fragment names, and their
+associated `FragmentInformation`. This map is available through the `map`
+property. The class provides also a `clear` method and a `dispose` method.
+
+``` ts : <>=
+class FragmentMap {
+ map : Map;
+
+ constructor()
+ {
+ this.map = new Map();
+ }
+
+ clear()
+ {
+ this.map.clear();
+ }
+
+ dispose()
+ {
+ this.map.clear();
+ }
+};
+```
+
+### List of GrabbedState
+
+The class `GrabbedStateList` holds an array of `GrabbedState` accessible through
+the `list` property. The class provides `clear` and `dispose` properties.
+
+``` ts : <>=
+class GrabbedStateList {
+ list : Array;
+
+ constructor()
+ {
+ this.list = new Array();
+ }
+
+ clear()
+ {
+ this.list = new Array();
+ }
+
+ dispose()
+ {
+ while(this.list.length>0)
+ {
+ this.list.pop();
+ }
+ }
+};
+```
+
+### The FragmentRepository class
+
+The `FragmentRepository` uses several helper classes, these we introduce right
+before defining the repository class.
+
+``` ts : <>=
+<>
+<>
+
+export class FragmentRepository {
+ <>
+ <>
+ <>
+
+ <>
+
+ dispose() {
+ for(let fragmentMap of this.fragmentsForWorkspaceFolders.values())
+ {
+ fragmentMap.dispose();
+ }
+ this.fragmentsForWorkspaceFolders.clear();
+
+ for(let grabbedState of this.grabbedStateForWorkspaceFolders.values())
+ {
+ grabbedState.dispose();
+ }
+ this.grabbedStateForWorkspaceFolders.clear();
+ }
+}
+```
+
+#### Member variables
+
+Our `FragmentRepository` needs a couple of member variables to function
+properly. We'll need an instance of a properly configured *MarkdownIt* parser.
+
+``` ts : <>=
+private md : MarkdownIt;
+```
+
+Since we work with a multi-root workspace we'll create a map of maps. The keys
+for this top-level map will be the workspace folder names. The actual
+`FragmentMap`s will be the values to each workspace folder.
+
+``` ts : <>=+
+readonly fragmentsForWorkspaceFolders : Map;
+```
+
+For our parsing functionality we need an `Array`, which we have
+encapsulated in the class `GrabbedStateList` and is available through the `list`
+property. Each `GrabbedStateList` is saved to the map of workspace folder name
+and list key-value pair.
+
+``` ts : <>=+
+readonly grabbedStateForWorkspaceFolders : Map;
+```
+
+Finally we need a `DiagnosticCollection` to be able to keep track of detected
+problems in **literate** projects. TBD: this probably needs to be changed into a
+map of `DiagnosticCollection`, again with the workspace folder names as keys.
+
+``` ts : <>=+
+readonly diagnostics : vscode.DiagnosticCollection;
+```
+
+#### Constructor
+
+The constructor takes an extension context to register any disposables there.
+
+``` ts : <>=
+constructor(
+ context : vscode.ExtensionContext
+)
+{
+ <>
+
+ <>
+ <>
+ context.subscriptions.push(
+ vscode.workspace.onDidChangeWorkspaceFolders(
+ async (e : vscode.WorkspaceFoldersChangeEvent) =>
+ {
+ for(const addedWorkspaceFolder of e.added) {
+ await this.processLiterateFiles(addedWorkspaceFolder);
+ }
+ for(const removedWorkspaceFolder of e.removed)
+ {
+ this.fragmentsForWorkspaceFolders.delete(removedWorkspaceFolder.name);
+ this.grabbedStateForWorkspaceFolders.delete(removedWorkspaceFolder.name);
+ }
+ }
+ )
+ );
+}
+```
+
+##### Initializing members
+
+``` ts : <>=
+this.md = createMarkdownItParserForLiterate();
+this.fragmentsForWorkspaceFolders = new Map();
+this.grabbedStateForWorkspaceFolders = new Map();
+this.diagnostics = vscode.languages.createDiagnosticCollection('literate');
+context.subscriptions.push(this.diagnostics);
+```
+
+##### Subscribing to text document changes
+
+The repository subscribes to the `onDidChangeTextDocument` event on the
+workspace. It could process **literate** files on each change, but the
+completion item provider needs to trigger itself processing of literate files.
+Since completion item provider gets called on typing a opening chevron (`<`) we
+skip triggering the processing here when such a character has been typed.
+
+``` ts : <>=
+context.subscriptions.push(
+ vscode.workspace.onDidChangeTextDocument(
+ async (e : vscode.TextDocumentChangeEvent) =>
+ {
+ if(!(e.contentChanges.length>0 && e.contentChanges[0].text.startsWith('<')))
+ {
+ await this.processLiterateFiles(e.document);
+ }
+ }
+ )
+);
+```
+
+##### Subscribing to workspace changes
+
+Triggering of processing **literate** documents is necessary when new workspace
+folders have been added. Additionally we need to clean up fragment maps and
+grabbed states for those workspace folders that have been removed from the
+workspace folder.
+
+``` ts : <>=
+context.subscriptions.push(
+ vscode.workspace.onDidChangeWorkspaceFolders(
+ async (e : vscode.WorkspaceFoldersChangeEvent) =>
+ {
+ for(const addedWorkspaceFolder of e.added) {
+ await this.processLiterateFiles(addedWorkspaceFolder);
+ }
+ for(const removedWorkspaceFolder of e.removed)
+ {
+ this.fragmentsForWorkspaceFolders.delete(removedWorkspaceFolder.name);
+ this.grabbedStateForWorkspaceFolders.delete(removedWorkspaceFolder.name);
+ }
+ }
+ )
+);
+```
+
+#### Processing literate files
+
+The parsing and setting up of the `fragments` map is handled with the method
+`processLiterateFiles`. Additionally the method will write out all specified
+source files.
+
+Processing the literate files is started generally in one of three cases: 1) change
+in workspace due to addition or removal of a workspace folder, 2) change to a
+literate document or through triggering of the `literate.process` command.
+
+``` ts : <>=
+async processLiterateFiles(
+ trigger :
+ vscode.WorkspaceFolder
+ | vscode.TextDocument
+ | undefined) {
+ <>
+ <>
+}
+```
+
+First we determine the workspace folder or workspace folders to process. In the
+case where `trigger` is a workspace folder or a text document we use the given
+workspace folder or determine the one to which the text document belongs. In
+these cases we'll have an array with just the one workspace folder as element.
+When the trigger is `undefined` we'll use all workspace folders registered to
+this workspace.
+
+``` ts : <>=
+const workspaceFolders : Array | undefined = (() => {
+ if(trigger)
+ {
+ <>
+ <>
+ if("eol" in trigger) {
+ const ws = determineWorkspaceFolder(trigger);
+ if(ws)
+ {
+ return [ws];
+ }
+ } else {
+ return [trigger];
+ }
+ }
+ if(vscode.workspace.workspaceFolders && vscode.workspace.workspaceFolders.length>0) {
+ let folders = new Array();
+ for(const ws of vscode.workspace.workspaceFolders)
+ {
+ folders.push(ws);
+ }
+ return folders;
+ }
+ return undefined;
+}
+)();
+```
+
+We can check if our `trigger` is a `TextDocument` to see if `eol` is a property.
+Otherwise it is a `Workspace`.
+
+``` ts : <>=
+if("eol" in trigger) {
+ const ws = determineWorkspaceFolder(trigger);
+ if(ws)
+ {
+ return [ws];
+ }
+}
+```
+
+``` ts : <>=
+else
+{
+ return [trigger];
+}
+```
+
+``` ts : <>=
+if(workspaceFolders) {
+ const writeOutHtml : WriteRenderCallback =
+ (fname : string,
+ folderUri : vscode.Uri,
+ rendered : string) : Thenable => {
+ const html =
+`
+
+
+
+
+ ${rendered}
+
+`;
+ const encoded = Buffer.from(html, 'utf-8');
+ fname = fname.replace(".literate", ".html");
+ const fileUri = vscode.Uri.joinPath(folderUri, fname);
+ return Promise.resolve(vscode.workspace.fs.writeFile(fileUri, encoded));
+ };
+ for(const folder of workspaceFolders)
+ {
+ if(!this.fragmentsForWorkspaceFolders.has(folder.name))
+ {
+ this.fragmentsForWorkspaceFolders.set(folder.name, new FragmentMap());
+ }
+ if(!this.grabbedStateForWorkspaceFolders.has(folder.name))
+ {
+ this.grabbedStateForWorkspaceFolders.set(folder.name, new GrabbedStateList());
+ }
+ const fragments = this.fragmentsForWorkspaceFolders.get(folder.name);
+ const grabbedStateList = this.grabbedStateForWorkspaceFolders.get(folder.name);
+ if(fragments && grabbedStateList) {
+ fragments.clear();
+ grabbedStateList.clear();
+ await iterateLiterateFiles(folder,
+ writeOutHtml, /* writeHtml : WriteRenderCallback*/
+ grabbedStateList.list,
+ this.md);
+ this.diagnostics.clear();
+ fragments.map = await handleFragments(folder, grabbedStateList.list, this.diagnostics, false, undefined);
+ this.diagnostics.clear();
+ await handleFragments(folder, grabbedStateList.list, this.diagnostics, true, writeSourceFiles);
+ }
+ }
+}
+```
+
+#### Fetching fragments for workspace folder
+
+When we call `getFragments` we assume the **literate** projects have all been
+process properly. In most cases that is triggered automatically, but it may be
+necessary to trigger the processing manually before calling `getFragments`. When
+the projects have been properly processed, though, this function returns the
+`FragmentMap` for the given workspace folder.
+
+``` ts : <>=
+getFragments(workspaceFolder : vscode.WorkspaceFolder) : FragmentMap
+{
+ let fragmentMap : FragmentMap = new FragmentMap();
+ this.fragmentsForWorkspaceFolders.forEach(
+ (value, key, _) =>
+ {
+ if(key===workspaceFolder.name)
+ {
+ fragmentMap = value;
+ }
+ }
+ );
+
+ return fragmentMap;
+}
+```
+
## Iterating all literate files
As mentioned in the introduction the main idea of the extension is to collect
@@ -164,26 +540,15 @@ are passed to a *MarkdownIt* renderer. The renderer will have the
states of each rendered file. The grabbed state is collected in `gstate`, which
is an instance of the `StateCore`, provided by *MarkdownIt*.
+The interface defines `literateFileName`, which is the filename of the
+**literate** document to which the grabbed state belongs. `literateUri` is the
+full uri for this document. Finally `gstate` holds the `StateCore` of the
+parsing result.
+
``` ts : <>=
-/**
- * Interface for environment to hold the Markdown file name and the StateCore
- * grabbed by the grabberPlugin.
- * The gstate we use to access all the tokens generated by the MarkdownIt parser.
- *
- * @see StateCore
- */
interface GrabbedState {
- /**
- * File name of the Markdown document to which the state belongs.
- */
literateFileName: string;
- /**
- * Uri for the Markdown document.
- */
literateUri: vscode.Uri;
- /**
- * State grabbed from the MarkdownIt parser.
- */
gstate: StateCore;
}
```
@@ -193,9 +558,6 @@ interface GrabbedState {
In the `iterateLiterateFiles` we start by setting up the *MarkdownIt* parser.
``` ts : <>=
-/**
- * MarkdownIt instance with grabber_plugin in use.
- */
const md : MarkdownIt = createMarkdownItParserForLiterate();
```
@@ -261,7 +623,7 @@ of fragments in code we use `FRAGMENT_USE_IN_CODE_RE`.
``` ts : <>=
//let HTML_ENCODED_FRAGMENT_TAG_RE = /(<<.*?>>)/g;
let FRAGMENT_USE_IN_CODE_RE =
- /(?[ \t]*)<<(?.*)>>(?=)?(?\+)?/g;
+ /(?[ \t]*)<<(?.+)>>(?=)?(?\+)?/g;
```
The regular expression captures four groups. A match will give us 5 or more
@@ -289,7 +651,7 @@ following the language specifier.
``` ts : <>=+
let FRAGMENT_RE =
- /(?.*):.*<<(?.*)>>(?=)?(?\+)?\s*(?.*)/;
+ /(?.*):.*<<(?.+)>>(?=)?(?\+)?\s*(?.*)/;
```
Most of the groups correspond to the ones defined by `FRAGMENT_USE_IN_CODE_RE`
@@ -363,17 +725,10 @@ which is of type `Map`. The name of a fragment will
function as the key, and an instance of `FragmentInformation` will be the value.
```ts : <>=
-/**
- * Map of fragment names and tuples of code fragments for these. The
- * tuples contain code language identifier followed by the filename and
- * lastly followed by the actual code fragment.
- */
const fragments = new Map();
-// Now we have the state, we have access to the tokens
-// over which we can iterate to extract all the code
-// fragments and build up the map with the fragments concatenated
-// where necessary. We'll extrapolate all fragments in the second
-// pass.
+const overwriteAttempts = new Array();
+const missingFilenames = new Array();
+const addingToNonExistant = new Array();
for (let env of envList) {
for (let token of env.gstate.tokens) {
<>
@@ -424,15 +779,17 @@ added to the `fragments` map.
``` ts : <>=
if (root && !add) {
- if (fragments.has(name)) {
+ if (fragments.has(name) && !overwriteAttempts.includes(name)) {
let msg = `Trying to overwrite existing fragment fragment ${name}. ${env.literateFileName}${linenumber}`;
const diag = createErrorDiagnostic(token, msg);
updateDiagnostics(env.literateUri, diagnostics, diag);
+ overwriteAttempts.push(name);
} else {
- if (!fileName && name.indexOf(".*") > -1) {
+ if (!fileName && name.indexOf(".*") > -1 && !missingFilenames.includes(name)) {
let msg = `Expected filename for star fragment ${name}`;
const diag = createErrorDiagnostic(token, msg);
updateDiagnostics(env.literateUri, diagnostics, diag);
+ missingFilenames.push(name);
} else {
let code = token.content;
let fragmentInfo: FragmentInformation = {
@@ -475,9 +832,12 @@ if (root && add) {
fragments.set(name, fragmentInfo);
}
} else {
- let msg = `Trying to add to non-existant fragment ${name}. ${env.literateFileName}:${linenumber}`;
- const diag = createErrorDiagnostic(token, msg);
- updateDiagnostics(env.literateUri, diagnostics, diag);
+ if(!addingToNonExistant.includes(name)) {
+ let msg = `Trying to add to non-existant fragment ${name}. ${env.literateFileName}:${linenumber}`;
+ const diag = createErrorDiagnostic(token, msg);
+ updateDiagnostics(env.literateUri, diagnostics, diag);
+ addingToNonExistant.push(name);
+ }
}
}
```
@@ -538,6 +898,9 @@ fragments can be combined into source code.
``` ts : <>=
// for now do several passes
let pass: number = 0;
+const rootIncorrect = new Array();
+const addIncorrect = new Array();
+const fragmentNotFound = new Array();
do {
pass++;
let fragmentReplaced = false;
@@ -557,20 +920,23 @@ do {
let tagName = match.groups.tagName;
let root = match.groups.root;
let add = match.groups.add;
- if (root) {
+ if (root && !rootIncorrect.includes(tag)) {
let msg = `Found '=': incorrect fragment tag in fragment, ${tag}`;
const diag = createErrorDiagnostic(fragmentInfo.tokens[0], msg);
updateDiagnostics(fragmentInfo.env.literateUri, diagnostics, diag);
+ rootIncorrect.push(tag);
}
- if (add) {
+ if (add && !addIncorrect.includes(tag)) {
let msg = `Found '+': incorrect fragment tag in fragment: ${tag}`;
const diag = createErrorDiagnostic(fragmentInfo.tokens[0], msg);
updateDiagnostics(fragmentInfo.env.literateUri, diagnostics, diag);
+ addIncorrect.push(tag);
}
- if (!fragments.has(match.groups.tagName) && tagName !== "(?.*)") {
+ if (!fragments.has(match.groups.tagName) && tagName !== "(?.+)" && !fragmentNotFound.includes(tagName)) {
let msg = `Could not find fragment ${tag} (${tagName})`;
const diag = createErrorDiagnostic(fragmentInfo.tokens[0], msg);
updateDiagnostics(fragmentInfo.env.literateUri, diagnostics, diag);
+ fragmentNotFound.push(tagName);
}
let fragmentToReplaceWith = fragments.get(tagName) || undefined;
if (fragmentToReplaceWith) {
@@ -648,8 +1014,7 @@ ${rendered}
The command `literate.process` is registered with Visual Studio Code. The
disposable that gets returned by `registerCommand` is held in
-`literateProcessDisposable` so that it can be used later on, for instance for
-diagnostics management.
+`literateProcessDisposable` so that we can push it into `context.subscriptions`.
Here we find the main program of our `literate.process` command. Our
*MarkdownIt* is set up, `.literate` files are searched and iterated. Each
@@ -668,704 +1033,11 @@ task. That is obviously not good for the workflow.
let literateProcessDisposable = vscode.commands.registerCommand(
'literate.process',
async function () {
-
- <>
-
- diagnostics.clear();
-
- if (!vscode.workspace.workspaceFolders) {
- return vscode.window.showInformationMessage("No workspace or folder opened");
- }
-
-
- const writeOutHtml : WriteRenderCallback =
- (fname : string,
- folderUri : vscode.Uri,
- rendered : string) : Thenable => {
- const html =
-`
-
-
-
-
- ${rendered}
-
-`;
- const encoded = Buffer.from(html, 'utf-8');
- fname = fname.replace(".literate", ".html");
- const fileUri = vscode.Uri.joinPath(folderUri, fname);
- return Promise.resolve(vscode.workspace.fs.writeFile(fileUri, encoded));
- };
-
- for(const workspaceFolder of vscode.workspace.workspaceFolders) {
- const envList: Array = new Array();
- await iterateLiterateFiles(workspaceFolder, writeOutHtml, envList, md);
- let _ = await handleFragments(workspaceFolder, envList, diagnostics, true, writeSourceFiles);
- }
-
- let hasAnyDiagnostics = false;
- diagnostics.forEach(
- function(
- _: vscode.Uri,
- diags: readonly vscode.Diagnostic[],
- __: vscode.DiagnosticCollection
- ) : any {
- hasAnyDiagnostics ||= (diags.length > 0);
- }
- );
-
- if (hasAnyDiagnostics) {
- return vscode.window.setStatusBarMessage(
- (new vscode.MarkdownString(
- "$(error) Error encountered during process"
- )).value, 2000);
- }
- else {
+ theOneRepository.processLiterateFiles(undefined);
return vscode.window.setStatusBarMessage("Literate Process completed", 5000);
- }
});
```
-## Fragment explorer
-
-The Literate Fragment Explorer is a `TreeView` that uses `FragmentNodeProvider`
-to show fragments available in a workspace. The tree view has `FragmentNode` as
-its type parameter.
-
-``` ts : <>=
-export class FragmentExplorer {
- private fragmentView : vscode.TreeView;
- constructor(context : vscode.ExtensionContext) {
- const fragmentNodeProvider = new FragmentNodeProvider();
- context.subscriptions.push(
- vscode.window.registerTreeDataProvider(
- 'fragmentExplorer',
- fragmentNodeProvider
- )
- );
- this.fragmentView = vscode.window.createTreeView(
- 'fragmentExplorer',
- {
- treeDataProvider : fragmentNodeProvider
- });
-
- context.subscriptions.push(
- vscode.commands.registerCommand(
- 'fragmentExplorer.refreshEntry',
- () => fragmentNodeProvider.refresh())
- );
- context.subscriptions.push(vscode.workspace.onDidChangeTextDocument(
- _ => {
- fragmentNodeProvider.refresh();
- }
- ));
- context.subscriptions.push(this.fragmentView);
- }
-}
-```
-
-### Fragment tree provider
-
-The Literate Fragment Explorer needs a
-[`TreeDataProvider`](https://code.visualstudio.com/api/extension-guides/tree-view)
-implementation to present the fragment structure to Visual Studio Code so that
-the data can be visualized in the `fragmentExplorer` custom view.
-
-The class `FragmentNodeProvider` implements a `TreeDataProvider` with
-`FragmentNode` as the tree item.
-
-``` ts : <>=
-export class FragmentNodeProvider implements vscode.TreeDataProvider
-{
- <>
- <>
-}
-```
-
-The constructor takes care of all necessary initialization.
-
-``` ts : <>=
-constructor()
-{
- <>
-}
-```
-
-The constructor for the `FragmentNodeProvider` creates an instance of the
-`MarkdownIt` module, fully configured for our **literate programming** needs.
-Additionally a `DiagnosticCollection` is created so that it can be passed on to
-the `handleFragments` function that is utilized in the `FragmentNodeProvider`.
-
-``` ts : <>=
-this.md = createMarkdownItParserForLiterate();
-this.diagnostics = vscode.languages.createDiagnosticCollection('literate-treeview');
-```
-
-This means we need two members to hold these instances.
-
-``` ts : <>=
-private md : MarkdownIt;
-private diagnostics : vscode.DiagnosticCollection;
-```
-
-The API for `FragmentNodeProvider` gives as method to update the tree view
-
-``` ts : <>=+
-refresh(): void {
- <>
-}
-```
-
-The current implementation simply fires the `onDidChangeTreeData` event but
-could do more work if needed. To that end there is a private member for emitting
-the event, and the actual event to which the event emitter is published.
-
-``` ts : <>=+
-private _onDidChangeTreeData:
- vscode.EventEmitter<
- FragmentNode |
- undefined |
- void
- > = new vscode.EventEmitter();
-readonly onDidChangeTreeData :
- vscode.Event<
- FragmentNode |
- undefined |
- void
- > = this._onDidChangeTreeData.event;
-```
-
-With those two in place the `refresh` function can fire the event whenever
-called.
-
-``` ts : <>=
-this._onDidChangeTreeData.fire();
-```
-
-The `TreeDataProvider` implementation provided by `FragmentNodeProvider` is
-completed by `getTreeItem` and `getChildren`. The first one is simple, it just
-returns the element that is passed to it, as there is no need to find out more
-information about this. Instead, elements have been already created by the
-`getChildren` function, where all `FragmentNode` instances are created with all
-the data necessary.
-
-``` ts : <>=+
-getTreeItem(element : FragmentNode): vscode.TreeItem {
- <>
-}
-```
-
-As said, the `getTreeItem` implementation remains simple
-
-``` ts : <>=
-return element;
-```
-
-On the other hand the `getChildren` function is more involved. Yet its job is
-simple: get all `FragmentNode`s that represent the direct children of the
-element given.
-
-``` ts : <>=+
-async getChildren(element? : FragmentNode): Promise
-{
- <>
-}
-```
-
-When the workspace has no workspace folders at all there will be no children to
-return, as there are no **literate** documents to begin with.
-
-``` ts : <>=
-if(!vscode.workspace.workspaceFolders ||
- (
- vscode.workspace.workspaceFolders &&
- vscode.workspace.workspaceFolders.length < 1
- )) {
- vscode.window.showInformationMessage('No fragments in empty workspace');
- return Promise.resolve([]);
-}
-```
-
-If we do have workspace folders, but no element is given to look for children we
-need to look at the all the fragments available in all documents across all
-workspace folders. If on the other hand an element is given then its children
-are retrieved.
-
-``` ts : <>=+
-if(!element)
-{
- <>
-}
-else
-{
- <>
-}
-```
-
-When no element is passed we want the root of all the branches, where each
-workspace folder is the root of its own branch.
-
-To this end the children are all essentially the workspace folder names. Since
-these are the work folders the fragments representing them have no `parentName`
-specified. As `folderName` we pass on the workspace folder name. This is a
-property all its children and the rest of its offspring inherit. The
-`folderName` is used to find the correct workspace folder to search for the
-given element and its offspring.
-
-``` ts : <>=
-let arr = new Array();
-for(const wsFolder of vscode.workspace.workspaceFolders)
-{
- arr.push(
- new FragmentNode(
- wsFolder.name,
- new vscode.MarkdownString('$(book) (workspace folder)', true),
- 'Workspace folder containing a literate project',
- vscode.TreeItemCollapsibleState.Collapsed,
- wsFolder.name,
- undefined,
- wsFolder,
- undefined));
-}
-return Promise.resolve(arr);
-```
-
-Getting the children for a given element is a bit more involved. First we set
-up a constant `folderName` for ease of access. Then we also creat an array of
-`FragmentNode`s.
-
-``` ts : <>=
-const folderName : string = element.folderName;
-const fldr : vscode.WorkspaceFolder = element.workspaceFolder;
-let arr = new Array();
-```
-
-From the element we already learned the workspace folder for its project, so we
-can use that directly to parse the **literate** content. With the `fragments`
-map of the workspace folder in hand we can iterate over the keys in the
-`fragments` map.
-
-There are essentially two cases we need to check for. If the given element has
-no `parentName` set we know it is a fragment in the document level, so a
-fragment that was created. In contrast for a fragment there are child fragments,
-meaning that in the fragment code block other fragments were used. These are
-presented in the tree view as children to that fragment.
-
-``` ts : <>=+
-<>
-for(const fragmentName of fragments.keys() )
-{
- if(!element.parentName) {
- <>
- }
- else if (fragmentName === element.label) {
- <>
- }
-}
-
-return Promise.resolve(arr);
-```
-
-### Getting all fragments
-
-To find the fragment information to build `FragmentNode`s from iterate over the
-**literate** files in the workspace folder that we determined we need to search.
-Then build the fragment map based on the tokens generated by the iteration pass.
-As a reminder the fragments map has the fragment name as key and the
-corresponding `FragmentInformation` as the value to that key.
-
-``` ts : <>=
-let envList: Array = new Array();
-await iterateLiterateFiles(fldr, undefined, envList, this.md);
-const fragments = await handleFragments(fldr, envList, this.diagnostics, false, undefined);
-```
-
-### TODO: build proper fragment hierarchy from fragments map
-
-Still to do. Right now essentially the map structure is shown, but that isn't
-very useful. What we really need is a hierarchical form with each fragment under
-its parent fragment so that the structure of the literate program can be seen.
-
-Another improvement we could make is to show Markdown outline of chapters, with
-fragment occurance under that shown.
-
-### Fragment used in other fragment
-
-When we have found the fragment the passed in element represents we can find the
-child fragment names, that is the fragment names used in this fragment. All
-matches against `FRAGMENT_USE_IN_CODE_RE` are found and for each case a
-corresponding `FragmentNode` is created to function as a child to our parent
-element.
-
-``` ts : <>=
-let fragmentInfo = fragments.get(fragmentName) || undefined;
-if (fragmentInfo) {
- const casesToReplace = [...fragmentInfo.code.matchAll(FRAGMENT_USE_IN_CODE_RE)];
- for (let match of casesToReplace) {
- if(!match || !match.groups)
- {
- continue;
- }
- let tag = match[0];
- let ident = match.groups.ident;
- let tagName = match.groups.tagName;
- let root = match.groups.root;
- let add = match.groups.add;
- arr.push(
- new FragmentNode(
- tagName,
- new vscode.MarkdownString(`$(symbol-file) ${fragmentInfo.literateFileName}`, true),
- fragmentName,
- vscode.TreeItemCollapsibleState.Collapsed,
- folderName,
- element.label,
- element.workspaceFolder,
- undefined
- )
- );
- }
-}
-```
-
-### Fragment on document level
-
-When the workspace folder is given as the element, or rather the `parentName`
-of the given element is undefined, we have a fragment on document level. There
-are two types of fragments we want to discern beetween: top level fragments, or
-fragments that also tell us what file to create, and other fragments. A
-**literate** document can contain multiple top level fragments. But each top
-level fragment will generate only one source code file.
-
-``` ts : <>=
-let fragmentType : vscode.MarkdownString;
-let fragmentInfo = fragments.get(fragmentName) || undefined;
-if (fragmentInfo) {
- if(fragmentName.indexOf(".*") >= 0)
- {
- fragmentType = new vscode.MarkdownString(
- `$(globe): ${fragmentInfo.literateFileName}`,
- true);
- }
- else
- {
- fragmentType = new vscode.MarkdownString(
- `$(code): ${fragmentInfo.literateFileName}`,
- true);
- }
- arr.push(
- new FragmentNode(
- fragmentName,
- fragmentType,
- fragmentInfo.literateFileName,
- vscode.TreeItemCollapsibleState.Collapsed,
- folderName,
- element.label,
- element.workspaceFolder,
- undefined));
-}
-```
-
-### Fragment node for tree view
-
-A fragment node represents a **literate** project fragment in a Visual Studio
-Code tree view. The class `FragmentNode` extends the `vscode.TreeItem`. Apart
-from just showing basic information like the fragment name and the file it is
-defined in we use `FragmentNode` also to keep track of the workspace folder it
-is hosted in as well as the text document if there is one. Text documents are
-documents the workspace currently has opened. We need to take these into
-account so that we can directly use these as part of the **literate** document
-parsing.
-
-``` ts : <>=
-class FragmentNode extends vscode.TreeItem
-{
- constructor (
- <>
- )
- {
- <>
- }
-}
-```
-
-For the visualization part we need a `label`, a `tooltip`, a `description` and a
-`collapsibleState`. These are the only pieces of information needed that show up
-in the tree view.
-
-``` ts : <>=
-public readonly label : string,
-public readonly tooltip : vscode.MarkdownString,
-public readonly description : string,
-public readonly collapsibleState : vscode.TreeItemCollapsibleState,
-```
-
-We further encode some more information in `FragmentNode` so that subsequent
-parsing can be done much more efficiently.
-
-``` ts : <>=+
-public readonly folderName: string,
-public readonly parentName : string | undefined,
-public readonly workspaceFolder : vscode.WorkspaceFolder,
-public readonly textDocument : vscode.TextDocument | undefined
-```
-
-Each node in the tree view represents a fragment. When the tree item is used to
-denote a workspace folder the theme icon for `'book'` is used. Actual fragments
-get the theme icon for `'code'`.
-
-``` ts : <>=
-super(label, collapsibleState);
-this.tooltip = tooltip;
-this.description = description;
-this.iconPath = this.parentName ?
- new vscode.ThemeIcon('code')
- : new vscode.ThemeIcon('book');
-this.contextValue = 'literate_fragment';
-```
-
-### registering FragmentNodeProvider
-
-The `FragmentNodeProvide` needs to be registered with Visual Studio Code so it
-can work when literate files are found in a work space.
-
-``` ts : <>=
-new FragmentExplorer(context);
-```
-
-## Code completion
-
-A simple implementation to provide code completion will help authors writing
-their literate programs. Having possible tag names suggested will help
-decreasing the cognitive load of remembering all code fragment names in a
-literate project. This project itself has well over 50 fragments, and having to
-remember them by name is not easy.
-
-Until there is a good **literate** file type integration with Visual Studio Code
-we'll be relying on the built-in **Markdown** functionality.
-
-``` ts : <>=
-const completionItemProvider =
- vscode.languages.registerCompletionItemProvider('markdown', {
- <>
-}, '<');
-context.subscriptions.push(completionItemProvider);
-```
-
-### Providing completion items
-
-The completion item provider will generate a `CompletionItem` for each fragment
-we currently know of. Although the provider gets passed in the `TextDocument`
-for which it was triggered we will present fragments from the entire project.
-
-``` ts : <>=
-async provideCompletionItems(
- document : vscode.TextDocument,
- ..._
-)
-{
-```
-
-After setting up the necessary variables with
-`<>` we figure out to which
-workspace folder the current `TextDocument`. If no workspace folder can be
-determined we return an empty array. This can happen with an unsaved new file,
-or when documents were opened that are not part of the workspace.
-
-``` ts : <>=+
- <>
- <>
-```
-
-After the workspace folder has been determined we can gather all fragments in
-our project.
-
-``` ts : <>=+
- <>
-```
-
-Finally we generate the completion items into the array `completionItems` that
-we return when done.
-
-``` ts : <>=+
- <>
- return completionItems;
-}
-```
-
-#### Setting up variables
-
-Completion items are going to be collected in an `Array`.
-Further, creating completion items for code completion needs to parse the
-entire project, so we need an `Array`. Iterating and parsing
-through the project also needs a `DiagnosticCollection`, although we won't be
-using it any further. Lastly we create an instance of the *MarkdownIt* parser to
-give to `iterateLiterateFiles`.
-
-``` ts : <>=
-let completionItems : Array =
- new Array();
-let envForCompletion : Array = new Array();
- new Array();
-const diagnostics = vscode.languages.createDiagnosticCollection('literate-completionitems');
-const md : MarkdownIt = createMarkdownItParserForLiterate();
-```
-
-#### Workspace folder for TextDocument
-
-Determining the workspace folder for the given TextDocument is done by creating
-relative paths from each workspace folder to the document. If the path does not
-start with `..` we found the workspace folder where the document is from.
-
-If no workspace folders were found, or if the TextDocument did not have a
-workspace folder we essentially end up returning an empty array from the
-completion item provider.
-
-``` ts : <>=
-const workspaceFolder : vscode.WorkspaceFolder | undefined = determineWorkspaceFolder(document);
-if(!workspaceFolder) { return []; }
-```
-
-#### Retrieving fragments of project
-
-Getting the fragments for our project means we `iterateLiterateFiles`with the
-`envForCompletion` given, along with the workspace folder and the *MarkdownIt*
-parser. Once we have iterated over all files, and thus `envForCompletion` now
-contains all literate documents tokenized we can pass those to `handleFragments`
-so we can end up with a map of all fragments. We pass in `false` to the function
-to ensure fragments aren't extrapolated: we want to show the fragments as they
-are in code completion.
-
-``` ts : <>=
- await iterateLiterateFiles(workspaceFolder, undefined, envForCompletion, md);
- let fragments = await handleFragments(workspaceFolder, envForCompletion, diagnostics, false, writeSourceFiles);
-```
-
-#### Creating the CompletionItems
-
-With all fragments in the map we iterate over all the keys. For each key we
-fetch the corresponding `FragmentInformation`. Now we can create the
-`CompletionItem` with the `fragmentName` as its content.
-
-Further the fragment code is set to be the detail of the completion item. This
-will provide a tooltip with the code fragment readable, so that it is easy to
-understand what fragment is currently highlighted in the completion list.
-
-Finally the set the completion item kind to `Reference` so that we get a nice
-icon in the completion list pop-up.
-
-``` ts : <>=
- for(const fragmentName of fragments.keys())
- {
- const fragment : FragmentInformation | undefined = fragments.get(fragmentName);
- if(!fragment) {
- continue;
- }
- const fragmentCompletion = new vscode.CompletionItem(fragmentName);
- fragmentCompletion.detail = fragment.code;
- fragmentCompletion.kind = vscode.CompletionItemKind.Reference;
- completionItems.push(fragmentCompletion);
- }
-```
-
-## Hover elements
-
-In addition to code completion we can provide hover information. We want to see
-the implementation of fragments when hovering of fragment usages. That way code
-inspection can be easier done.
-
-We'll create `FragmentHoverProvider` which implements `HoverProvider`.
-
-``` ts : <>=
-class FragmentHoverProvider implements vscode.HoverProvider {
- <>
-}
-```
-
-The `FragmentHoverProvider` implements `provideHover`. This will create the
-`Hover` item if under the current cursor position there is a fragment, including
-its opening and closing double chevrons.
-
-``` ts : <>=
-public async provideHover(
- document : vscode.TextDocument,
- position : vscode.Position,
- _: vscode.CancellationToken
-)
-{
- <>
- <>
- <>
- return null;
-}
-```
-
-We get the current line of text from the document. We are going to look only for
-tags that are on one line. In the future it would be nice to add support for
-cases where mentioning a fragment in explaining text is split over several lines
-due to word wrapping, but with the current implementation we'll look only at
-those that are on one line.
-
-``` ts : <>=
-const currentLine = document.lineAt(position.line);
-```
-
-Next we need to know the the workspace folder for the given document so that we
-can query the correct project for the fragments. If no workspace folder was
-determined return `null`, as there is no literate project associated with the
-given document.
-
-``` ts : <>=
-const workspaceFolder : vscode.WorkspaceFolder | undefined = determineWorkspaceFolder(document);
-if(!workspaceFolder) { return null; }
-```
-
-Fragments are now available so we can see if we have a fragment under our
-cursor. If we do, and the fragment is not one that defines or appends to a
-fragment we know our cursor is over either fragment usage in a code fence or a
-fragment mention in explaining text. For this we can create a `Hover` with the
-code of the fragment as a `MarkdownString` in a code fence.
-
-If that is not the case our `provideHover` implementation will return `null`.
-
-``` ts : <>=
-const matchesOnLine = [...currentLine.text.matchAll(FRAGMENT_USE_IN_CODE_RE)];
-for(const match of matchesOnLine)
-{
- if(!match || !match.groups) {
- continue;
- }
- const foundIndex = currentLine.text.indexOf(match[0]);
- if(foundIndex>-1) {
- <>
- if(foundIndex <= position.character && position.character <= foundIndex + match[0].length && fragments.has(match.groups.tagName))
- {
- const startPosition = new vscode.Position(currentLine.lineNumber, foundIndex);
- const endPosition = new vscode.Position(currentLine.lineNumber, foundIndex + match[0].length);
- let range : vscode.Range = new vscode.Range(startPosition, endPosition);
- let fragment = fragments.get(match.groups.tagName) || undefined;
- if (fragment && !match.groups.root) {
- return new vscode.Hover(
- new vscode.MarkdownString(`~~~ ${fragment.lang}\n${fragment.code}\n~~~`, true),
- range);
- }
- }
- }
-}
-```
-
-With the workspace folder in hand we can iterate over all literate files in the
-workspace and get the fragments for the project. We don't want extrapolated
-fragments, we want to see them as they are with fragment usages intact.
-
-``` ts : <>=
-const diagnostics = vscode.languages.createDiagnosticCollection('literate-completionitems');
-const md : MarkdownIt = createMarkdownItParserForLiterate();
-let envForCompletion : Array = new Array();
- new Array();
-await iterateLiterateFiles(workspaceFolder, undefined, envForCompletion, md);
-let fragments = await handleFragments(workspaceFolder, envForCompletion, diagnostics, false, writeSourceFiles);
-```
-
## Diagnostics
``` ts : <>=
@@ -1518,6 +1190,7 @@ source file or source files as written in the **literate** program.
<>
<>
<>
+<>
```
Utility function to determine the workspace folder for a TextDocument
@@ -1603,19 +1276,24 @@ interface WriteSourceCallback {
### Extension activation
``` ts : <>=
-export function activate(context: vscode.ExtensionContext) {
+let theOneRepository : FragmentRepository;
+export async function activate(context: vscode.ExtensionContext) {
const rootPath = (vscode.workspace.workspaceFolders && (vscode.workspace.workspaceFolders.length > 0))
? vscode.workspace.workspaceFolders[0].uri.fsPath : undefined;
console.log('Ready to do some Literate Programming');
const diagnostics = vscode.languages.createDiagnosticCollection('literate');
+ theOneRepository = new FragmentRepository(context);
+ await theOneRepository.processLiterateFiles(undefined);
+ context.subscriptions.push(theOneRepository);
+
<>
<>
<>
context.subscriptions.push(
- vscode.languages.registerHoverProvider('markdown', new FragmentHoverProvider())
+ vscode.languages.registerHoverProvider('markdown', new FragmentHoverProvider(theOneRepository))
);
if (vscode.window.activeTextEditor) {
@@ -1628,11 +1306,6 @@ export function activate(context: vscode.ExtensionContext) {
}));
context.subscriptions.push(literateProcessDisposable);
- context.subscriptions.push(vscode.workspace.onDidChangeTextDocument(
- _ => {
- vscode.commands.executeCommand('literate.process');
- }
- ));
return {
extendMarkdownIt(md: any) {
diff --git a/package-lock.json b/package-lock.json
index da36c89..41357d0 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -1,12 +1,12 @@
{
"name": "literate",
- "version": "0.4.0",
+ "version": "0.4.1",
"lockfileVersion": 2,
"requires": true,
"packages": {
"": {
"name": "literate",
- "version": "0.4.0",
+ "version": "0.4.1",
"license": "MIT",
"dependencies": {
"highlight.js": "^11.0.1",
diff --git a/package.json b/package.json
index 2ae2d6d..5267660 100644
--- a/package.json
+++ b/package.json
@@ -7,7 +7,7 @@
"repository": {
"url": "https://github.com/jesterKing/literate"
},
- "version": "0.4.0",
+ "version": "0.4.1",
"engines": {
"vscode": "^1.63.2"
},
@@ -31,7 +31,7 @@
"onview:fragmentExplorer",
"onCommand:literate.process"
],
- "main": "./out/extension.js",
+ "main": "./out/main.js",
"contributes": {
"views": {
"explorer": [
diff --git a/src/extension.ts b/src/extension.ts
index 8d107ce..a1081f1 100644
--- a/src/extension.ts
+++ b/src/extension.ts
@@ -31,25 +31,9 @@ interface WriteSourceCallback {
) : Thenable
};
-/**
- * Interface for environment to hold the Markdown file name and the StateCore
- * grabbed by the grabberPlugin.
- * The gstate we use to access all the tokens generated by the MarkdownIt parser.
- *
- * @see StateCore
- */
interface GrabbedState {
- /**
- * File name of the Markdown document to which the state belongs.
- */
literateFileName: string;
- /**
- * Uri for the Markdown document.
- */
literateUri: vscode.Uri;
- /**
- * State grabbed from the MarkdownIt parser.
- */
gstate: StateCore;
}
/**
@@ -85,14 +69,14 @@ interface FragmentInformation {
//let HTML_ENCODED_FRAGMENT_TAG_RE = /(<<.*?>>)/g;
let FRAGMENT_USE_IN_CODE_RE =
- /(?[ \t]*)<<(?.*)>>(?=)?(?\+)?/g;
+ /(?[ \t]*)<<(?.+)>>(?=)?(?\+)?/g;
let FRAGMENT_RE =
- /(?.*):.*<<(?.*)>>(?=)?(?\+)?\s*(?.*)/;
+ /(?.*):.*<<(?.+)>>(?=)?(?\+)?\s*(?.*)/;
class FragmentNode extends vscode.TreeItem
{
constructor (
- public readonly label : string,
+ public readonly label : string,
public readonly tooltip : vscode.MarkdownString,
public readonly description : string,
public readonly collapsibleState : vscode.TreeItemCollapsibleState,
@@ -102,7 +86,7 @@ class FragmentNode extends vscode.TreeItem
public readonly textDocument : vscode.TextDocument | undefined
)
{
- super(label, collapsibleState);
+ super(label, collapsibleState);
this.tooltip = tooltip;
this.description = description;
this.iconPath = this.parentName ?
@@ -114,8 +98,6 @@ class FragmentNode extends vscode.TreeItem
export class FragmentNodeProvider implements vscode.TreeDataProvider
{
- private md : MarkdownIt;
- private diagnostics : vscode.DiagnosticCollection;
private _onDidChangeTreeData:
vscode.EventEmitter<
FragmentNode |
@@ -128,20 +110,15 @@ export class FragmentNodeProvider implements vscode.TreeDataProvider = this._onDidChangeTreeData.event;
- constructor()
- {
- this.md = createMarkdownItParserForLiterate();
- this.diagnostics = vscode.languages.createDiagnosticCollection('literate-treeview');
- }
refresh(): void {
- this._onDidChangeTreeData.fire();
+ this._onDidChangeTreeData.fire();
}
getTreeItem(element : FragmentNode): vscode.TreeItem {
- return element;
+ return element;
}
async getChildren(element? : FragmentNode): Promise
{
- if(!vscode.workspace.workspaceFolders ||
+ if(!vscode.workspace.workspaceFolders ||
(
vscode.workspace.workspaceFolders &&
vscode.workspace.workspaceFolders.length < 1
@@ -151,7 +128,7 @@ export class FragmentNodeProvider implements vscode.TreeDataProvider();
+ let arr = new Array();
for(const wsFolder of vscode.workspace.workspaceFolders)
{
arr.push(
@@ -169,16 +146,14 @@ export class FragmentNodeProvider implements vscode.TreeDataProvider();
- let envList: Array = new Array();
- await iterateLiterateFiles(fldr, undefined, envList, this.md);
- const fragments = await handleFragments(fldr, envList, this.diagnostics, false, undefined);
+ const fragments = theOneRepository.getFragments(fldr).map;
for(const fragmentName of fragments.keys() )
{
if(!element.parentName) {
- let fragmentType : vscode.MarkdownString;
+ let fragmentType : vscode.MarkdownString;
let fragmentInfo = fragments.get(fragmentName) || undefined;
if (fragmentInfo) {
if(fragmentName.indexOf(".*") >= 0)
@@ -206,7 +181,7 @@ export class FragmentNodeProvider implements vscode.TreeDataProvider-1) {
- const diagnostics = vscode.languages.createDiagnosticCollection('literate-completionitems');
- const md : MarkdownIt = createMarkdownItParserForLiterate();
- let envForCompletion : Array = new Array();
- new Array();
- await iterateLiterateFiles(workspaceFolder, undefined, envForCompletion, md);
- let fragments = await handleFragments(workspaceFolder, envForCompletion, diagnostics, false, writeSourceFiles);
+ let fragments = this.fragmentRepository.getFragments(workspaceFolder).map;
if(foundIndex <= position.character && position.character <= foundIndex + match[0].length && fragments.has(match.groups.tagName))
{
const startPosition = new vscode.Position(currentLine.lineNumber, foundIndex);
@@ -375,14 +350,14 @@ async function iterateLiterateFiles(workspaceFolder : vscode.WorkspaceFolder,
envList : Array,
md : MarkdownIt)
{
- const literateFilesInWorkspace : vscode.RelativePattern =
+ const literateFilesInWorkspace : vscode.RelativePattern =
new vscode.RelativePattern(workspaceFolder, '**/*.literate');
const foundLiterateFiles = await vscode.workspace
.findFiles(literateFilesInWorkspace)
.then(files => Promise.all(files.map(file => file)));
try {
for (let fl of foundLiterateFiles) {
- const currentContent = (() =>
+ const currentContent = (() =>
{
for(const textDocument of vscode.workspace.textDocuments) {
if(vscode.workspace.asRelativePath(fl) === vscode.workspace.asRelativePath(textDocument.uri)) {
@@ -392,13 +367,13 @@ async function iterateLiterateFiles(workspaceFolder : vscode.WorkspaceFolder,
return '';
}
)();
- const content = currentContent ? null : await vscode.workspace.fs.readFile(fl);
+ const content = currentContent ? null : await vscode.workspace.fs.readFile(fl);
const text = currentContent ? currentContent : new TextDecoder('utf-8').decode(content);
- const fname = path.relative(workspaceFolder.uri.path, fl.path);
+ const fname = path.relative(workspaceFolder.uri.path, fl.path);
const env: GrabbedState = { literateFileName: fname, literateUri: fl, gstate: new StateCore('', md, {}) };
envList.push(env);
const rendered = md.render(text, env);
- if(writeHtml)
+ if(writeHtml)
{
await writeHtml(fname, workspaceFolder.uri, rendered);
}
@@ -415,20 +390,13 @@ async function handleFragments(
writeSource : WriteSourceCallback | undefined) : Promise