diff --git a/css/activities.css b/css/activities.css index 1c09e1fec1..28312f0e8c 100644 --- a/css/activities.css +++ b/css/activities.css @@ -90,6 +90,72 @@ font-size: large; } +.trash-view { + position: relative; + background-color: white; + max-width: 396px; + max-height: 200px; + overflow-y: auto; + font-size: 16px; + color: black; + border: 2px solid #87cefa; + list-style-type: none; + margin: 0; + padding: 0; + text-align: left; +} + +.button-container { + position: sticky; + display: flex; + justify-content: space-between; + top: 0; + z-index: 10; + display: flex; + gap: 10px; + background: #2196F3; + margin: 0; + padding: 5px; + border-bottom: 1px solid #d9d9d9; +} + +.trash-item { + padding: 2px 12px; + margin: 1px 0; + border-radius: 4px; + transition: background-color 0.3s; +} + +.trash-item.hover { + background-color: #d9d9d9; +} + +.trash-item-icon { + width: 30px; + height: 30px; + margin-right: 10px; + vertical-align: middle; +} + +#restoreLastIcon, #restoreAllIcon { + display: flex; + align-items: center; + justify-content: center; + width: 48px; + height: 48px; + cursor: pointer; +} + + +.material-icons.md-48 { + font-size: 32px; +} + + +.hidden { + display: none; +} + .ui-menu { position: relative; background-color: rgba(255, 255, 255, 1); diff --git a/index.html b/index.html index aea8f6d80a..ff5c876b22 100644 --- a/index.html +++ b/index.html @@ -780,6 +780,7 @@ > space    +
  • restore_from_trash +
  • { if (!activity.blocks || !activity.blocks.trashStacks || activity.blocks.trashStacks.length === 0) { activity.textMsg( - _("Nothing in the trash to restore."), + _("Trash can is empty."), 3000 ); return; } - activity._restoreTrash(); + + if (docById("helpfulWheelDiv").style.display !== "none") { + docById("helpfulWheelDiv").style.display = "none"; + activity.__tick(); + } + }; + + const restoreTrashPop = (activity) => { + if (!activity.blocks || !activity.blocks.trashStacks || activity.blocks.trashStacks.length === 0) { + activity.textMsg( + _("Trash can is empty."), + 3000 + ); + return; + } + this._restoreTrashById(this.blocks.trashStacks[this.blocks.trashStacks.length - 1]); activity.textMsg( _("Item restored from the trash."), 3000 ); - + if (docById("helpfulWheelDiv").style.display !== "none") { docById("helpfulWheelDiv").style.display = "none"; activity.__tick(); } - }; + + } - this._restoreTrash = () => { + this._restoreTrashById = (blockId) => { + const blockIndex = this.blocks.trashStacks.indexOf(blockId); + if (blockIndex === -1) return; // Block not found in trash + + this.blocks.trashStacks.splice(blockIndex, 1); // Remove from trash + for (const name in this.palettes.dict) { this.palettes.dict[name].hideMenu(true); } - + this.blocks.activeBlock = null; this.refreshCanvas(); const dx = 0; const dy = -this.cellSize * 3; // Reposition - - if (this.blocks.trashStacks.length === 0) { - return; - } - - const thisBlock = this.blocks.trashStacks.pop(); - - // Restore drag group in trash - this.blocks.findDragGroup(thisBlock); + + // Restore drag group + this.blocks.findDragGroup(blockId); for (let b = 0; b < this.blocks.dragGroup.length; b++) { const blk = this.blocks.dragGroup[b]; this.blocks.blockList[blk].trash = false; this.blocks.moveBlockRelative(blk, dx, dy); this.blocks.blockList[blk].show(); } - - this.blocks.raiseStackToTop(thisBlock); - - if ( - this.blocks.blockList[thisBlock].name === "start" || - this.blocks.blockList[thisBlock].name === "drum" - ) { - const turtle = this.blocks.blockList[thisBlock].value; + + this.blocks.raiseStackToTop(blockId); + const restoredBlock = this.blocks.blockList[blockId]; + + if (restoredBlock.name === 'start' || restoredBlock.name === 'drum') { + const turtle = restoredBlock.value; this.turtles.turtleList[turtle].inTrash = false; this.turtles.turtleList[turtle].container.visible = true; - } else if (this.blocks.blockList[thisBlock].name === "action") { - // We need to add a palette entry for this action. - // But first we need to ensure we have a unqiue name, - // as the name could have been taken in the interim. - const actionArg = this.blocks.blockList[ - this.blocks.blockList[thisBlock].connections[1] - ]; + } else if (restoredBlock.name === 'action') { + const actionArg = this.blocks.blockList[restoredBlock.connections[1]]; if (actionArg !== null) { let label; const oldName = actionArg.value; - // Mark the action block as still being in the - // trash so that its name won't be considered when - // looking for a unique name. - this.blocks.blockList[thisBlock].trash = true; + restoredBlock.trash = true; const uniqueName = this.blocks.findUniqueActionName(oldName); - this.blocks.blockList[thisBlock].trash = false; + restoredBlock.trash = false; if (uniqueName !== actionArg) { actionArg.value = uniqueName; - - label = actionArg.value.toString(); - if (label.length > 8) { - label = label.substr(0, 7) + "..."; - } + label = uniqueName.length > 8 ? uniqueName.substr(0, 7) + '...' : uniqueName; actionArg.text.text = label; if (actionArg.label !== null) { actionArg.label.value = uniqueName; } - actionArg.container.updateCache(); - - // Check the drag group to ensure any do blocks are updated (in case of recursion). for (let b = 0; b < this.blocks.dragGroup.length; b++) { const me = this.blocks.blockList[this.blocks.dragGroup[b]]; if ( @@ -3415,11 +3413,7 @@ class Activity { ) { me.privateData = uniqueName; me.value = uniqueName; - - label = me.value.toString(); - if (label.length > 8) { - label = label.substr(0, 7) + "..."; - } + label = uniqueName.length > 8 ? uniqueName.substr(0, 7) + '...' : uniqueName; me.text.text = label; me.overrideName = label; me.regenerateArtwork(); @@ -3429,21 +3423,110 @@ class Activity { } } } - + activity.textMsg( + _("Item restored from the trash."), + 3000 + ); + this.refreshCanvas(); - }; + }; + + // Add event listener for trash icon click + document.getElementById('restoreIcon').addEventListener('click', () => { + this._renderTrashView(); + }); + + // function to hide trashView from canvas + function handleClickOutsideTrashView(trashView) { + let firstClick = true; + document.addEventListener('click', (event) => { + if (firstClick) { + firstClick = false; + return; + } + if (!trashView.contains(event.target) && event.target !== trashView) { + trashView.style.display = 'none'; + } + }); + } - this.handleKeyDown = (event) => { - - if (event.ctrlKey && event.key === "z") { - this._restoreTrash(activity); - activity.__tick(); - event.preventDefault(); + this._renderTrashView = () => { + if (!activity.blocks || !activity.blocks.trashStacks || activity.blocks.trashStacks.length === 0) { + return; } - }; - - // Attach keydown event listener to document - document.addEventListener("keydown", this.handleKeyDown); + const trashList = document.getElementById('trashList'); + const trashView = document.createElement('div'); + trashView.id = 'trashView'; + trashView.classList.add('trash-view'); + + // Sticky icons + const buttonContainer = document.createElement('div'); + buttonContainer.classList.add('button-container'); + + const restoreLastIcon = document.createElement('a'); + restoreLastIcon.id = 'restoreLastIcon'; + restoreLastIcon.classList.add('restore-last-icon'); + restoreLastIcon.innerHTML = 'restore_from_trash'; + restoreLastIcon.addEventListener('click', () => { + this._restoreTrashById(this.blocks.trashStacks[this.blocks.trashStacks.length - 1]); + trashView.classList.add('hidden'); + }); + + const restoreAllIcon = document.createElement('a'); + restoreAllIcon.id = 'restoreAllIcon'; + restoreAllIcon.classList.add('restore-all-icon'); + restoreAllIcon.innerHTML = 'delete_sweep'; + restoreAllIcon.addEventListener('click', () => { + while (this.blocks.trashStacks.length > 0) { + this._restoreTrashById(this.blocks.trashStacks[0]); + } + trashView.classList.add('hidden'); + }); + restoreLastIcon.setAttribute("title", _("Restore last item")); + restoreAllIcon.setAttribute("title", _("Restore all items")); + + buttonContainer.appendChild(restoreLastIcon); + buttonContainer.appendChild(restoreAllIcon); + trashView.appendChild(buttonContainer); + + // Render trash items + this.blocks.trashStacks.forEach((blockId) => { + const block = this.blocks.blockList[blockId]; + const listItem = document.createElement('div'); + listItem.classList.add('trash-item'); + + const svgData = block.artwork; + const encodedData = 'data:image/svg+xml;utf8,' + encodeURIComponent(svgData); + + const img = document.createElement('img'); + img.src = encodedData; + img.alt = 'Block Icon'; + img.classList.add('trash-item-icon'); + + const textNode = document.createTextNode(block.name); + + listItem.appendChild(img); + listItem.appendChild(textNode); + listItem.dataset.blockId = blockId; + + listItem.addEventListener('mouseover', () => listItem.classList.add('hover')); + listItem.addEventListener('mouseout', () => listItem.classList.remove('hover')); + listItem.addEventListener('click', () => { + this._restoreTrashById(blockId); + trashView.classList.add('hidden'); + }); + handleClickOutsideTrashView(trashView); + + trashView.appendChild(listItem); + }); + + const existingView = document.getElementById('trashView'); + if (existingView) { + trashList.replaceChild(trashView, existingView); + } else { + trashList.appendChild(trashView); + } + }; /* * Open aux menu @@ -5814,7 +5897,7 @@ class Activity { this.helpfulWheelItems.push({label: "Increase block size", icon: "imgsrc:data:image/svg+xml;base64," + window.btoa(base64Encode(BIGGERBUTTON)), display: true, fn: doLargerBlocks}); if (!this.helpfulWheelItems.find(ele => ele.label === "Restore")) - this.helpfulWheelItems.push({label: "Restore", icon: "imgsrc:header-icons/restore-from-trash.svg", display: true, fn: restoreTrash}); + this.helpfulWheelItems.push({label: "Restore", icon: "imgsrc:header-icons/restore-from-trash.svg", display: true, fn: restoreTrashPop}); if (!this.helpfulWheelItems.find(ele => ele.label === "Turtle Wrap Off")) this.helpfulWheelItems.push({label: "Turtle Wrap Off", icon: "imgsrc:header-icons/wrap-text.svg", display: true, fn: this.toolbar.changeWrap});