Skip to content

Commit

Permalink
⤵️ Local download links
Browse files Browse the repository at this point in the history
See #158
  • Loading branch information
rowanc1 committed Jun 24, 2023
1 parent 277f4a0 commit 15ffc78
Show file tree
Hide file tree
Showing 4 changed files with 56 additions and 49 deletions.
2 changes: 1 addition & 1 deletion src/MySTMarkdownCell.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -210,7 +210,7 @@ export class MySTMarkdownCell
}

// The notebook update is asynchronous
renderNotebook(this.parent as StaticNotebook);
await renderNotebook(this.parent as StaticNotebook);

// Let's wait for this cell to be rendered
await this._mystWidget.renderPromise;
Expand Down
84 changes: 43 additions & 41 deletions src/links.tsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
import React from 'react';
import type { Plugin } from 'unified';
import type { Root, Link } from 'myst-spec';
import { selectAll } from 'unist-util-select';
import { URLExt } from '@jupyterlab/coreutils';
Expand All @@ -8,10 +7,9 @@ import { IRenderMime } from '@jupyterlab/rendermime';

/**
* Handle an anchor node.
* NOTE: This is copied from @jupyterlab/rendermime renderers.ts
* ideally this should be removed and exported from there?
* Originally from @jupyterlab/rendermime renderers.ts
*/
function handleAnchor(
async function handleAnchor(
anchor: HTMLAnchorElement,
resolver: IRenderMime.IResolver,
linkHandler: IRenderMime.ILinkHandler | undefined
Expand All @@ -24,41 +22,37 @@ function handleAnchor(
: URLExt.isLocal(href);
// Bail if it is not a file-like url.
if (!href || !isLocal) {
return Promise.resolve(undefined);
return;
}
// Remove the hash until we can handle it.
const hash = anchor.hash;
if (hash) {
// Handle internal link in the file.
if (hash === href) {
anchor.target = '_self';
return Promise.resolve(undefined);
return;
}
// For external links, remove the hash until we have hash handling.
href = href.replace(hash, '');
}
// Get the appropriate file path.
return resolver
.resolveUrl(href)
.then(urlPath => {
// decode encoded url from url to api path
const path = decodeURIComponent(urlPath);
// Handle the click override.
if (linkHandler) {
linkHandler.handleLink(anchor, path, hash);
}
// Get the appropriate file download path.
return resolver.getDownloadUrl(urlPath);
})
.then(url => {
// Set the visible anchor.
anchor.href = url + hash;
})
.catch(err => {
// If there was an error getting the url,
// just make it an empty link.
anchor.href = '';
});
try {
// Get the appropriate file path.
const urlPath = await resolver.resolveUrl(href);
// decode encoded url from url to api path
const path = decodeURIComponent(urlPath);
// Handle the click override.
if (linkHandler) {
linkHandler.handleLink(anchor, path, hash);
}
// Get the appropriate file download path.
const url = await resolver.getDownloadUrl(urlPath);
// Set the visible anchor.
anchor.href = url + hash;
} catch (error) {
// If there was an error getting the url,
// just make it an empty link.
anchor.href = '';
}
}

export const linkFactory =
Expand Down Expand Up @@ -92,17 +86,25 @@ export async function internalLinksTransform(
opts: Options
): Promise<void> {
const links = selectAll('link,linkBlock', tree) as Link[];
links.forEach(async link => {
if (!link || !link.url) return;
const resolver = opts.resolver;
const isLocal = resolver?.isLocal
? resolver.isLocal(link.url)
: URLExt.isLocal(link.url);
if (isLocal) (link as any).internal = true;
});
await Promise.all(
links.map(async link => {
if (!link || !link.url) return;
const resolver = opts.resolver;
const href = link.url;
const isLocal = resolver?.isLocal
? resolver.isLocal(href)
: URLExt.isLocal(href);
if (!isLocal) return;
if (!resolver) return;
if ((link as any).static) {
// TODO: remove hash
const urlPath = await resolver.resolveUrl(href);
const url = await resolver.getDownloadUrl(urlPath);
(link as any).urlSource = href;
link.url = url;
} else {
(link as any).internal = true;
}
})
);
}

export const internalLinksPlugin: Plugin<[Options], Root, Root> =
opts => tree => {
internalLinksTransform(tree, opts);
};
1 change: 1 addition & 0 deletions src/mime.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ export class RenderedMySTMarkdown
this.node.dataset['mimeType'] = MIME_TYPE;
this.addClass('jp-RenderedMySTMarkdown');
}

async renderModel(model: IRenderMime.IMimeModel): Promise<void> {
if ((window as any).trigger) {
throw Error('triggered!');
Expand Down
18 changes: 11 additions & 7 deletions src/myst.ts
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@ import { StaticNotebook } from '@jupyterlab/notebook';
import { IRenderMimeRegistry } from '@jupyterlab/rendermime';
import { IRenderMime } from '@jupyterlab/rendermime-interfaces';
import { imageUrlSourceTransform } from './images';
import { internalLinksPlugin } from './links';
import { internalLinksTransform } from './links';
import { addCiteChildrenPlugin } from './citations';
import { evalRole } from './roles';
import { IUserExpressionMetadata } from './metadata';
Expand Down Expand Up @@ -119,12 +119,12 @@ export async function processArticleMDAST(
.use(linksPlugin, { transformers: linkTransforms })
.use(footnotesPlugin)
.use(resolveReferencesPlugin, { state })
.use(internalLinksPlugin, { resolver })
.use(addCiteChildrenPlugin)
.use(keysPlugin)
.runSync(mdast as any, file);

// Go through all links and replace the source if they are local
await internalLinksTransform(mdast, { resolver });
await imageUrlSourceTransform(mdast, { resolver });

return {
Expand All @@ -143,10 +143,10 @@ export function buildNotebookMDAST(mystCells: IMySTMarkdownCell[]): any {
return { type: 'root', children: blocks };
}

export function processNotebookMDAST(
export async function processNotebookMDAST(
mdast: any,
resolver: IRenderMime.IResolver | undefined
): IMySTDocumentState {
): Promise<IMySTDocumentState> {
const linkTransforms = [
new WikiTransformer(),
new GithubTransformer(),
Expand Down Expand Up @@ -185,11 +185,12 @@ export function processNotebookMDAST(
.use(linksPlugin, { transformers: linkTransforms })
.use(footnotesPlugin)
.use(resolveReferencesPlugin, { state })
.use(internalLinksPlugin, { resolver: resolver })
.use(addCiteChildrenPlugin)
.use(keysPlugin)
.runSync(mdast as any, file);

await internalLinksTransform(mdast, { resolver });

if (file.messages.length > 0) {
// TODO: better error messages in the future
console.error(file.messages.map(m => m.message).join('\n'));
Expand All @@ -215,7 +216,7 @@ export async function processCellMDAST(
return mdast;
}

export function renderNotebook(notebook: StaticNotebook) {
export async function renderNotebook(notebook: StaticNotebook) {
const mystCells = notebook.widgets.filter(isMySTMarkdownCell).filter(
// In the future, we may want to process the code cells as well, but not now
cell => cell.fragmentMDAST !== undefined
Expand All @@ -225,7 +226,10 @@ export function renderNotebook(notebook: StaticNotebook) {
references,
frontmatter,
mdast: processedMDAST
} = processNotebookMDAST(mdast, notebook.rendermime.resolver ?? undefined);
} = await processNotebookMDAST(
mdast,
notebook.rendermime.resolver ?? undefined
);

mystCells.forEach((cell, index) => {
if (cell.rendered) {
Expand Down

0 comments on commit 15ffc78

Please sign in to comment.