| 
									
										
										
										
											2021-02-15 19:41:40 +01:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Copyright (c) Microsoft Corporation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  |  * you may not use this file except in compliance with the License. | 
					
						
							|  |  |  |  * You may obtain a copy of the License at | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  |  * distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  |  * See the License for the specific language governing permissions and | 
					
						
							|  |  |  |  * limitations under the License. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // @ts-check
 | 
					
						
							|  |  |  | const Documentation = require('./documentation'); | 
					
						
							|  |  |  | const { visitAll } = require('../markdown'); | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @param {Documentation.MarkdownNode[]} nodes | 
					
						
							|  |  |  |  * @param {number} maxColumns | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function renderXmlDoc(nodes, maxColumns = 80, prefix = '/// ') { | 
					
						
							|  |  |  |   if (!nodes) | 
					
						
							|  |  |  |     return []; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const renderResult = _innerRenderNodes(nodes, maxColumns); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const doc = []; | 
					
						
							|  |  |  |   _wrapInNode('summary', renderResult.summary, doc); | 
					
						
							|  |  |  |   _wrapInNode('remarks', renderResult.remarks, doc); | 
					
						
							|  |  |  |   return doc.map(x => `${prefix}${x}`); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function _innerRenderNodes(nodes, maxColumns = 80, wrapParagraphs = true) { | 
					
						
							|  |  |  |   const summary = []; | 
					
						
							|  |  |  |   const remarks = []; | 
					
						
							|  |  |  |   const handleListItem = (lastNode, node) => { | 
					
						
							|  |  |  |     if (node && node.type === 'li' && (!lastNode || lastNode.type !== 'li')) | 
					
						
							|  |  |  |       summary.push(`<list type="${node.liType}">`); | 
					
						
							|  |  |  |     else if (lastNode && lastNode.type === 'li' && (!node || node.type !== 'li')) | 
					
						
							|  |  |  |       summary.push('</list>'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let lastNode; | 
					
						
							|  |  |  |   visitAll(nodes, node => { | 
					
						
							|  |  |  |     // handle special cases first
 | 
					
						
							|  |  |  |     if (_nodeShouldBeIgnored(node)) | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     if (node.text && node.text.startsWith('extends: ')) { | 
					
						
							|  |  |  |       remarks.push('Inherits from ' + node.text.replace('extends: ', '')); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     handleListItem(lastNode, node); | 
					
						
							|  |  |  |     if (node.type === 'text') { | 
					
						
							|  |  |  |       if (wrapParagraphs) | 
					
						
							|  |  |  |         _wrapInNode('para', _wrapAndEscape(node, maxColumns), summary); | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         summary.push(..._wrapAndEscape(node, maxColumns)); | 
					
						
							|  |  |  |     } else if (node.type === 'code' && node.codeLang === 'csharp') { | 
					
						
							| 
									
										
										
										
											2021-03-03 19:39:06 +01:00
										 |  |  |       _wrapInNode('code', _wrapCode(node.lines), summary); | 
					
						
							| 
									
										
										
										
											2021-02-15 19:41:40 +01:00
										 |  |  |     } else if (node.type === 'li') { | 
					
						
							|  |  |  |       _wrapInNode('item><description', _wrapAndEscape(node, maxColumns), summary, '/description></item'); | 
					
						
							|  |  |  |     } else if (node.type === 'note') { | 
					
						
							|  |  |  |       _wrapInNode('para', _wrapAndEscape(node, maxColumns), remarks); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     lastNode = node; | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  |   handleListItem(lastNode, null); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { summary, remarks }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-03 19:39:06 +01:00
										 |  |  | function _wrapCode(lines) { | 
					
						
							|  |  |  |   let i = 0; | 
					
						
							|  |  |  |   let out = []; | 
					
						
							|  |  |  |   for (let line of lines) { | 
					
						
							| 
									
										
										
										
											2021-05-16 09:58:40 -07:00
										 |  |  |     line = line.replace(/[&]/g, '&').replace(/</g, '<').replace(/>/g, '>'); | 
					
						
							| 
									
										
										
										
											2021-03-03 19:39:06 +01:00
										 |  |  |     if (i < lines.length - 1) | 
					
						
							|  |  |  |       line = line + "<br/>"; | 
					
						
							|  |  |  |     out.push(line); | 
					
						
							|  |  |  |     i++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return out; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-15 19:41:40 +01:00
										 |  |  | function _wrapInNode(tag, nodes, target, closingTag = null) { | 
					
						
							|  |  |  |   if (nodes.length === 0) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!closingTag) | 
					
						
							|  |  |  |     closingTag = `/${tag}`; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (nodes.length === 1) { | 
					
						
							|  |  |  |     target.push(`<${tag}>${nodes[0]}<${closingTag}>`); | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   target.push(`<${tag}>`); | 
					
						
							|  |  |  |   target.push(...nodes); | 
					
						
							|  |  |  |   target.push(`<${closingTag}>`); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {Documentation.MarkdownNode} node | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function _wrapAndEscape(node, maxColumns = 0) { | 
					
						
							|  |  |  |   const lines = []; | 
					
						
							|  |  |  |   const pushLine = text => { | 
					
						
							|  |  |  |     if (text === '') | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     text = text.trim(); | 
					
						
							|  |  |  |     lines.push(text); | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-17 19:45:15 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   let text = node.text; | 
					
						
							| 
									
										
										
										
											2021-05-12 12:08:12 -03:00
										 |  |  |   text = text.replace(/\[([^\]]*)\]\((.*?)\)/g, (match, linkName, linkUrl) => { | 
					
						
							| 
									
										
										
										
											2022-04-08 18:29:51 +02:00
										 |  |  |     const isInternal = !linkUrl.startsWith('http://') && !linkUrl.startsWith('https://'); | 
					
						
							|  |  |  |     if (isInternal) | 
					
						
							|  |  |  |       linkUrl = new URL(linkUrl.replace('.md', ''), 'https://playwright.dev/dotnet/docs/api/').toString(); | 
					
						
							| 
									
										
										
										
											2021-02-17 19:45:15 +01:00
										 |  |  |     return `<a href="${linkUrl}">${linkName}</a>`; | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-03-11 18:07:45 +01:00
										 |  |  |   text = text.replace(/(?<!`)\[(.*?)\]/g, (match, link) => `<see cref="${link}"/>`); | 
					
						
							| 
									
										
										
										
											2021-12-07 16:33:28 -08:00
										 |  |  |   text = text.replace(/`([^`]*)`/g, (match, code) => `<c>${code.replace(/</g, '<').replace(/>/g, '>')}</c>`); | 
					
						
							| 
									
										
										
										
											2021-04-30 01:08:46 -03:00
										 |  |  |   text = text.replace(/ITimeoutError/, 'TimeoutException'); | 
					
						
							|  |  |  |   text = text.replace(/Promise/, 'Task'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-02-15 19:41:40 +01:00
										 |  |  |   const words = text.split(' '); | 
					
						
							|  |  |  |   let line = ''; | 
					
						
							|  |  |  |   for (let i = 0; i < words.length; i++) { | 
					
						
							|  |  |  |     line = line + ' ' + words[i]; | 
					
						
							|  |  |  |     if (line.length >= maxColumns) { | 
					
						
							|  |  |  |       pushLine(line); | 
					
						
							|  |  |  |       line = ''; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   pushLine(line); | 
					
						
							|  |  |  |   return lines; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param {Documentation.MarkdownNode} node | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function _nodeShouldBeIgnored(node) { | 
					
						
							|  |  |  |   if (!node | 
					
						
							|  |  |  |     || (node.text === 'extends: [EventEmitter]')) | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * @param {Documentation.MarkdownNode[]} nodes | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function renderTextOnly(nodes, maxColumns = 80) { | 
					
						
							|  |  |  |   const result = _innerRenderNodes(nodes, maxColumns, false); | 
					
						
							|  |  |  |   return result.summary; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports = { renderXmlDoc, renderTextOnly } |