diff --git a/atest/05_Features/Signature.robot b/atest/05_Features/Signature.robot index 7fafebe28..d441000f9 100644 --- a/atest/05_Features/Signature.robot +++ b/atest/05_Features/Signature.robot @@ -6,8 +6,11 @@ Test Setup Setup Notebook Python Signature.ipynb Test Teardown Clean Up After Working With File Signature.ipynb *** Variables *** +${SIGNATURE PLUGIN ID} @krassowski/jupyterlab-lsp:signature ${SIGNATURE_BOX} css:.lsp-signature-help ${SIGNATURE_HIGHLIGHTED_ARG} css:.lsp-signature-help mark +${SIGNATURE_DETAILS_CSS} .lsp-signature-help details +${SIGNATURE_DETAILS} css:${SIGNATURE_DETAILS_CSS} *** Test Cases *** Triggers Signature Help After A Keystroke @@ -51,3 +54,27 @@ Invalidates On Cell Change Wait Until Keyword Succeeds 20x 0.5s Page Should Contain Element ${SIGNATURE_BOX} Enter Cell Editor 2 Wait Until Keyword Succeeds 20x 0.5s Page Should Not Contain Element ${SIGNATURE_BOX} + +Details Should Expand On Click + Configure JupyterLab Plugin {"maxLines": 4} plugin id=${SIGNATURE PLUGIN ID} + Enter Cell Editor 3 line=11 + Press Keys None ( + Wait Until Keyword Succeeds 20x 0.5s Page Should Contain Element ${SIGNATURE_BOX} + Wait Until Keyword Succeeds 10x 0.5s Element Should Contain ${SIGNATURE_BOX} Short description. + Page Should Contain Element ${SIGNATURE_DETAILS} + Details Should Be Collapsed ${SIGNATURE_DETAILS_CSS} + Click Element ${SIGNATURE_DETAILS} + Details Should Be Expanded ${SIGNATURE_DETAILS_CSS} + +*** Keywords *** + +Details Should Be Expanded + [Arguments] ${css_locator} + ${is_open} Execute JavaScript return document.querySelector('${css_locator}').open + Should Be True ${is_open} == True + + +Details Should Be Collapsed + [Arguments] ${css_locator} + ${is_open} Execute JavaScript return document.querySelector('${css_locator}').open + Should Be True ${is_open} == False diff --git a/atest/examples/Signature.ipynb b/atest/examples/Signature.ipynb index cf4cdb7e7..95bda6be3 100644 --- a/atest/examples/Signature.ipynb +++ b/atest/examples/Signature.ipynb @@ -27,6 +27,25 @@ "source": [ "list" ] + }, + { + "cell_type": "code", + "execution_count": null, + "metadata": {}, + "outputs": [], + "source": [ + "def with_long():\n", + " \"\"\"Short description.\n", + "\n", + " This is a longer description.\n", + " The longer description is spread across several lines.\n", + " This is to see the details collapse.\n", + " The number of lines needed depends on `maxLines` setting.\n", + " \"\"\"\n", + "\n", + "\n", + "with_long" + ] } ], "metadata": { diff --git a/packages/jupyterlab-lsp/src/features/signature.spec.ts b/packages/jupyterlab-lsp/src/features/signature.spec.ts index 10eddacbb..ceb8a8823 100644 --- a/packages/jupyterlab-lsp/src/features/signature.spec.ts +++ b/packages/jupyterlab-lsp/src/features/signature.spec.ts @@ -76,7 +76,7 @@ describe('Signature', () => { new BrowserConsole() ); expect(text).to.be.equal( - 'str(text)\nCreate a new \\*string\\* object from the given object.\n' + 'str(text)\n\nCreate a new \\*string\\* object from the given object.\n' ); }); @@ -101,7 +101,7 @@ describe('Signature', () => { new BrowserConsole() ); expect(text).to.be.equal( - 'str(text)\nCreate a new \\*string\\* object from the given object.\n' + 'str(text)\n\nCreate a new \\*string\\* object from the given object.\n' ); }); @@ -126,7 +126,54 @@ describe('Signature', () => { new BrowserConsole() ); expect(text).to.be.equal( - 'str(text)\nCreate a new *string* object from the given object.' + 'str(text)\n\nCreate a new *string* object from the given object.' + ); + }); + + it('renders plaintext with details and paramaters', async () => { + let text = signatureToMarkdown( + { + label: 'str(text)', + documentation: { + value: 'line 1\n\nline 2\nline 3\nline 4\nline 5', + kind: 'plaintext' + }, + parameters: [ + { + label: 'text', + documentation: undefined + } + ], + activeParameter: 0 + }, + 'python', + MockHighlighter, + new BrowserConsole(), + undefined, + 4 + ); + expect(text).to.be.equal( + 'str(text)\n\nline 1\n
\nline 2\nline 3\nline 4\nline 5\n
' + ); + }); + + it('renders plaintext with details and no parameters', async () => { + let text = signatureToMarkdown( + { + label: 'str()', + documentation: { + value: 'line 1\n\nline 2\nline 3\nline 4\nline 5', + kind: 'plaintext' + } + }, + 'python', + MockHighlighter, + new BrowserConsole(), + undefined, + 4 + ); + expect(text).to.be.equal( + '```python\nstr()\n```\n\nline 1\n
\nline 2\nline 3\nline 4\nline 5\n
' ); }); }); diff --git a/packages/jupyterlab-lsp/src/features/signature.ts b/packages/jupyterlab-lsp/src/features/signature.ts index 92217cecc..f35c9da0f 100644 --- a/packages/jupyterlab-lsp/src/features/signature.ts +++ b/packages/jupyterlab-lsp/src/features/signature.ts @@ -104,7 +104,6 @@ export function signatureToMarkdown( } let details = ''; if (item.documentation) { - details += '\n'; if ( typeof item.documentation === 'string' || item.documentation.kind === 'plaintext' @@ -123,7 +122,7 @@ export function signatureToMarkdown( } } else { if (item.documentation.kind !== 'markdown') { - console.warn('Unknown MarkupContent kind:', item.documentation.kind); + logger.warn('Unknown MarkupContent kind:', item.documentation.kind); } details += item.documentation.value; } @@ -146,7 +145,9 @@ export function signatureToMarkdown( details = '
\n' + details + '\n
'; } } - markdown += details; + markdown += '\n\n' + details; + } else { + markdown += '\n'; } return markdown; } @@ -181,7 +182,7 @@ export class SignatureCM extends CodeMirrorIntegration { // (allowing user to select/copy from signature) if ( this.isSignatureShown() && - (event.target as Element).closest('.' + CLASS_NAME) === null + (event.relatedTarget as Element).closest('.' + CLASS_NAME) === null ) { this._hideTooltip(); } @@ -275,7 +276,7 @@ export class SignatureCM extends CodeMirrorIntegration { code.appendChild(element); } ); - return pre.outerHTML + '\n\n'; + return pre.outerHTML; } /**