@defra/forms-model 3.0.636 → 3.0.637
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
|
@@ -5,18 +5,49 @@ import { Marked, Renderer } from 'marked';
|
|
|
5
5
|
*/
|
|
6
6
|
export const marked = new Marked({
|
|
7
7
|
breaks: true,
|
|
8
|
-
gfm: true
|
|
8
|
+
gfm: true,
|
|
9
|
+
/**
|
|
10
|
+
* Inline extension to support {:target="_blank"} attribute syntax on links.
|
|
11
|
+
* Allows form designers to force any link to open in a new tab, e.g.
|
|
12
|
+
* [Open the service](https://example.gov.uk){:target="_blank"}
|
|
13
|
+
*/
|
|
14
|
+
extensions: [{
|
|
15
|
+
name: 'linkAttributes',
|
|
16
|
+
level: 'inline',
|
|
17
|
+
start(src) {
|
|
18
|
+
return src.indexOf('{:');
|
|
19
|
+
},
|
|
20
|
+
tokenizer(src, tokens) {
|
|
21
|
+
// Quotes are HTML-escaped before parsing, so " becomes "
|
|
22
|
+
const match = /^\{:target="_blank"\}/.exec(src);
|
|
23
|
+
if (match && tokens.length > 0) {
|
|
24
|
+
const last = tokens[tokens.length - 1];
|
|
25
|
+
if (last.type === 'link') {
|
|
26
|
+
;
|
|
27
|
+
last.forceNewTab = true;
|
|
28
|
+
return {
|
|
29
|
+
type: 'linkAttributes',
|
|
30
|
+
raw: match[0]
|
|
31
|
+
};
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
},
|
|
35
|
+
renderer() {
|
|
36
|
+
return '';
|
|
37
|
+
}
|
|
38
|
+
}]
|
|
9
39
|
});
|
|
10
|
-
function renderLink(href, text, baseUrl) {
|
|
40
|
+
function renderLink(href, text, baseUrl, forceNewTab = false) {
|
|
11
41
|
let isLocalLink = true;
|
|
12
42
|
if (baseUrl) {
|
|
13
43
|
isLocalLink = href.startsWith(baseUrl) || href.startsWith('mailto:');
|
|
14
44
|
}
|
|
45
|
+
const openInNewTab = !isLocalLink || forceNewTab;
|
|
15
46
|
const attrs = [`class="govuk-link"`, `href="${href}"`];
|
|
16
|
-
if (
|
|
47
|
+
if (openInNewTab) {
|
|
17
48
|
attrs.push(`target="_blank" rel="noreferrer noopener"`);
|
|
18
49
|
}
|
|
19
|
-
const label =
|
|
50
|
+
const label = openInNewTab ? `${text} (opens in new tab)` : text;
|
|
20
51
|
return `<a ${attrs.join(' ')}>${label}</a>`;
|
|
21
52
|
}
|
|
22
53
|
function demoteHeading(text, depth, startingHeaderLevel) {
|
|
@@ -35,11 +66,8 @@ export function markdownToHtml(markdown, options) {
|
|
|
35
66
|
}
|
|
36
67
|
const escaped = markdown.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>').replace(/"/g, '"').replace(/'/g, ''');
|
|
37
68
|
const renderer = new Renderer();
|
|
38
|
-
renderer.link =
|
|
39
|
-
href,
|
|
40
|
-
text
|
|
41
|
-
}) => {
|
|
42
|
-
return renderLink(href, text, options?.baseUrl);
|
|
69
|
+
renderer.link = token => {
|
|
70
|
+
return renderLink(token.href, token.text, options?.baseUrl, token.forceNewTab);
|
|
43
71
|
};
|
|
44
72
|
if (options?.startingHeaderLevel) {
|
|
45
73
|
renderer.heading = ({
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"markdown.js","names":["Marked","Renderer","marked","breaks","gfm","renderLink","href","text","baseUrl","isLocalLink","startsWith","attrs","push","label","join","demoteHeading","depth","startingHeaderLevel","Math","min","markdownToHtml","markdown","options","undefined","escaped","replace","
|
|
1
|
+
{"version":3,"file":"markdown.js","names":["Marked","Renderer","marked","breaks","gfm","extensions","name","level","start","src","indexOf","tokenizer","tokens","match","exec","length","last","type","forceNewTab","raw","renderer","renderLink","href","text","baseUrl","isLocalLink","startsWith","openInNewTab","attrs","push","label","join","demoteHeading","depth","startingHeaderLevel","Math","min","markdownToHtml","markdown","options","undefined","escaped","replace","link","token","heading","parse","async"],"sources":["../../../src/utils/markdown.ts"],"sourcesContent":["import { Marked, Renderer, type Token, type Tokens } from 'marked'\n\n/**\n * Marked instance (avoids global option/extension scope)\n */\nexport const marked = new Marked({\n breaks: true,\n gfm: true,\n\n /**\n * Inline extension to support {:target=\"_blank\"} attribute syntax on links.\n * Allows form designers to force any link to open in a new tab, e.g.\n * [Open the service](https://example.gov.uk){:target=\"_blank\"}\n */\n extensions: [\n {\n name: 'linkAttributes',\n level: 'inline',\n start(src: string) {\n return src.indexOf('{:')\n },\n tokenizer(src: string, tokens: Token[]) {\n // Quotes are HTML-escaped before parsing, so \" becomes "\n const match = /^\\{:target="_blank"\\}/.exec(src)\n if (match && tokens.length > 0) {\n const last = tokens[tokens.length - 1]\n if (last.type === 'link') {\n ;(last as Tokens.Link & { forceNewTab: boolean }).forceNewTab = true\n return {\n type: 'linkAttributes',\n raw: match[0]\n }\n }\n }\n },\n renderer() {\n return ''\n }\n }\n ]\n})\n\nfunction renderLink(\n href: string,\n text: string,\n baseUrl?: string,\n forceNewTab = false\n) {\n let isLocalLink = true\n\n if (baseUrl) {\n isLocalLink = href.startsWith(baseUrl) || href.startsWith('mailto:')\n }\n\n const openInNewTab = !isLocalLink || forceNewTab\n\n const attrs = [`class=\"govuk-link\"`, `href=\"${href}\"`]\n\n if (openInNewTab) {\n attrs.push(`target=\"_blank\" rel=\"noreferrer noopener\"`)\n }\n\n const label = openInNewTab ? `${text} (opens in new tab)` : text\n\n return `<a ${attrs.join(' ')}>${label}</a>`\n}\n\nfunction demoteHeading(\n text: string,\n depth: number,\n startingHeaderLevel: number\n) {\n // Max heading is h6 so don't demote further than that\n depth = Math.min(depth + startingHeaderLevel - 1, 6)\n return `<h${depth}>${text}</h${depth}>\n`\n}\n\n/**\n * Convert markdown to HTML, escaping any HTML tags first\n */\nexport function markdownToHtml(\n markdown: string | null | undefined,\n options?: {\n baseUrl?: string // optional in some contexts, e.g. from the designer where it might not make sense,\n startingHeaderLevel?: number\n }\n) {\n if (markdown === undefined || markdown === null) {\n return ''\n }\n\n const escaped = markdown\n .replace(/&/g, '&')\n .replace(/</g, '<')\n .replace(/>/g, '>')\n .replace(/\"/g, '"')\n .replace(/'/g, ''')\n\n const renderer = new Renderer()\n renderer.link = (token: Tokens.Link): string => {\n return renderLink(\n token.href,\n token.text,\n options?.baseUrl,\n (token as Tokens.Link & { forceNewTab?: boolean }).forceNewTab\n )\n }\n if (options?.startingHeaderLevel) {\n renderer.heading = ({ text, depth }: Tokens.Heading): string => {\n return demoteHeading(text, depth, options.startingHeaderLevel ?? 1)\n }\n }\n return marked.parse(escaped, { async: false, renderer })\n}\n"],"mappings":"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","ignoreList":[]}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"markdown.d.ts","sourceRoot":"","sources":["../../../src/utils/markdown.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,
|
|
1
|
+
{"version":3,"file":"markdown.d.ts","sourceRoot":"","sources":["../../../src/utils/markdown.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAqC,MAAM,QAAQ,CAAA;AAElE;;GAEG;AACH,eAAO,MAAM,MAAM,QAmCjB,CAAA;AAsCF;;GAEG;AACH,wBAAgB,cAAc,CAC5B,QAAQ,EAAE,MAAM,GAAG,IAAI,GAAG,SAAS,EACnC,OAAO,CAAC,EAAE;IACR,OAAO,CAAC,EAAE,MAAM,CAAA;IAChB,mBAAmB,CAAC,EAAE,MAAM,CAAA;CAC7B,UA4BF"}
|
package/package.json
CHANGED
package/src/utils/markdown.ts
CHANGED
|
@@ -1,27 +1,66 @@
|
|
|
1
|
-
import { Marked, Renderer, type Tokens } from 'marked'
|
|
1
|
+
import { Marked, Renderer, type Token, type Tokens } from 'marked'
|
|
2
2
|
|
|
3
3
|
/**
|
|
4
4
|
* Marked instance (avoids global option/extension scope)
|
|
5
5
|
*/
|
|
6
6
|
export const marked = new Marked({
|
|
7
7
|
breaks: true,
|
|
8
|
-
gfm: true
|
|
8
|
+
gfm: true,
|
|
9
|
+
|
|
10
|
+
/**
|
|
11
|
+
* Inline extension to support {:target="_blank"} attribute syntax on links.
|
|
12
|
+
* Allows form designers to force any link to open in a new tab, e.g.
|
|
13
|
+
* [Open the service](https://example.gov.uk){:target="_blank"}
|
|
14
|
+
*/
|
|
15
|
+
extensions: [
|
|
16
|
+
{
|
|
17
|
+
name: 'linkAttributes',
|
|
18
|
+
level: 'inline',
|
|
19
|
+
start(src: string) {
|
|
20
|
+
return src.indexOf('{:')
|
|
21
|
+
},
|
|
22
|
+
tokenizer(src: string, tokens: Token[]) {
|
|
23
|
+
// Quotes are HTML-escaped before parsing, so " becomes "
|
|
24
|
+
const match = /^\{:target="_blank"\}/.exec(src)
|
|
25
|
+
if (match && tokens.length > 0) {
|
|
26
|
+
const last = tokens[tokens.length - 1]
|
|
27
|
+
if (last.type === 'link') {
|
|
28
|
+
;(last as Tokens.Link & { forceNewTab: boolean }).forceNewTab = true
|
|
29
|
+
return {
|
|
30
|
+
type: 'linkAttributes',
|
|
31
|
+
raw: match[0]
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
},
|
|
36
|
+
renderer() {
|
|
37
|
+
return ''
|
|
38
|
+
}
|
|
39
|
+
}
|
|
40
|
+
]
|
|
9
41
|
})
|
|
10
42
|
|
|
11
|
-
function renderLink(
|
|
43
|
+
function renderLink(
|
|
44
|
+
href: string,
|
|
45
|
+
text: string,
|
|
46
|
+
baseUrl?: string,
|
|
47
|
+
forceNewTab = false
|
|
48
|
+
) {
|
|
12
49
|
let isLocalLink = true
|
|
13
50
|
|
|
14
51
|
if (baseUrl) {
|
|
15
52
|
isLocalLink = href.startsWith(baseUrl) || href.startsWith('mailto:')
|
|
16
53
|
}
|
|
17
54
|
|
|
55
|
+
const openInNewTab = !isLocalLink || forceNewTab
|
|
56
|
+
|
|
18
57
|
const attrs = [`class="govuk-link"`, `href="${href}"`]
|
|
19
58
|
|
|
20
|
-
if (
|
|
59
|
+
if (openInNewTab) {
|
|
21
60
|
attrs.push(`target="_blank" rel="noreferrer noopener"`)
|
|
22
61
|
}
|
|
23
62
|
|
|
24
|
-
const label =
|
|
63
|
+
const label = openInNewTab ? `${text} (opens in new tab)` : text
|
|
25
64
|
|
|
26
65
|
return `<a ${attrs.join(' ')}>${label}</a>`
|
|
27
66
|
}
|
|
@@ -59,8 +98,13 @@ export function markdownToHtml(
|
|
|
59
98
|
.replace(/'/g, ''')
|
|
60
99
|
|
|
61
100
|
const renderer = new Renderer()
|
|
62
|
-
renderer.link = (
|
|
63
|
-
return renderLink(
|
|
101
|
+
renderer.link = (token: Tokens.Link): string => {
|
|
102
|
+
return renderLink(
|
|
103
|
+
token.href,
|
|
104
|
+
token.text,
|
|
105
|
+
options?.baseUrl,
|
|
106
|
+
(token as Tokens.Link & { forceNewTab?: boolean }).forceNewTab
|
|
107
|
+
)
|
|
64
108
|
}
|
|
65
109
|
if (options?.startingHeaderLevel) {
|
|
66
110
|
renderer.heading = ({ text, depth }: Tokens.Heading): string => {
|