[-] Split color picker

This commit is contained in:
Azalea Gui
2023-03-09 15:22:27 -05:00
parent 58462de365
commit ac9bea5fd2
3 changed files with 1 additions and 313 deletions
+1 -2
View File
@@ -18,8 +18,7 @@
"tg-blog": "^1.1.0",
"vue": "^3.2.47",
"vue-i18n": "^9.2.2",
"vue-router": "^4.1.6",
"vue3-colorpicker": "^2.1.2"
"vue-router": "^4.1.6"
},
"devDependencies": {
"@types/jquery": "^3.5.16",
-285
View File
@@ -1,285 +0,0 @@
<template>
<div id="ColorPicker" ref="el" @keydown.esc="close">
<div id="title-colors" class="fbox-h">
<div class="text" @mousedown="windowDrag" @click.right.alt="close">Colors</div>
<div class="close fbox-vcenter" @click="close" v-if="showClose"><i class="fas fa-times"></i></div>
<input v-model="colorInput" spellcheck="false" @change="colorModel = colorInput">
</div>
<ColorPicker id="picker" :isWidget="true" pickerType="chrome" v-model:pureColor="colorModel"
:disableHistory="true" @pureColorChange="change" format="hex8"/>
<div id="palette">
<div class="row" v-for="(p, i) of palette" :key="i">
<div class="color" v-for="(c, j) in p" :key="j" :style="{'background-color': c ? c : '#333'}"
@click.exact="setPalette(i, j)"
@click.right.alt="(e) => altClickPalette(e, i, j)"
@click.right.exact="(e) => rightClickPalette(e, i, j)"
draggable="true" @dragstart="paletteDragStart(i, j)" @drop="(e) => dropPalette(e, i, j)"
@dragenter="(e) => paletteDragEnter(e, i, j)" @dragover="(e) => paletteDragOver(e, i, j)"/>
</div>
</div>
</div>
</template>
<script lang="ts">
import "vue3-colorpicker/style.css";
import {ColorPicker} from "vue3-colorpicker";
import {range} from "@/scripts/utils";
import {Emit, Model, Prop, Ref} from "vue-property-decorator";
import {Options, Vue} from "vue-class-component";
/**
* Mouse Usage:
* - Alt + Right-click the title to close
* - Left-click on a color block to add color to palette
* - Left-drag on a color block to move color
* - Right-click on a color block to select color
* - Alt + Right-click on a color block to remove color
*/
@Options({components: {ColorPicker}})
export default class MyColorPicker extends Vue
{
@Model('color') color!: string // Color prop in #ffffffff format
colorModel = '' // Color model in #ffffffff format
colorInput = '' // Color input in ffffff format
palette: string[][] = []
showClose = false
@Ref() el!: HTMLElement
@Prop({default: null}) initialPos?: {x: number, y: number}
/**
* Init
*/
created(): void
{
this.colorModel = this.color
this.colorInput = this.colorModel.substr(1, 6)
const storedPalette = localStorage.getItem('palette')
this.palette = !storedPalette ? range(6).map(_ => range(10).map(_ => '')) : JSON.parse(storedPalette);
this.storePalette()
}
mounted(): void
{
if (this.initialPos) this.setPos(this.initialPos.x, this.initialPos.y)
}
/**
* Color change
*/
@Emit('update:color')
change(color: string): string
{
this.colorInput = color.substr(1, 6)
return this.colorModel
}
/**
* Set window position
*/
setPos(x: number, y: number): void
{
this.el.style.left = x + 'px'
this.el.style.top = y + 'px'
}
/**
* Window dragging
*/
windowDrag(e: MouseEvent): void
{
e.preventDefault()
let lastX = e.clientX, lastY = e.clientY
const mousemove = (e: MouseEvent) =>
{
const dx = lastX - e.clientX, dy = lastY - e.clientY
lastX = e.clientX; lastY = e.clientY
this.setPos(this.el.offsetLeft - dx, this.el.offsetTop - dy)
}
const mouseup = () => {document.removeEventListener('mouseup', mouseup); document.removeEventListener('mousemove', mousemove)}
document.addEventListener('mouseup', mouseup)
document.addEventListener('mousemove', mousemove)
}
@Emit('updatePalette')
storePalette(): string[][]
{
localStorage.setItem('palette', JSON.stringify(this.palette))
return this.palette
}
@Emit()
close(e?: Event): void
{
if (e) e.preventDefault()
console.log('Color picker close')
}
/**
* Left click to override
*/
setPalette(i: number, j: number): void
{
this.palette[i][j] = this.colorModel
this.storePalette()
}
/**
* Right click to select
*/
rightClickPalette(e: Event, i: number, j: number): void
{
e.preventDefault()
if (!this.palette[i][j]) return
this.colorModel = this.palette[i][j]
this.change(this.colorModel)
}
/**
* Alt right click to remove
*/
altClickPalette(e: Event, i: number, j: number): void
{
e.preventDefault()
this.palette[i][j] = ''
this.storePalette()
}
dragging = {i: 0, j: 0}
paletteDragStart(i: number, j: number): void
{
this.dragging = {i, j}
}
dropPalette(e: DragEvent, i: number, j: number): void
{
// We can assume that toI != fromI
const fromI = this.dragging.i * 10 + this.dragging.j
const toI = i * 10 + j
const incr = toI > fromI ? 1 : -1
const currentColor = this.palette[this.dragging.i][this.dragging.j]
for (let index of range(fromI, toI))
{
const col = index % 10, row = Math.floor(index / 10)
const lastI = index + incr
const lastC = lastI % 10, lastR = Math.floor(lastI / 10)
console.log(lastR, lastC, 'TO', row, col)
this.palette[row][col] = this.palette[lastR][lastC]
}
this.palette[i][j] = currentColor
this.storePalette()
}
paletteDragEnter(e: DragEvent, i: number, j: number): void
{
// TODO: Drag preview
console.log('Drag enter')
console.log(e)
console.log('' + i + ' ' + j)
}
paletteDragOver(e: DragEvent, i: number, j: number): void
{
// Only allow drag if it's not dragging onto itself
if (!(i == this.dragging.i && j == this.dragging.j)) e.preventDefault()
}
}
</script>
<style lang="sass">
#ColorPicker
position: absolute
border-radius: 20px
overflow: hidden
$cp-color: #4d4d4d
$txt-color: #cbcbcb
background-color: $cp-color
box-shadow: 0 0 10px #00000026
color: $txt-color
#title-colors
margin: 8px 0
font-size: 1.2em
padding: 0 15px
div.close
font-size: 10px
margin-right: 10px
div.text
flex-grow: 1
text-align: left
user-select: none
input
background-color: lighten($cp-color, 6)
font-family: monospace
color: $txt-color
border: none
padding: 0 10px
width: 60px
text-align: center
border-radius: 8px
transition: all 0.25s ease
input:focus-visible
outline: none
background-color: lighten($cp-color, 10)
.vc-colorpicker
width: 300px
background-color: transparent
box-shadow: none
padding-bottom: 0
border-radius: 0
.vc-colorpicker--container
padding: 0 3px
.vc-chrome-colorPicker, .vc-chrome-colorPicker-body
background-color: transparent
.vc-display
display: none
.vc-saturation, .vc-saturation__white, .vc-saturation__black
border-radius: 5px
.vc-saturation
height: 200px
.vc-chrome-colorPicker-body
margin-left: 10px
margin-right: 10px
#palette
width: 300px
// Transparency texture
background-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAwAAAAMCAIAAADZF8uwAAAAGUlEQVQYV2M4gwH+YwCGIasIUwhT25BVBADtzYNYrHvv4gAAAABJRU5ErkJggg==)
background-repeat: repeat
.row
width: 100%
display: flex
.color:first-child
border-left: none
.row:first-child .color
border-top: none
.color
width: 30px
height: 30px
box-sizing: border-box
border-left: 1px solid $cp-color
border-top: 1px solid $cp-color
flex-grow: 1
</style>
-26
View File
@@ -1,26 +0,0 @@
<template>
<div id="Projects">
<MyColorPicker v-model:color="color"></MyColorPicker>
<button @click="log"></button>
</div>
</template>
<script lang="ts">
import {Options, Vue} from 'vue-class-component';
import MyColorPicker from "@/components/color/ColorPicker.vue";
@Options({components: {MyColorPicker}})
export default class Projects extends Vue
{
color = '#ffffff'
log(): void
{
console.log(this.color)
}
}
</script>
<style lang="sass" scoped>
</style>