74 lines
2.2 KiB
JavaScript
74 lines
2.2 KiB
JavaScript
import { uniqueId } from 'lodash';
|
|
import { __ } from '~/locale';
|
|
import { spriteIcon } from '~/lib/utils/common_utils';
|
|
import { setAttributes } from '~/lib/utils/dom_utils';
|
|
|
|
class CopyCodeButton extends HTMLElement {
|
|
connectedCallback() {
|
|
if (this.querySelector('.btn')) return;
|
|
|
|
this.for = uniqueId('code-');
|
|
|
|
const target = this.parentNode.querySelector('pre');
|
|
if (!target || this.closest('.suggestions')) return;
|
|
|
|
target.setAttribute('id', this.for);
|
|
|
|
this.appendChild(this.createButton());
|
|
}
|
|
|
|
createButton() {
|
|
const button = document.createElement('button');
|
|
|
|
setAttributes(button, {
|
|
type: 'button',
|
|
class: 'btn btn-default btn-md gl-button btn-icon has-tooltip',
|
|
'data-title': __('Copy to clipboard'),
|
|
'data-clipboard-target': `pre#${this.for}`,
|
|
});
|
|
|
|
// eslint-disable-next-line no-unsanitized/property
|
|
button.innerHTML = spriteIcon('copy-to-clipboard');
|
|
|
|
return button;
|
|
}
|
|
}
|
|
|
|
function addCodeButton() {
|
|
[...document.querySelectorAll('pre.code.js-syntax-highlight:not(.content-editor-code-block)')]
|
|
.filter((el) => el.getAttribute('lang') !== 'mermaid')
|
|
.filter((el) => !el.closest('.js-markdown-code'))
|
|
.forEach((el) => {
|
|
const copyCodeEl = document.createElement('copy-code');
|
|
copyCodeEl.setAttribute('for', uniqueId('code-'));
|
|
|
|
const wrapper = document.createElement('div');
|
|
wrapper.className = 'gl-relative markdown-code-block js-markdown-code';
|
|
wrapper.appendChild(el.cloneNode(true));
|
|
wrapper.appendChild(copyCodeEl);
|
|
|
|
el.parentNode.insertBefore(wrapper, el);
|
|
|
|
el.remove();
|
|
});
|
|
}
|
|
|
|
export const initCopyCodeButton = (selector = '#content-body') => {
|
|
if (!customElements.get('copy-code')) {
|
|
customElements.define('copy-code', CopyCodeButton);
|
|
}
|
|
|
|
const exclude = document.querySelector('.file-content.code'); // this behavior is not needed when viewing raw file content, so excluding it as the unnecessary dom lookups can become expensive
|
|
const el = document.querySelector(selector);
|
|
|
|
if (!el || exclude) return () => {};
|
|
|
|
const observer = new MutationObserver(() => addCodeButton());
|
|
|
|
observer.observe(document.querySelector(selector), {
|
|
childList: true,
|
|
subtree: true,
|
|
});
|
|
|
|
return () => observer.disconnect();
|
|
};
|