diff --git a/modules/project_browser_source_example/src/Plugin/ProjectBrowserSource/ProjectBrowserSourceExample.php b/modules/project_browser_source_example/src/Plugin/ProjectBrowserSource/ProjectBrowserSourceExample.php index 75b1be393e546a5be96dbfca70bc2e020eeaa705..c003e7b28c491f8fdcd52f826944a065e6b8fbf2 100644 --- a/modules/project_browser_source_example/src/Plugin/ProjectBrowserSource/ProjectBrowserSourceExample.php +++ b/modules/project_browser_source_example/src/Plugin/ProjectBrowserSource/ProjectBrowserSourceExample.php @@ -2,6 +2,7 @@ namespace Drupal\project_browser_source_example\Plugin\ProjectBrowserSource; +use Drupal\Core\Extension\ModuleExtensionList; use Drupal\project_browser\Plugin\ProjectBrowserSourceBase; use Drupal\project_browser\ProjectBrowser\Project; use Drupal\project_browser\ProjectBrowser\ProjectsResultsPage; @@ -30,12 +31,15 @@ class ProjectBrowserSourceExample extends ProjectBrowserSourceBase { * The plugin implementation definition. * @param \Symfony\Component\HttpFoundation\RequestStack $requestStack * The request from the browser. + * @param \Drupal\Core\Extension\ModuleExtensionList $moduleExtensionList + * The module extension list. */ public function __construct( array $configuration, $plugin_id, $plugin_definition, protected readonly RequestStack $requestStack, + protected ModuleExtensionList $moduleExtensionList, ) { parent::__construct($configuration, $plugin_id, $plugin_definition); } @@ -49,6 +53,7 @@ class ProjectBrowserSourceExample extends ProjectBrowserSourceBase { $plugin_id, $plugin_definition, $container->get('request_stack'), + $container->get('extension.list.module'), ); } @@ -155,6 +160,42 @@ class ProjectBrowserSourceExample extends ProjectBrowserSourceBase { // Images: Array of images using the same structure as $logo, above. images: [], ); + // @phpstan-ignore-next-line + $pb_path = $this->moduleExtensionList->getPath('project_browser'); + $projects[] = new Project( + id: $project_from_source['identifier'] . '2', + logo: $logo, + // Maybe the source won't have all fields, but we still need to + // populate the values of all the properties. + isCompatible: TRUE, + isMaintained: TRUE, + isCovered: TRUE, + isActive: TRUE, + starUserCount: 0, + projectUsageTotal: 0, + machineName: $project_from_source['unique_name'] . '2', + body: [ + 'summary' => $project_from_source['short_description'] . ' (different commands)', + 'value' => $project_from_source['long_description'] . ' (different commands)', + ], + title: 'A project with different commands', + // Status: 1 enabled / 0 disabled. + status: 1, + changed: $project_from_source['updated_at'], + created: $project_from_source['created_at'], + author: $author, + composerNamespace: $project_from_source['composer_namespace'], + categories: $categories, + // Images: Array of images using the same structure as $logo, above. + images: [], + type: 'different-commands', + commands: "<b>Steps to doing this thing</b> + <p>You can do it!</p> + <div class=\"command-box\"> + <input id=\"{$project_from_source['identifier']}-download-command\" value=\"composer require {$project_from_source['unique_name'] }\" readonly=\"\"> + <button data-copy-command><img src=\"/{$pb_path}/images/copy-icon.svg\" alt=\"Copy steps for {$project_from_source['identifier']}\"/></button> + </div>", + ); } // Return one page of results. The first parameter is the total number of diff --git a/project_browser.libraries.yml b/project_browser.libraries.yml index 351181ad81547644b068f7c72df47c5a0ea498b0..af1f1f0764f127572eb586f69fb60b5e229c8d61 100644 --- a/project_browser.libraries.yml +++ b/project_browser.libraries.yml @@ -12,6 +12,7 @@ svelte: - core/drupal.debounce - core/drupal.dialog - core/drupal.announce + - core/once - project_browser/project_browser project_browser: diff --git a/src/ProjectBrowser/Project.php b/src/ProjectBrowser/Project.php index 49ba0ca3f16963d994d78ac498f930132550ec1f..74df556f349937d1348fa837e96c938bd8a3a86f 100644 --- a/src/ProjectBrowser/Project.php +++ b/src/ProjectBrowser/Project.php @@ -4,6 +4,7 @@ namespace Drupal\project_browser\ProjectBrowser; use Drupal\Component\Utility\Html; use Drupal\Component\Utility\Unicode; +use Drupal\Component\Utility\Xss; /** * Defines a single Project. @@ -53,6 +54,26 @@ class Project implements \JsonSerializable { * Images of the project. * @param array $warnings * Warnings for the project. + * @param string $type + * The project type. Defaults to 'module:drupalorg' to indicate modules from + * D.O., but may be changed to anything else that could helpfully identify + * a project type. + * @param string|bool $commands + * When FALSE, the project browser UI will not provide a "View Commands" + * button for the project UNLESS the type 'module:drupalorg', in which case + * it displays Svelte-generated install instructions. + * When it is a string and NOT 'module:drupalorg', that string will become + * the contents of the "View Commands" popup. + * To include a paste-able command that includes a copy button, use this + * markup structure: + * @code + * <div class="command-box"> + * <input value="THE_COMMAND_TO_BE_COPIED" readonly="" /> + * <button data-copy-command> + * <img src="/PATH_TO_PROJECT_BROWSER/images/copy-icon.svg\" alt="ALT TEXT"/> + * </button> + * </div> + * @endcode */ public function __construct( public string $id, @@ -75,6 +96,8 @@ class Project implements \JsonSerializable { public array $categories = [], public array $images = [], public array $warnings = [], + public string $type = 'module:drupalorg', + public string|bool $commands = FALSE, ) { $this->setSummary($body); } @@ -135,6 +158,8 @@ class Project implements \JsonSerializable { 'changed' => $this->changed, 'created' => $this->created, 'selector_id' => $this->getSelectorId(), + 'type' => $this->type, + 'commands' => Xss::filter($this->commands, [...Xss::getAdminTagList(), 'input', 'button']), ]; } diff --git a/sveltejs/css/claro.css b/sveltejs/css/claro.css index d6bca4dd0fbc152593fc743a3bd62d23b19ba84c..0adf3d560e906b455dba0a4f44333e5cc3c78da7 100644 --- a/sveltejs/css/claro.css +++ b/sveltejs/css/claro.css @@ -32,6 +32,7 @@ padding-inline: 1rem 0.25rem; } .project-browser-popup .copied-action { + position: absolute; z-index: 1; right: -18px; width: fit-content; diff --git a/sveltejs/public/build/bundle.js b/sveltejs/public/build/bundle.js index cdae5fdb3a9fb43b536867f2c4fcefbfd8fb3561..a6faced07b6d5205aa1ed7267371cc23a7871558 100644 Binary files a/sveltejs/public/build/bundle.js and b/sveltejs/public/build/bundle.js differ diff --git a/sveltejs/public/build/bundle.js.map b/sveltejs/public/build/bundle.js.map index bf947ccf70fa162466436c8fb9b1add946174626..e9d7a93e4517405cb95e552ee052a1f0303bbe05 100644 Binary files a/sveltejs/public/build/bundle.js.map and b/sveltejs/public/build/bundle.js.map differ diff --git a/sveltejs/src/Project/ActionButton.svelte b/sveltejs/src/Project/ActionButton.svelte index b71951961c68a32ccb45a8344a59b0d7faa3bc52..e92b4627c0ffa4bc8cf6b922d060bd4c5c7edcaa 100644 --- a/sveltejs/src/Project/ActionButton.svelte +++ b/sveltejs/src/Project/ActionButton.svelte @@ -224,7 +224,7 @@ {showStatus} /> {/if} - {:else} + {:else if project.type === 'module:drupalorg' || project.commands} <ProjectButtonBase click={() => openPopup(getCommandsPopupMessage(project), project)} >{Drupal.t('View Commands')} diff --git a/sveltejs/src/ProjectBrowser.svelte b/sveltejs/src/ProjectBrowser.svelte index aeb73acc29021813d43f26495af4d8f3a9f2c075..5de0db2492acfb1f6eea8e9077124289c2a5efe6 100644 --- a/sveltejs/src/ProjectBrowser.svelte +++ b/sveltejs/src/ProjectBrowser.svelte @@ -176,7 +176,7 @@ } await load($page); - const focus = document.getElementById(element); + const focus = element ? document.getElementById(element) : false; if (focus) { focus.focus(); $focusedElement = ''; diff --git a/sveltejs/src/popup.js b/sveltejs/src/popup.js index a327a7c120d3d36e94f97c6255c9bdbbec963ce1..c6d80261a19e651b0e9609242082a4f98179ac79 100644 --- a/sveltejs/src/popup.js +++ b/sveltejs/src/popup.js @@ -1,96 +1,117 @@ import { FULL_MODULE_PATH, ORIGIN_URL } from './constants'; // cspell:ignore dont +const { once, Drupal } = window; -export const copyCommand = (cmd, project) => { - const getCopyElements = () => { - const getCopyElement = (suffix) => document.querySelector(`#${project.project_machine_name}-${suffix}`) - const action = ['Download', 'Install'].includes(cmd) ? cmd.toLowerCase() : 'install-drush'; - return [ - getCopyElement(`${action}-command`), - getCopyElement(`copied-${action}`), - ] - } - const [copiedCommand, copyReceipt] = getCopyElements(); - - copiedCommand.select(); - // For mobile devices. - copiedCommand.setSelectionRange(0, 99999); - navigator.clipboard.writeText(copiedCommand.value); - copyReceipt.style.opacity = '1'; +/** + * Finds [data-copy-command] buttons and adds copy functionality to them. + */ +const enableCopyButtons = () => { setTimeout(() => { - copyReceipt.style.transition = 'opacity 0.3s'; - copyReceipt.style.opacity = '0'; - }, 1000); -}; + once('copyButton', '[data-copy-command]').forEach((copyButton) => { + // If clipboard is not supported (likely due to non-https), then hide the + // button and do not bother with event listeners + if (!navigator.clipboard) { + // copyButton.hidden = true; + // return; + } + copyButton.addEventListener('click', (e) => { + // The copy button must be contained in a div + const container = e.target.closest('div'); + // The only <input> within the parent dive should have its value set + // to the command that should be copied. + const input = container.querySelector('input'); + + // Make the input value the selected text + input.select() + input.setSelectionRange(0, 99999); + navigator.clipboard.writeText(input.value); + Drupal.announce(Drupal.t('Copied text to clipboard')); + + // Create a "receipt" that will visually show the text has been copied. + const receipt = document.createElement('div') + receipt.textContent = Drupal.t('Copied') + receipt.classList.add('copied-action') + receipt.style.opacity = '1'; + input.insertAdjacentElement('afterend', receipt) + // eslint-disable-next-line max-nested-callbacks + setTimeout(() => { + // Remove the receipt after 1 second. + receipt.remove() + }, 1000); + }) + }) + }) +} export const getCommandsPopupMessage = (project) => { - const download = Drupal.t('Download'); - const composerText = Drupal.t( - 'The !use_composer_open recommended way!close to download any Drupal module is with !get_composer_open Composer!close.', - { - '!close': '</a>', - '!use_composer_open': - '<a href="https://www.drupal.org/docs/develop/using-composer/using-composer-to-install-drupal-and-manage-dependencies#managing-contributed" target="_blank" rel="noreferrer noopener">', - '!get_composer_open': - '<a href="https://getcomposer.org/" target="_blank" rel="noopener noreferrer">', - }, - ); - const composerExistsText = Drupal.t( - "If you already manage your Drupal application dependencies with Composer, run the following from the command line in your application's Composer root directory", - ); - const infoText = Drupal.t('This will download the module to your codebase.'); - const composerDontWorkText = Drupal.t( - "Didn't work? !learn_open Learn how to troubleshoot Composer!close", - { - '!learn_open': - '<a href="https://getcomposer.org/doc/articles/troubleshooting.md" target="_blank" rel="noopener noreferrer">', - '!close': '</a>', - }, - ); - const downloadModuleText = Drupal.t( - 'If you cannot use Composer, you may !dl_manually_open download the module manually through your browser!close', - { - '!dl_manually_open': - '<a href="https://www.drupal.org/docs/user_guide/en/extend-module-install.html#s-using-the-administrative-interface" target="_blank" rel="noreferrer">', - '!close': '</a>', - }, - ); - const install = Drupal.t('Install'); - const installText = Drupal.t( - 'Go to the !module_page_open Extend page!close (admin/modules), check the box next to each module you wish to enable, then click the Install button at the bottom of the page.', - { - '!module_page_open': `<a href="${ORIGIN_URL}/admin/modules" target="_blank" rel="noopener noreferrer">`, - '!close': '</a>', - }, - ); - const drushText = Drupal.t( - 'Alternatively, you can use !drush_openDrush!close to install it via the command line', - { - '!drush_open': '<a href="https://www.drush.org/latest/" target="_blank" rel="noopener noreferrer">', - '!close': '</a>', - }, - ); - const installDrush = Drupal.t( - 'If Drush is not installed, this will add the tool to your codebase', - ); - const copied = Drupal.t('Copied!'); - const downloadAlt = Drupal.t('Copy the download command'); - const installAlt = Drupal.t('Copy the install command'); - const drushAlt = Drupal.t('Copy the install Drush command'); - const copyIcon = `${FULL_MODULE_PATH}/images/copy-icon.svg`; - const makeButton = (altText, action) => `<button id="${action}-btn"><img src="${copyIcon}" alt="${altText}"/></button> - <div id="${project.project_machine_name}-copied-${action}" class="copied-action">${copied}</div>` - const downloadCopyButton = navigator.clipboard ? makeButton(downloadAlt, 'download') : ''; - const installCopyButton = navigator.clipboard ? makeButton(installAlt, 'install') : ''; - const installDrushCopyButton = navigator.clipboard ? makeButton(drushAlt, 'install-drush') : ''; + // @todo move the message provided in this condition to the 'commands' + // property of the project definition. + if (project.type === 'module:drupalorg') { + const download = Drupal.t('Download'); + const composerText = Drupal.t( + 'The !use_composer_open recommended way!close to download any Drupal module is with !get_composer_open Composer!close.', + { + '!close': '</a>', + '!use_composer_open': + '<a href="https://www.drupal.org/docs/develop/using-composer/using-composer-to-install-drupal-and-manage-dependencies#managing-contributed" target="_blank" rel="noreferrer noopener">', + '!get_composer_open': + '<a href="https://getcomposer.org/" target="_blank" rel="noopener noreferrer">', + }, + ); + const composerExistsText = Drupal.t( + "If you already manage your Drupal application dependencies with Composer, run the following from the command line in your application's Composer root directory", + ); + const infoText = Drupal.t('This will download the module to your codebase.'); + const composerDontWorkText = Drupal.t( + "Didn't work? !learn_open Learn how to troubleshoot Composer!close", + { + '!learn_open': + '<a href="https://getcomposer.org/doc/articles/troubleshooting.md" target="_blank" rel="noopener noreferrer">', + '!close': '</a>', + }, + ); + const downloadModuleText = Drupal.t( + 'If you cannot use Composer, you may !dl_manually_open download the module manually through your browser!close', + { + '!dl_manually_open': + '<a href="https://www.drupal.org/docs/user_guide/en/extend-module-install.html#s-using-the-administrative-interface" target="_blank" rel="noreferrer">', + '!close': '</a>', + }, + ); + const install = Drupal.t('Install'); + const installText = Drupal.t( + 'Go to the !module_page_open Extend page!close (admin/modules), check the box next to each module you wish to enable, then click the Install button at the bottom of the page.', + { + '!module_page_open': `<a href="${ORIGIN_URL}/admin/modules" target="_blank" rel="noopener noreferrer">`, + '!close': '</a>', + }, + ); + const drushText = Drupal.t( + 'Alternatively, you can use !drush_openDrush!close to install it via the command line', + { + '!drush_open': '<a href="https://www.drush.org/latest/" target="_blank" rel="noopener noreferrer">', + '!close': '</a>', + }, + ); + const installDrush = Drupal.t( + 'If Drush is not installed, this will add the tool to your codebase', + ); + const downloadAlt = Drupal.t('Copy the download command'); + const installAlt = Drupal.t('Copy the install command'); + const drushAlt = Drupal.t('Copy the install Drush command'); + const copyIcon = `${FULL_MODULE_PATH}/images/copy-icon.svg`; + const makeButton = (altText, action) => `<button data-copy-command id="${action}-btn"><img src="${copyIcon}" alt="${altText}"/></button>` + const downloadCopyButton = makeButton(downloadAlt, 'download'); + const installCopyButton = makeButton(installAlt, 'install'); + const installDrushCopyButton = makeButton(drushAlt, 'install-drush'); - const div = document.createElement('div'); - div.classList.add('window'); - div.innerHTML = `<h3>1. ${download}</h3> + const div = document.createElement('div'); + div.classList.add('window'); + div.innerHTML = `<h3>1. ${download}</h3> <p>${composerText}</p> <p>${composerExistsText}:</p> <div class="command-box"> - <input id="${project.project_machine_name}-download-command" value="composer require ${project.composer_namespace}" readonly/> + <input value="composer require ${project.composer_namespace}" readonly/> ${downloadCopyButton} </div> @@ -101,14 +122,14 @@ export const getCommandsPopupMessage = (project) => { <p>${installText}</p> <p>${drushText}:</p> <div class="command-box"> - <input id="${project.project_machine_name}-install-command" value="drush pm:install ${project.project_machine_name}" readonly/> + <input value="drush pm:install ${project.project_machine_name}" readonly/> ${installCopyButton} </div> </div> <p>${installDrush}:</p> <div class="command-box"> - <input id="${project.project_machine_name}-install-drush-command" value="composer require drush/drush" readonly/> + <input value="composer require drush/drush" readonly/> ${installDrushCopyButton} </div> <style> @@ -118,21 +139,23 @@ export const getCommandsPopupMessage = (project) => { border: 1px solid; } </style>`; - if (navigator.clipboard) { - [['download', 'Download'], ['install', 'Install'], ['install-drush', 'Drush']].forEach(([id, command]) => { - div.querySelector(`#${id}-btn`).addEventListener('click', () => { - copyCommand(command, project); - }); - }); + enableCopyButtons(); + return div; + } + if (project.commands) { + const div = document.createElement('div'); + div.innerHTML = project.commands; + enableCopyButtons(); + return div; } - return div; + }; export const openPopup = (getMessage, project) => { const message = typeof getMessage === 'function' ? getMessage() : getMessage; const popupModal = Drupal.dialog(message, { title: project.title, - dialogClass: 'project-browser-popup', + classes: {'ui-dialog': 'project-browser-popup'}, width: '50rem', }); popupModal.showModal(); diff --git a/tests/modules/project_browser_test/project_browser_test.info.yml b/tests/modules/project_browser_test/project_browser_test.info.yml index ccb21531ef1ae4b3e4fe4e1cbd8cdb957d1df14c..bdeba4dcba4b2a674186231989460cd6693b4a05 100644 --- a/tests/modules/project_browser_test/project_browser_test.info.yml +++ b/tests/modules/project_browser_test/project_browser_test.info.yml @@ -1,6 +1,6 @@ name: Project Browser test type: module description: 'Support module for testing Project Browser.' -core_version_requirement: ^9 || ^10 +package: Testing dependencies: - project_browser:project_browser diff --git a/tests/src/FunctionalJavascript/ProjectBrowserUiTest.php b/tests/src/FunctionalJavascript/ProjectBrowserUiTest.php index ba9a61eb35f3e994ff02871186a6d6cc0d2042be..174ef53b3f6901c17c17521c85af6bdcd99cb942 100644 --- a/tests/src/FunctionalJavascript/ProjectBrowserUiTest.php +++ b/tests/src/FunctionalJavascript/ProjectBrowserUiTest.php @@ -200,18 +200,18 @@ class ProjectBrowserUiTest extends WebDriverTestBase { $this->getSession()->executeScript('navigator.clipboard = true'); $this->assertTrue($assert_session->waitForText('By Hel Vetica')); $this->clickWithWait('#project-browser .project__action_button'); - $allowed_html_field = $assert_session->fieldExists('helvetica-download-command'); - $this->assertTrue($allowed_html_field->hasAttribute('readonly')); - $allowed_html_field = $assert_session->fieldExists('helvetica-install-command'); - $this->assertTrue($allowed_html_field->hasAttribute('readonly')); + $require_command = $assert_session->waitForElement('css', 'input[value="composer require drupal/helvetica"]'); + $this->assertTrue($require_command->hasAttribute('readonly')); + $install_command = $assert_session->waitForElement('css', 'input[value="drush pm:install helvetica"]'); + $this->assertTrue($install_command->hasAttribute('readonly')); // Tests alt text for copy command image. - $download_command = $page->find('css', '#download-btn img'); - $this->assertEquals('Copy the download command', $download_command->getAttribute('alt')); + $download_commands = $page->findAll('css', '.command-box img'); + $this->assertEquals('Copy the download command', $download_commands[0]->getAttribute('alt')); $install_command = $page->find('css', '#install-btn img'); - $this->assertEquals('Copy the install command', $install_command->getAttribute('alt')); + $this->assertEquals('Copy the install command', $download_commands[1]->getAttribute('alt')); $install_command = $page->find('css', '#install-drush-btn img'); - $this->assertEquals('Copy the install Drush command', $install_command->getAttribute('alt')); + $this->assertEquals('Copy the install Drush command', $download_commands[2]->getAttribute('alt')); } /**