4 Commits

Author SHA1 Message Date
a49ca8f98e 0.4.1
All checks were successful
Deploy to Production / deploy (push) Successful in 8s
2026-02-23 01:14:19 +00:00
afac47c634 chore: adjust panel background for modal 2026-02-23 01:14:10 +00:00
31015366be 0.4.0
All checks were successful
Deploy to Production / deploy (push) Successful in 9s
2026-02-23 01:09:36 +00:00
880d46be1c chore: tweak add-moves modal layout 2026-02-23 01:09:10 +00:00
4 changed files with 141 additions and 7 deletions

4
package-lock.json generated
View File

@@ -1,12 +1,12 @@
{ {
"name": "rubic-cube", "name": "rubic-cube",
"version": "0.3.1", "version": "0.4.1",
"lockfileVersion": 3, "lockfileVersion": 3,
"requires": true, "requires": true,
"packages": { "packages": {
"": { "": {
"name": "rubic-cube", "name": "rubic-cube",
"version": "0.3.1", "version": "0.4.1",
"dependencies": { "dependencies": {
"lucide-vue-next": "^0.564.0", "lucide-vue-next": "^0.564.0",
"rubiks-js": "^1.0.0", "rubiks-js": "^1.0.0",

View File

@@ -1,7 +1,7 @@
{ {
"name": "rubic-cube", "name": "rubic-cube",
"private": true, "private": true,
"version": "0.3.1", "version": "0.4.1",
"type": "module", "type": "module",
"scripts": { "scripts": {
"dev": "vite", "dev": "vite",

View File

@@ -271,6 +271,8 @@ const movesHistory = ref([])
const movesHistoryEl = ref(null) const movesHistoryEl = ref(null)
const samplePillEl = ref(null) const samplePillEl = ref(null)
const movesPerRow = ref(0) const movesPerRow = ref(0)
const isAddModalOpen = ref(false)
const addMovesText = ref('')
const displayMoves = computed(() => { const displayMoves = computed(() => {
const list = movesHistory.value.slice() const list = movesHistory.value.slice()
@@ -374,6 +376,47 @@ const resetQueue = () => {
nextTick(recalcMovesLayout) nextTick(recalcMovesLayout)
} }
const openAddModal = () => {
addMovesText.value = ''
isAddModalOpen.value = true
}
const closeAddModal = () => {
isAddModalOpen.value = false
}
const handleAddMoves = () => {
const text = addMovesText.value || ''
const tokens = text.split(/\s+/).filter(Boolean)
const moves = []
tokens.forEach((token) => {
const t = token.trim()
if (!t) return
const base = t[0]
if (!'UDLRFB'.includes(base)) return
const rest = t.slice(1)
let key = null
if (rest === '') key = base
else if (rest === '2') key = base + '2'
else if (rest === "'" || rest === '') key = base + '-prime'
if (key && MOVE_MAP[key]) {
moves.push(key)
}
})
moves.forEach((m) => applyMove(m))
addMovesText.value = ''
isAddModalOpen.value = false
}
const handleKeydown = (e) => {
if (e.key === 'Escape' && isAddModalOpen.value) {
e.preventDefault()
closeAddModal()
}
}
const getCubieStyle = (c) => { const getCubieStyle = (c) => {
// Base Position // Base Position
const x = c.x * (SCALE + GAP) const x = c.x * (SCALE + GAP)
@@ -593,6 +636,7 @@ onMounted(() => {
window.addEventListener('mousemove', onMouseMove) window.addEventListener('mousemove', onMouseMove)
window.addEventListener('mouseup', onMouseUp) window.addEventListener('mouseup', onMouseUp)
window.addEventListener('resize', recalcMovesLayout) window.addEventListener('resize', recalcMovesLayout)
window.addEventListener('keydown', handleKeydown)
nextTick(recalcMovesLayout) nextTick(recalcMovesLayout)
}) })
@@ -600,6 +644,7 @@ onUnmounted(() => {
window.removeEventListener('mousemove', onMouseMove) window.removeEventListener('mousemove', onMouseMove)
window.removeEventListener('mouseup', onMouseUp) window.removeEventListener('mouseup', onMouseUp)
window.removeEventListener('resize', recalcMovesLayout) window.removeEventListener('resize', recalcMovesLayout)
window.removeEventListener('keydown', handleKeydown)
}) })
watch(displayMoves, () => { watch(displayMoves, () => {
@@ -691,9 +736,42 @@ watch(displayMoves, () => {
</span> </span>
</div> </div>
</div> </div>
<div v-if="displayMoves.length" class="moves-actions"> <div class="moves-actions">
<button class="queue-action" @click="copyQueueToClipboard">copy</button> <button class="queue-action" @click="openAddModal">add</button>
<button class="queue-action" @click="resetQueue">reset</button> <button
v-if="displayMoves.length"
class="queue-action"
@click="copyQueueToClipboard"
>
copy
</button>
<button
v-if="displayMoves.length"
class="queue-action"
@click="resetQueue"
>
reset
</button>
</div>
</div>
<div
v-if="isAddModalOpen"
class="moves-modal-backdrop"
@click.self="closeAddModal"
>
<div class="moves-modal">
<textarea
v-model="addMovesText"
class="moves-modal-textarea"
/>
<div class="moves-modal-actions">
<button class="btn-neon move-btn moves-modal-button" @click="closeAddModal">
cancel
</button>
<button class="btn-neon move-btn moves-modal-button" @click="handleAddMoves">
add moves
</button>
</div>
</div> </div>
</div> </div>
</div> </div>
@@ -842,6 +920,62 @@ watch(displayMoves, () => {
box-shadow: none; box-shadow: none;
} }
.moves-modal-backdrop {
position: fixed;
inset: 0;
background: rgba(0, 0, 0, 0.65);
display: flex;
align-items: center;
justify-content: center;
z-index: 200;
}
.moves-modal {
background: var(--panel-bg);
border: 1px solid var(--panel-border);
color: var(--text-color);
border-radius: 10px;
padding: 24px;
min-width: 480px;
max-width: 800px;
box-shadow: 0 20px 40px rgba(0, 0, 0, 0.7);
}
.moves-modal-textarea {
width: 100%;
min-height: 220px;
background: var(--panel-bg);
color: var(--text-color);
box-sizing: border-box;
border-radius: 6px;
border: 1px solid var(--panel-border);
padding: 10px;
resize: vertical;
font-family: inherit;
font-size: 0.85rem;
}
.moves-modal-textarea:focus {
outline: none;
box-shadow: none;
}
.moves-modal-actions {
margin-top: 20px;
display: flex;
justify-content: flex-end;
gap: 12px;
}
.moves-modal-button {
font-size: 0.85rem;
}
.moves-modal-button:focus {
outline: none;
box-shadow: none;
}
/* Projection Styles */ /* Projection Styles */
.projections { .projections {
position: absolute; position: absolute;

View File

@@ -31,7 +31,7 @@
--toggle-btn-border: rgba(255, 255, 255, 0.2); --toggle-btn-border: rgba(255, 255, 255, 0.2);
--toggle-hover-border: #ffffff; --toggle-hover-border: #ffffff;
--toggle-active-shadow: 0 0 10px rgba(0, 242, 255, 0.3); --toggle-active-shadow: 0 0 10px rgba(0, 242, 255, 0.3);
--panel-bg: rgba(255, 255, 255, 0.1); --panel-bg: rgba(0, 0, 0, 0.4);
--panel-border: rgba(255, 255, 255, 0.1); --panel-border: rgba(255, 255, 255, 0.1);
--panel-shadow: 0 4px 15px rgba(0, 0, 0, 0.2); --panel-shadow: 0 4px 15px rgba(0, 0, 0, 0.2);
--button-bg: rgba(255, 255, 255, 0.1); --button-bg: rgba(255, 255, 255, 0.1);