Skip to content

Commit 1193250

Browse files
authored
Add the ability to "Copy to Clipboard" in the UI #1120 (#1627)
Signed-off-by: tdruez <[email protected]>
1 parent 1b22e79 commit 1193250

File tree

4 files changed

+90
-2
lines changed

4 files changed

+90
-2
lines changed

scancodeio/static/main.css

+31-2
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,8 @@
3232
--nexb-orange: 247, 191, 60;
3333
--alt-orange-rgb: 255, 153, 112;
3434
--hl-gray-09: #24292e;
35+
--custom-tooltip-bg-rgb: 54, 59, 65;
36+
--custom-tooltip-padding: 0.35em 0.75em;
3537
}
3638
.theme-dark,
3739
[data-theme="dark"] {
@@ -220,12 +222,39 @@ a[target="_blank"] .fa-up-right-from-square {
220222
.is-wider-x2 .dropdown-menu {
221223
min-width: 24rem;
222224
}
225+
.copy-to-clipboard {
226+
position: relative;
227+
}
228+
.copy-tooltip {
229+
position: absolute;
230+
background-color: rgb(var(--custom-tooltip-bg-rgb));
231+
color: var(--bulma-white);
232+
padding: var(--custom-tooltip-padding);
233+
border-radius: var(--bulma-radius-small);
234+
font-size: var(--bulma-size-small);
235+
width: max-content;
236+
visibility: hidden;
237+
opacity: 0;
238+
transition: opacity 0.3s ease;
239+
pointer-events: none;
240+
text-align: center;
241+
word-wrap: break-word;
242+
bottom: 100%;
243+
left: 50%;
244+
transform: translateX(-50%);
245+
}
246+
.copy-tooltip.visible {
247+
visibility: visible;
248+
opacity: 1;
249+
transition: opacity 0.3s ease, visibility 0s linear 0s;
250+
}
223251
.is-tooltip .dropdown-content {
224-
background-color: #363636;
252+
background-color: rgb(var(--custom-tooltip-bg-rgb));
253+
color: var(--bulma-white);
225254
}
226255
.is-tooltip .dropdown-item {
227256
font-weight: normal;
228-
color: #fff;
257+
color: var(--bulma-white);
229258
}
230259
.dropdown.is-hoverable.is-disabled:hover .dropdown-menu {
231260
display: none;

scancodeio/static/main.js

+49
Original file line numberDiff line numberDiff line change
@@ -356,6 +356,54 @@ class PaginationNavigator {
356356
}
357357
}
358358

359+
// Copy to Clipboard (using `navigator.clipboard`)
360+
361+
function enableCopyToClipboard(selector) {
362+
const elements = document.querySelectorAll(selector);
363+
364+
elements.forEach(element => {
365+
element.addEventListener("click", async () => {
366+
let textToCopy = "";
367+
368+
// Determine the text to copy
369+
if (element.hasAttribute("data-copy")) {
370+
textToCopy = element.getAttribute("data-copy"); // From a custom attribute
371+
} else if (element.value) {
372+
textToCopy = element.value; // From an input field
373+
} else {
374+
textToCopy = element.innerText.trim(); // From the element's text
375+
}
376+
377+
// Default tooltip text or custom text from data-copy-feedback attribute
378+
const tooltipText = element.getAttribute('data-copy-feedback') || 'Copied!';
379+
380+
if (textToCopy) {
381+
try {
382+
await navigator.clipboard.writeText(textToCopy).then(() => {
383+
// Create a tooltip
384+
const tooltip = document.createElement('span');
385+
tooltip.classList.add('copy-tooltip');
386+
tooltip.textContent = tooltipText;
387+
element.appendChild(tooltip);
388+
389+
// Show the tooltip
390+
setTimeout(() => {
391+
tooltip.classList.add('visible');
392+
}, 0); // Add class immediately to trigger CSS transition
393+
394+
// Remove the tooltip after 1.5 seconds
395+
setTimeout(() => {
396+
element.removeChild(tooltip);
397+
}, 1500);
398+
});
399+
} catch (err) {
400+
console.error("Clipboard copy failed:", err);
401+
}
402+
}
403+
});
404+
});
405+
}
406+
359407
document.addEventListener('DOMContentLoaded', function () {
360408

361409
setupOpenModalButtons();
@@ -365,6 +413,7 @@ document.addEventListener('DOMContentLoaded', function () {
365413
setupTextarea();
366414
setupHighlightControls();
367415
setupSelectCheckbox();
416+
enableCopyToClipboard(".copy-to-clipboard");
368417

369418
const paginationContainer = document.querySelector("#pagination-header");
370419
if (paginationContainer) {

scanpipe/templates/scanpipe/includes/resource_path_links.html

+3
Original file line numberDiff line numberDiff line change
@@ -14,5 +14,8 @@ <h1 class="is-size-5 break-word">
1414
{% endif %}
1515
{% endfor %}
1616
{% include 'scanpipe/includes/admin_edit_link.html' %}
17+
<button class="copy-to-clipboard has-text-grey ml-2 is-size-6" aria-label="Copy path" data-copy="{{ object.path }}" data-copy-feedback="Resource path copied!">
18+
<i class="fa-regular fa-copy"></i>
19+
</button>
1720
</h1>
1821
{% endspaceless %}

scanpipe/templates/scanpipe/panels/project_inputs.html

+7
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,13 @@
3636
{% if input_source.is_file and input_source.size %}
3737
<span class="mr-1">{{ input_source.size|filesizeformat }}</span>
3838
{% endif %}
39+
{% if input_source.download_url %}
40+
<a class="copy-to-clipboard is-grey-link is-clickable mr-1" title="Copy URL" aria-label="Copy URL" data-copy="{{ input_source.download_url }}" data-copy-feedback="URL copied!">
41+
<span class="icon width-1 height-1">
42+
<i class="fa-regular fa-copy"></i>
43+
</span>
44+
</a>
45+
{% endif %}
3946
{% if input_source.is_file and input_source.exists and input_source.filename %}
4047
<a class="is-grey-link is-clickable" href="{% url 'project_download_input' project.slug input_source.filename %}"><span class="icon width-1 height-1"><i class="fa-solid fa-download"></i></span></a>
4148
{% endif %}

0 commit comments

Comments
 (0)