Clipboard API
The modern Clipboard API provides secure access to clipboard operations, replacing the older document.execCommand() method.
Basic Clipboard Operations
html
<div class="clipboard-demo">
<textarea id="text-input" placeholder="Type something to copy...">Hello, World!</textarea>
<br><br>
<button id="copy-btn">📋 Copy Text</button>
<button id="paste-btn">📄 Paste Text</button>
<button id="clear-btn">🗑️ Clear</button>
<div id="status"></div>
</div>
<script>
const textInput = document.getElementById('text-input');
const copyBtn = document.getElementById('copy-btn');
const pasteBtn = document.getElementById('paste-btn');
const clearBtn = document.getElementById('clear-btn');
const status = document.getElementById('status');
// Copy text to clipboard
copyBtn.addEventListener('click', async () => {
try {
const text = textInput.value;
await navigator.clipboard.writeText(text);
showStatus('✅ Text copied to clipboard!', 'success');
} catch (err) {
showStatus('❌ Failed to copy text: ' + err.message, 'error');
// Fallback for older browsers
fallbackCopy(textInput.value);
}
});
// Paste text from clipboard
pasteBtn.addEventListener('click', async () => {
try {
const text = await navigator.clipboard.readText();
textInput.value = text;
showStatus('✅ Text pasted from clipboard!', 'success');
} catch (err) {
showStatus('❌ Failed to paste text: ' + err.message, 'error');
}
});
// Clear text
clearBtn.addEventListener('click', () => {
textInput.value = '';
showStatus('🗑️ Text cleared', 'info');
});
// Fallback copy method for older browsers
function fallbackCopy(text) {
const textArea = document.createElement('textarea');
textArea.value = text;
document.body.appendChild(textArea);
textArea.select();
try {
document.execCommand('copy');
showStatus('✅ Text copied to clipboard! (fallback)', 'success');
} catch (err) {
showStatus('❌ Copy failed', 'error');
}
document.body.removeChild(textArea);
}
// Status display function
function showStatus(message, type = 'info') {
status.textContent = message;
status.className = `status ${type}`;
setTimeout(() => {
status.textContent = '';
status.className = '';
}, 3000);
}
// Check clipboard API support
if (!navigator.clipboard) {
showStatus('⚠️ Clipboard API not supported, using fallback', 'warning');
}
</script>
<style>
.clipboard-demo {
padding: 20px;
border: 1px solid #ddd;
border-radius: 8px;
margin: 20px 0;
}
#text-input {
width: 100%;
height: 100px;
padding: 10px;
border: 1px solid #ccc;
border-radius: 4px;
font-family: Arial, sans-serif;
}
button {
padding: 10px 15px;
margin: 5px;
border: none;
border-radius: 4px;
cursor: pointer;
background: #007bff;
color: white;
}
button:hover {
background: #0056b3;
}
.status {
margin-top: 10px;
padding: 8px;
border-radius: 4px;
}
.status.success {
background: #d4edda;
color: #155724;
border: 1px solid #c3e6cb;
}
.status.error {
background: #f8d7da;
color: #721c24;
border: 1px solid #f5c6cb;
}
.status.warning {
background: #fff3cd;
color: #856404;
border: 1px solid #ffeaa7;
}
.status.info {
background: #d1ecf1;
color: #0c5460;
border: 1px solid #bee5eb;
}
</style>
Advanced Clipboard Operations
html
<div class="advanced-clipboard">
<h3>Advanced Clipboard Demo</h3>
<!-- Rich text editor -->
<div id="editor" contenteditable="true"
style="border: 1px solid #ccc; padding: 10px; min-height: 100px; margin-bottom: 10px;">
<p>This is <strong>rich text</strong> with <em>formatting</em>!</p>
<p>You can copy and paste <span style="color: blue;">styled content</span>.</p>
</div>
<button id="copy-html">Copy HTML</button>
<button id="paste-html">Paste HTML</button>
<button id="copy-image">Copy Image</button>
<!-- Image display -->
<div id="image-container" style="margin-top: 15px;">
<img id="demo-image" src=""
alt="Demo image" style="max-width: 100px; margin: 10px 0;">
</div>
<div id="advanced-status"></div>
</div>
<script>
const editor = document.getElementById('editor');
const copyHtmlBtn = document.getElementById('copy-html');
const pasteHtmlBtn = document.getElementById('paste-html');
const copyImageBtn = document.getElementById('copy-image');
const demoImage = document.getElementById('demo-image');
const advancedStatus = document.getElementById('advanced-status');
// Copy HTML content
copyHtmlBtn.addEventListener('click', async () => {
try {
const html = editor.innerHTML;
const plainText = editor.textContent;
const clipboardItem = new ClipboardItem({
'text/html': new Blob([html], { type: 'text/html' }),
'text/plain': new Blob([plainText], { type: 'text/plain' })
});
await navigator.clipboard.write([clipboardItem]);
showAdvancedStatus('✅ Rich content copied!', 'success');
} catch (err) {
showAdvancedStatus('❌ Failed to copy HTML: ' + err.message, 'error');
}
});
// Paste HTML content
pasteHtmlBtn.addEventListener('click', async () => {
try {
const clipboardItems = await navigator.clipboard.read();
for (const clipboardItem of clipboardItems) {
// Try to get HTML first, then plain text
if (clipboardItem.types.includes('text/html')) {
const htmlBlob = await clipboardItem.getType('text/html');
const html = await htmlBlob.text();
editor.innerHTML = html;
showAdvancedStatus('✅ Rich content pasted!', 'success');
} else if (clipboardItem.types.includes('text/plain')) {
const textBlob = await clipboardItem.getType('text/plain');
const text = await textBlob.text();
editor.textContent = text;
showAdvancedStatus('✅ Plain text pasted!', 'success');
}
}
} catch (err) {
showAdvancedStatus('❌ Failed to paste: ' + err.message, 'error');
}
});
// Copy image to clipboard
copyImageBtn.addEventListener('click', async () => {
try {
// Convert image to canvas then to blob
const canvas = document.createElement('canvas');
const ctx = canvas.getContext('2d');
canvas.width = demoImage.naturalWidth || 100;
canvas.height = demoImage.naturalHeight || 100;
ctx.drawImage(demoImage, 0, 0);
canvas.toBlob(async (blob) => {
try {
const clipboardItem = new ClipboardItem({
[blob.type]: blob
});
await navigator.clipboard.write([clipboardItem]);
showAdvancedStatus('✅ Image copied to clipboard!', 'success');
} catch (err) {
showAdvancedStatus('❌ Failed to copy image: ' + err.message, 'error');
}
});
} catch (err) {
showAdvancedStatus('❌ Failed to process image: ' + err.message, 'error');
}
});
function showAdvancedStatus(message, type) {
advancedStatus.textContent = message;
advancedStatus.className = `status ${type}`;
setTimeout(() => {
advancedStatus.textContent = '';
advancedStatus.className = '';
}, 3000);
}
</script>
Drag and Drop API
The HTML Drag and Drop API enables drag-and-drop functionality for web applications.
Basic Drag and Drop
html
<div class="drag-drop-demo">
<h3>Basic Drag and Drop</h3>
<div class="drag-container">
<div class="draggable-item" draggable="true" data-item="apple">
🍎 Apple
</div>
<div class="draggable-item" draggable="true" data-item="banana">
🍌 Banana
</div>
<div class="draggable-item" draggable="true" data-item="orange">
🍊 Orange
</div>
</div>
<div class="drop-zones">
<div class="drop-zone" data-zone="fruits">
<h4>Fruits Basket</h4>
<div class="zone-content"></div>
</div>
<div class="drop-zone" data-zone="trash">
<h4>🗑️ Trash</h4>
<div class="zone-content"></div>
</div>
</div>
<button id="reset-items">Reset Items</button>
</div>
<style>
.drag-drop-demo {
padding: 20px;
border: 1px solid #ddd;
border-radius: 8px;
margin: 20px 0;
}
.drag-container {
display: flex;
gap: 10px;
margin: 20px 0;
padding: 15px;
background: #f8f9fa;
border-radius: 8px;
}
.draggable-item {
padding: 15px 20px;
background: #ffffff;
border: 2px solid #dee2e6;
border-radius: 8px;
cursor: grab;
user-select: none;
transition: all 0.2s ease;
}
.draggable-item:hover {
background: #e9ecef;
transform: translateY(-2px);
}
.draggable-item.dragging {
opacity: 0.5;
transform: rotate(5deg);
}
.drop-zones {
display: flex;
gap: 20px;
margin: 20px 0;
}
.drop-zone {
flex: 1;
min-height: 150px;
border: 3px dashed #ced4da;
border-radius: 8px;
padding: 15px;
background: #f8f9fa;
transition: all 0.3s ease;
}
.drop-zone.drag-over {
border-color: #007bff;
background: #e3f2fd;
}
.drop-zone h4 {
margin: 0 0 10px 0;
color: #495057;
text-align: center;
}
.zone-content {
display: flex;
flex-direction: column;
gap: 8px;
}
.zone-content .draggable-item {
margin: 0;
}
#reset-items {
padding: 10px 20px;
background: #28a745;
color: white;
border: none;
border-radius: 4px;
cursor: pointer;
}
</style>
<script>
let draggedElement = null;
// Get all draggable items and drop zones
const draggableItems = document.querySelectorAll('.draggable-item');
const dropZones = document.querySelectorAll('.drop-zone');
const resetBtn = document.getElementById('reset-items');
// Add drag event listeners to draggable items
draggableItems.forEach(item => {
item.addEventListener('dragstart', handleDragStart);
item.addEventListener('dragend', handleDragEnd);
});
// Add drop event listeners to drop zones
dropZones.forEach(zone => {
zone.addEventListener('dragover', handleDragOver);
zone.addEventListener('dragenter', handleDragEnter);
zone.addEventListener('dragleave', handleDragLeave);
zone.addEventListener('drop', handleDrop);
});
function handleDragStart(e) {
draggedElement = this;
this.classList.add('dragging');
// Set drag data
e.dataTransfer.effectAllowed = 'move';
e.dataTransfer.setData('text/html', this.outerHTML);
e.dataTransfer.setData('text/plain', this.textContent);
e.dataTransfer.setData('application/x-item-id', this.dataset.item);
}
function handleDragEnd(e) {
this.classList.remove('dragging');
draggedElement = null;
}
function handleDragOver(e) {
if (e.preventDefault) {
e.preventDefault();
}
e.dataTransfer.dropEffect = 'move';
return false;
}
function handleDragEnter(e) {
this.classList.add('drag-over');
}
function handleDragLeave(e) {
this.classList.remove('drag-over');
}
function handleDrop(e) {
if (e.stopPropagation) {
e.stopPropagation();
}
this.classList.remove('drag-over');
if (draggedElement) {
// Move the element to the drop zone
const zoneContent = this.querySelector('.zone-content');
const clonedElement = draggedElement.cloneNode(true);
// Add event listeners to the cloned element
clonedElement.addEventListener('dragstart', handleDragStart);
clonedElement.addEventListener('dragend', handleDragEnd);
zoneContent.appendChild(clonedElement);
// Remove original element
draggedElement.remove();
// Special handling for trash zone
if (this.dataset.zone === 'trash') {
setTimeout(() => {
clonedElement.style.animation = 'fadeOut 0.5s ease-out forwards';
setTimeout(() => clonedElement.remove(), 500);
}, 100);
}
}
return false;
}
// Reset functionality
resetBtn.addEventListener('click', () => {
// Clear all drop zones
document.querySelectorAll('.zone-content').forEach(content => {
content.innerHTML = '';
});
// Restore original items
const dragContainer = document.querySelector('.drag-container');
dragContainer.innerHTML = `
<div class="draggable-item" draggable="true" data-item="apple">🍎 Apple</div>
<div class="draggable-item" draggable="true" data-item="banana">🍌 Banana</div>
<div class="draggable-item" draggable="true" data-item="orange">🍊 Orange</div>
`;
// Re-add event listeners
dragContainer.querySelectorAll('.draggable-item').forEach(item => {
item.addEventListener('dragstart', handleDragStart);
item.addEventListener('dragend', handleDragEnd);
});
});
// Add CSS animation for fade out
const style = document.createElement('style');
style.textContent = `
@keyframes fadeOut {
from { opacity: 1; transform: scale(1); }
to { opacity: 0; transform: scale(0.8); }
}
`;
document.head.appendChild(style);
</script>
File Upload with Drag and Drop
html
<div class="file-upload-demo">
<h3>File Upload with Drag & Drop</h3>
<div id="drop-area" class="drop-area">
<div class="upload-icon">📁</div>
<p>Drag and drop files here or <button id="file-select">Browse Files</button></p>
<input type="file" id="file-input" multiple accept="image/*,text/*,.pdf" style="display: none;">
<div class="upload-info">Supported: Images, Text files, PDFs</div>
</div>
<div id="file-list" class="file-list"></div>
<div class="upload-actions">
<button id="upload-files" style="display: none;">Upload Selected Files</button>
<button id="clear-files" style="display: none;">Clear All</button>
</div>
<div class="upload-progress" id="upload-progress" style="display: none;">
<div class="progress-bar">
<div class="progress-fill" id="progress-fill"></div>
</div>
<div class="progress-text" id="progress-text">0%</div>
</div>
</div>
<style>
.file-upload-demo {
padding: 20px;
border: 1px solid #ddd;
border-radius: 8px;
margin: 20px 0;
}
.drop-area {
border: 3px dashed #ccc;
border-radius: 12px;
padding: 40px;
text-align: center;
background: #fafafa;
transition: all 0.3s ease;
cursor: pointer;
}
.drop-area.drag-over {
border-color: #007bff;
background: #e3f2fd;
transform: scale(1.02);
}
.upload-icon {
font-size: 3em;
margin-bottom: 15px;
}
.drop-area p {
font-size: 1.1em;
color: #666;
margin: 10px 0;
}
#file-select {
background: #007bff;
color: white;
border: none;
padding: 8px 16px;
border-radius: 4px;
cursor: pointer;
}
.upload-info {
font-size: 0.9em;
color: #999;
margin-top: 10px;
}
.file-list {
margin: 20px 0;
max-height: 300px;
overflow-y: auto;
}
.file-item {
display: flex;
align-items: center;
gap: 15px;
padding: 12px;
border: 1px solid #e0e0e0;
border-radius: 8px;
margin-bottom: 8px;
background: white;
}
.file-icon {
font-size: 1.5em;
min-width: 30px;
}
.file-info {
flex: 1;
}
.file-name {
font-weight: bold;
color: #333;
}
.file-size {
font-size: 0.9em;
color: #666;
}
.file-preview {
max-width: 60px;
max-height: 60px;
border-radius: 4px;
}
.file-remove {
background: #dc3545;
color: white;
border: none;
padding: 5px 10px;
border-radius: 4px;
cursor: pointer;
font-size: 0.8em;
}
.upload-actions {
margin: 20px 0;
text-align: center;
}
.upload-actions button {
margin: 0 10px;
padding: 10px 20px;
border: none;
border-radius: 4px;
cursor: pointer;
}
#upload-files {
background: #28a745;
color: white;
}
#clear-files {
background: #6c757d;
color: white;
}
.upload-progress {
margin: 20px 0;
}
.progress-bar {
width: 100%;
height: 20px;
background: #e0e0e0;
border-radius: 10px;
overflow: hidden;
margin-bottom: 8px;
}
.progress-fill {
height: 100%;
background: linear-gradient(90deg, #007bff, #0056b3);
border-radius: 10px;
width: 0%;
transition: width 0.3s ease;
}
.progress-text {
text-align: center;
font-weight: bold;
color: #333;
}
</style>
<script>
class FileUploader {
constructor() {
this.files = [];
this.initializeElements();
this.addEventListeners();
}
initializeElements() {
this.dropArea = document.getElementById('drop-area');
this.fileInput = document.getElementById('file-input');
this.fileSelect = document.getElementById('file-select');
this.fileList = document.getElementById('file-list');
this.uploadBtn = document.getElementById('upload-files');
this.clearBtn = document.getElementById('clear-files');
this.progressContainer = document.getElementById('upload-progress');
this.progressFill = document.getElementById('progress-fill');
this.progressText = document.getElementById('progress-text');
}
addEventListeners() {
// Drop area events
this.dropArea.addEventListener('dragover', this.handleDragOver.bind(this));
this.dropArea.addEventListener('dragenter', this.handleDragEnter.bind(this));
this.dropArea.addEventListener('dragleave', this.handleDragLeave.bind(this));
this.dropArea.addEventListener('drop', this.handleDrop.bind(this));
this.dropArea.addEventListener('click', () => this.fileInput.click());
// File input events
this.fileInput.addEventListener('change', this.handleFileSelect.bind(this));
this.fileSelect.addEventListener('click', (e) => {
e.stopPropagation();
this.fileInput.click();
});
// Button events
this.uploadBtn.addEventListener('click', this.uploadFiles.bind(this));
this.clearBtn.addEventListener('click', this.clearFiles.bind(this));
}
handleDragOver(e) {
e.preventDefault();
e.dataTransfer.dropEffect = 'copy';
}
handleDragEnter(e) {
e.preventDefault();
this.dropArea.classList.add('drag-over');
}
handleDragLeave(e) {
e.preventDefault();
this.dropArea.classList.remove('drag-over');
}
handleDrop(e) {
e.preventDefault();
this.dropArea.classList.remove('drag-over');
const files = Array.from(e.dataTransfer.files);
this.addFiles(files);
}
handleFileSelect(e) {
const files = Array.from(e.target.files);
this.addFiles(files);
e.target.value = ''; // Reset input
}
addFiles(newFiles) {
// Filter and validate files
const validFiles = newFiles.filter(file => {
if (file.size > 10 * 1024 * 1024) { // 10MB limit
alert(`File "${file.name}" is too large (max 10MB)`);
return false;
}
// Check if file already exists
const exists = this.files.some(existingFile =>
existingFile.name === file.name && existingFile.size === file.size
);
if (exists) {
alert(`File "${file.name}" already added`);
return false;
}
return true;
});
this.files.push(...validFiles);
this.updateFileList();
this.updateButtons();
}
updateFileList() {
this.fileList.innerHTML = '';
this.files.forEach((file, index) => {
const fileItem = document.createElement('div');
fileItem.className = 'file-item';
const fileIcon = this.getFileIcon(file.type);
const fileSize = this.formatFileSize(file.size);
fileItem.innerHTML = `
<div class="file-icon">${fileIcon}</div>
<div class="file-info">
<div class="file-name">${file.name}</div>
<div class="file-size">${fileSize}</div>
</div>
${file.type.startsWith('image/') ?
`<img class="file-preview" src="${URL.createObjectURL(file)}" alt="Preview">` :
''}
<button class="file-remove" onclick="fileUploader.removeFile(${index})">Remove</button>
`;
this.fileList.appendChild(fileItem);
});
}
getFileIcon(mimeType) {
if (mimeType.startsWith('image/')) return '🖼️';
if (mimeType.startsWith('text/')) return '📄';
if (mimeType === 'application/pdf') return '📕';
if (mimeType.startsWith('video/')) return '🎥';
if (mimeType.startsWith('audio/')) return '🎵';
return '📁';
}
formatFileSize(bytes) {
if (bytes === 0) return '0 Bytes';
const k = 1024;
const sizes = ['Bytes', 'KB', 'MB', 'GB'];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
}
removeFile(index) {
this.files.splice(index, 1);
this.updateFileList();
this.updateButtons();
}
updateButtons() {
const hasFiles = this.files.length > 0;
this.uploadBtn.style.display = hasFiles ? 'inline-block' : 'none';
this.clearBtn.style.display = hasFiles ? 'inline-block' : 'none';
}
clearFiles() {
this.files = [];
this.updateFileList();
this.updateButtons();
this.hideProgress();
}
async uploadFiles() {
if (this.files.length === 0) return;
this.showProgress();
this.uploadBtn.disabled = true;
try {
for (let i = 0; i < this.files.length; i++) {
const file = this.files[i];
await this.uploadFile(file);
const progress = ((i + 1) / this.files.length) * 100;
this.updateProgress(progress);
}
alert('All files uploaded successfully!');
this.clearFiles();
} catch (error) {
alert('Upload failed: ' + error.message);
} finally {
this.uploadBtn.disabled = false;
this.hideProgress();
}
}
async uploadFile(file) {
// Simulate file upload with delay
return new Promise((resolve) => {
setTimeout(() => {
console.log(`Uploaded: ${file.name}`);
resolve();
}, 1000);
});
// Real upload implementation would look like:
/*
const formData = new FormData();
formData.append('file', file);
const response = await fetch('/upload', {
method: 'POST',
body: formData
});
if (!response.ok) {
throw new Error('Upload failed');
}
return response.json();
*/
}
showProgress() {
this.progressContainer.style.display = 'block';
this.updateProgress(0);
}
hideProgress() {
this.progressContainer.style.display = 'none';
}
updateProgress(percentage) {
this.progressFill.style.width = percentage + '%';
this.progressText.textContent = Math.round(percentage) + '%';
}
}
// Initialize file uploader
const fileUploader = new FileUploader();
</script>
Advanced Drag and Drop Features
html
<div class="advanced-drag-drop">
<h3>Advanced Drag & Drop Features</h3>
<!-- Sortable list -->
<div class="sortable-container">
<h4>Sortable Task List</h4>
<ul id="sortable-list" class="sortable-list">
<li class="sortable-item" draggable="true">
<span class="drag-handle">⋮⋮</span>
<span>Complete project proposal</span>
</li>
<li class="sortable-item" draggable="true">
<span class="drag-handle">⋮⋮</span>
<span>Review client feedback</span>
</li>
<li class="sortable-item" draggable="true">
<span class="drag-handle">⋮⋮</span>
<span>Update website design</span>
</li>
<li class="sortable-item" draggable="true">
<span class="drag-handle">⋮⋮</span>
<span>Prepare presentation slides</span>
</li>
</ul>
</div>
<!-- Kanban board -->
<div class="kanban-board">
<h4>Kanban Board</h4>
<div class="kanban-columns">
<div class="kanban-column" data-status="todo">
<h5>To Do</h5>
<div class="kanban-cards">
<div class="kanban-card" draggable="true" data-card="1">
<div class="card-title">Design Homepage</div>
<div class="card-meta">Due: Jan 25</div>
</div>
<div class="kanban-card" draggable="true" data-card="2">
<div class="card-title">Write Documentation</div>
<div class="card-meta">Due: Jan 30</div>
</div>
</div>
</div>
<div class="kanban-column" data-status="progress">
<h5>In Progress</h5>
<div class="kanban-cards">
<div class="kanban-card" draggable="true" data-card="3">
<div class="card-title">API Development</div>
<div class="card-meta">Started: Jan 20</div>
</div>
</div>
</div>
<div class="kanban-column" data-status="done">
<h5>Done</h5>
<div class="kanban-cards">
<div class="kanban-card" draggable="true" data-card="4">
<div class="card-title">Setup Development Environment</div>
<div class="card-meta">Completed: Jan 18</div>
</div>
</div>
</div>
</div>
</div>
</div>
<style>
.advanced-drag-drop {
padding: 20px;
border: 1px solid #ddd;
border-radius: 8px;
margin: 20px 0;
}
.sortable-container, .kanban-board {
margin: 20px 0;
padding: 15px;
background: #f8f9fa;
border-radius: 8px;
}
.sortable-list {
list-style: none;
padding: 0;
margin: 0;
}
.sortable-item {
display: flex;
align-items: center;
padding: 12px;
margin: 8px 0;
background: white;
border: 1px solid #dee2e6;
border-radius: 6px;
cursor: move;
transition: all 0.2s ease;
}
.sortable-item:hover {
box-shadow: 0 2px 4px rgba(0,0,0,0.1);
transform: translateY(-1px);
}
.sortable-item.dragging {
opacity: 0.5;
}
.drag-handle {
color: #6c757d;
margin-right: 10px;
cursor: grab;
}
.kanban-columns {
display: grid;
grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));
gap: 20px;
}
.kanban-column {
background: white;
border-radius: 8px;
padding: 15px;
border: 1px solid #e0e0e0;
}
.kanban-column h5 {
margin: 0 0 15px 0;
padding: 8px;
background: #f1f3f4;
border-radius: 4px;
text-align: center;
color: #333;
}
.kanban-cards {
min-height: 100px;
}
.kanban-card {
background: #fff;
border: 1px solid #d1d5db;
border-radius: 6px;
padding: 12px;
margin: 8px 0;
cursor: move;
transition: all 0.2s ease;
}
.kanban-card:hover {
box-shadow: 0 2px 8px rgba(0,0,0,0.15);
}
.kanban-card.dragging {
opacity: 0.5;
transform: rotate(3deg);
}
.kanban-column.drag-over {
background: #e3f2fd;
border-color: #2196f3;
}
.card-title {
font-weight: bold;
margin-bottom: 6px;
color: #333;
}
.card-meta {
font-size: 0.85em;
color: #666;
}
</style>
<script>
class AdvancedDragDrop {
constructor() {
this.setupSortableList();
this.setupKanbanBoard();
}
setupSortableList() {
const list = document.getElementById('sortable-list');
let draggedItem = null;
list.addEventListener('dragstart', (e) => {
if (e.target.classList.contains('sortable-item')) {
draggedItem = e.target;
e.target.classList.add('dragging');
e.dataTransfer.effectAllowed = 'move';
}
});
list.addEventListener('dragend', (e) => {
if (e.target.classList.contains('sortable-item')) {
e.target.classList.remove('dragging');
draggedItem = null;
}
});
list.addEventListener('dragover', (e) => {
e.preventDefault();
const afterElement = this.getDragAfterElement(list, e.clientY);
if (afterElement == null) {
list.appendChild(draggedItem);
} else {
list.insertBefore(draggedItem, afterElement);
}
});
}
getDragAfterElement(container, y) {
const draggableElements = [...container.querySelectorAll('.sortable-item:not(.dragging)')];
return draggableElements.reduce((closest, child) => {
const box = child.getBoundingClientRect();
const offset = y - box.top - box.height / 2;
if (offset < 0 && offset > closest.offset) {
return { offset: offset, element: child };
} else {
return closest;
}
}, { offset: Number.NEGATIVE_INFINITY }).element;
}
setupKanbanBoard() {
const columns = document.querySelectorAll('.kanban-column');
const cards = document.querySelectorAll('.kanban-card');
let draggedCard = null;
// Add drag events to cards
cards.forEach(card => {
card.addEventListener('dragstart', (e) => {
draggedCard = e.target;
e.target.classList.add('dragging');
e.dataTransfer.setData('text/plain', e.target.dataset.card);
});
card.addEventListener('dragend', (e) => {
e.target.classList.remove('dragging');
draggedCard = null;
});
});
// Add drop events to columns
columns.forEach(column => {
column.addEventListener('dragover', (e) => {
e.preventDefault();
e.dataTransfer.dropEffect = 'move';
});
column.addEventListener('dragenter', (e) => {
e.preventDefault();
column.classList.add('drag-over');
});
column.addEventListener('dragleave', (e) => {
if (!column.contains(e.relatedTarget)) {
column.classList.remove('drag-over');
}
});
column.addEventListener('drop', (e) => {
e.preventDefault();
column.classList.remove('drag-over');
if (draggedCard) {
const cardsContainer = column.querySelector('.kanban-cards');
cardsContainer.appendChild(draggedCard);
// Update card status based on column
const newStatus = column.dataset.status;
console.log(`Card ${draggedCard.dataset.card} moved to ${newStatus}`);
}
});
});
}
}
// Initialize advanced drag and drop
const advancedDragDrop = new AdvancedDragDrop();
</script>
Integration Examples
Copy-Paste with Drag-Drop
html
<div class="integration-demo">
<h3>Copy-Paste + Drag-Drop Integration</h3>
<div class="demo-areas">
<div class="content-area">
<h4>Content Creator</h4>
<textarea id="content-input" placeholder="Create some content...">Sample text content</textarea>
<button id="copy-content">📋 Copy Content</button>
</div>
<div class="workspace">
<h4>Workspace</h4>
<div id="workspace-area" class="workspace-area">
<div class="workspace-item" draggable="true">
📝 Draggable Note 1
</div>
<div class="workspace-item" draggable="true">
📝 Draggable Note 2
</div>
</div>
<button id="paste-content">📄 Paste Here</button>
<button id="add-from-clipboard">➕ Add from Clipboard</button>
</div>
</div>
</div>
<style>
.integration-demo {
padding: 20px;
border: 1px solid #ddd;
border-radius: 8px;
margin: 20px 0;
}
.demo-areas {
display: grid;
grid-template-columns: 1fr 1fr;
gap: 20px;
}
.content-area, .workspace {
padding: 15px;
border: 1px solid #e0e0e0;
border-radius: 8px;
}
#content-input {
width: 100%;
height: 100px;
margin-bottom: 10px;
padding: 10px;
border: 1px solid #ccc;
border-radius: 4px;
}
.workspace-area {
min-height: 150px;
border: 2px dashed #ccc;
border-radius: 8px;
padding: 15px;
margin-bottom: 10px;
}
.workspace-item {
display: inline-block;
padding: 8px 12px;
margin: 4px;
background: #e3f2fd;
border: 1px solid #2196f3;
border-radius: 4px;
cursor: move;
}
.workspace-item.dragging {
opacity: 0.5;
}
</style>
<script>
class IntegratedDemo {
constructor() {
this.setupElements();
this.addEventListeners();
}
setupElements() {
this.contentInput = document.getElementById('content-input');
this.copyBtn = document.getElementById('copy-content');
this.pasteBtn = document.getElementById('paste-content');
this.addBtn = document.getElementById('add-from-clipboard');
this.workspace = document.getElementById('workspace-area');
}
addEventListeners() {
this.copyBtn.addEventListener('click', this.copyContent.bind(this));
this.pasteBtn.addEventListener('click', this.pasteContent.bind(this));
this.addBtn.addEventListener('click', this.addFromClipboard.bind(this));
// Drag and drop for workspace items
this.workspace.addEventListener('dragstart', this.handleDragStart.bind(this));
this.workspace.addEventListener('dragover', this.handleDragOver.bind(this));
this.workspace.addEventListener('drop', this.handleDrop.bind(this));
}
async copyContent() {
try {
await navigator.clipboard.writeText(this.contentInput.value);
this.showStatus('Content copied to clipboard!');
} catch (err) {
console.error('Copy failed:', err);
}
}
async pasteContent() {
try {
const text = await navigator.clipboard.readText();
this.contentInput.value = text;
this.showStatus('Content pasted from clipboard!');
} catch (err) {
console.error('Paste failed:', err);
}
}
async addFromClipboard() {
try {
const text = await navigator.clipboard.readText();
if (text.trim()) {
const item = document.createElement('div');
item.className = 'workspace-item';
item.draggable = true;
item.textContent = '📋 ' + text.substring(0, 30) + (text.length > 30 ? '...' : '');
this.workspace.appendChild(item);
this.showStatus('Added item from clipboard!');
}
} catch (err) {
console.error('Add from clipboard failed:', err);
}
}
handleDragStart(e) {
if (e.target.classList.contains('workspace-item')) {
e.target.classList.add('dragging');
e.dataTransfer.setData('text/plain', e.target.textContent);
}
}
handleDragOver(e) {
e.preventDefault();
}
handleDrop(e) {
e.preventDefault();
const draggedItem = this.workspace.querySelector('.dragging');
if (draggedItem) {
draggedItem.classList.remove('dragging');
// Reorder logic could be added here
}
}
showStatus(message) {
// Simple status display
console.log(message);
// You could add a toast notification here
const toast = document.createElement('div');
toast.textContent = message;
toast.style.cssText = `
position: fixed;
top: 20px;
right: 20px;
background: #28a745;
color: white;
padding: 10px 15px;
border-radius: 4px;
z-index: 1000;
`;
document.body.appendChild(toast);
setTimeout(() => toast.remove(), 3000);
}
}
const integratedDemo = new IntegratedDemo();
</script>