@operato/input 8.0.0-alpha.0 → 8.0.0-alpha.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -2,7 +2,7 @@
2
2
  "name": "@operato/input",
3
3
  "description": "Webcomponents for input following open-wc recommendations",
4
4
  "author": "heartyoh@hatiolab.com",
5
- "version": "8.0.0-alpha.0",
5
+ "version": "8.0.0-alpha.10",
6
6
  "main": "dist/src/index.js",
7
7
  "module": "dist/src/index.js",
8
8
  "license": "MIT",
@@ -57,6 +57,7 @@
57
57
  "./ox-input-mass-fraction.js": "./dist/src/ox-input-mass-fraction.js",
58
58
  "./ox-input-textarea.js": "./dist/src/ox-input-textarea.js",
59
59
  "./ox-input-direction.js": "./dist/src/ox-input-direction.js",
60
+ "./ox-input-signature.js": "./dist/src/ox-input-signature.js",
60
61
  "./ox-input-table-column-config.js": "./dist/src/ox-input-table-column-config.js"
61
62
  },
62
63
  "typesVersions": {
@@ -175,6 +176,9 @@
175
176
  "./ox-input-direction.js": [
176
177
  "./dist/src/ox-input-direction.d.ts"
177
178
  ],
179
+ "./ox-input-signature.js": [
180
+ "./dist/src/ox-input-signature.d.ts"
181
+ ],
178
182
  "./ox-input-table-column-config.js": [
179
183
  "./dist/src/ox-input-table-column-config.d.ts"
180
184
  ]
@@ -208,8 +212,8 @@
208
212
  "@material/web": "^2.0.0",
209
213
  "@operato/color-picker": "^8.0.0-alpha.0",
210
214
  "@operato/i18n": "^8.0.0-alpha.0",
211
- "@operato/popup": "^8.0.0-alpha.0",
212
- "@operato/styles": "^8.0.0-alpha.0",
215
+ "@operato/popup": "^8.0.0-alpha.4",
216
+ "@operato/styles": "^8.0.0-alpha.4",
213
217
  "@operato/utils": "^8.0.0-alpha.0",
214
218
  "@polymer/paper-dropdown-menu": "^3.2.0",
215
219
  "@polymer/paper-item": "^3.0.1",
@@ -254,5 +258,5 @@
254
258
  "prettier --write"
255
259
  ]
256
260
  },
257
- "gitHead": "094233a9ce34f6922283581c088d5c4395ba54d0"
261
+ "gitHead": "f937b76c9831c46e26acde8e94cc71514068a961"
258
262
  }
package/src/index.ts CHANGED
@@ -31,3 +31,4 @@ export * from './ox-input-textarea.js'
31
31
  export * from './ox-input-direction.js'
32
32
  export * from './ox-input-table-column-config.js'
33
33
  export * from './ox-input-search.js'
34
+ export * from './ox-input-signature.js'
@@ -79,12 +79,13 @@ export class OxInputFile extends OxFormField {
79
79
  border-bottom: var(--file-uploader-li-border-bottom);
80
80
  font-size: var(--md-sys-typescale-label-large-size, 0.875rem);
81
81
  color: var(--md-sys-color-on-primary-container);
82
+ display: flex;
83
+ align-items: center;
82
84
  }
83
85
  li md-icon {
84
- float: right;
85
86
  cursor: pointer;
86
- margin: var(--file-uploader-li-icon-margin);
87
87
  font-size: var(--icon-size-small);
88
+ margin-left: auto;
88
89
  }
89
90
  li md-icon:hover,
90
91
  li md-icon:active {
@@ -0,0 +1,180 @@
1
+ import '@material/web/icon/icon.js'
2
+
3
+ import { css, html, nothing } from 'lit'
4
+ import { customElement, property, query, state } from 'lit/decorators.js'
5
+
6
+ import { OxFormField } from './ox-form-field.js'
7
+
8
+ @customElement('ox-input-signature')
9
+ export class OxInputSignature extends OxFormField {
10
+ static styles = [
11
+ css`
12
+ :host {
13
+ display: flex;
14
+ flex-direction: column;
15
+ min-height: var(--signature-min-height, 80px);
16
+ min-width: var(--signature-min-width, 120px);
17
+
18
+ background-color: var(--signature-background-color, white);
19
+
20
+ overflow: hidden;
21
+ }
22
+
23
+ .signature-preview {
24
+ flex: 1;
25
+ align-self: stretch;
26
+
27
+ border: 1px solid var(--md-sys-color-outline);
28
+ background-size: contain;
29
+ background-repeat: no-repeat;
30
+ background-position: center;
31
+ }
32
+
33
+ dialog canvas {
34
+ width: 100%;
35
+ height: 100%;
36
+ border: 1px solid var(--md-sys-color-outline);
37
+ }
38
+
39
+ .controls {
40
+ margin-top: 10px;
41
+ display: flex;
42
+ flex-direction: row;
43
+ gap: var(--spacing-medium);
44
+ }
45
+
46
+ .filler {
47
+ flex: 1;
48
+ }
49
+ `
50
+ ]
51
+
52
+ @property({ type: String }) value: string | null = null
53
+
54
+ @query('.signature-preview')
55
+ private previewDiv!: HTMLDivElement
56
+ @query('dialog')
57
+ private dialog!: HTMLDialogElement
58
+ @query('canvas')
59
+ private canvas!: HTMLCanvasElement
60
+
61
+ private ctx!: CanvasRenderingContext2D
62
+ private isDrawing = false
63
+
64
+ render() {
65
+ return html`
66
+ <div class="signature-preview" @click=${this.openDialog}></div>
67
+
68
+ <dialog>
69
+ <canvas
70
+ width="800"
71
+ height="400"
72
+ @mousedown=${this.startDrawing}
73
+ @mouseup=${this.stopDrawing}
74
+ @mousemove=${this.draw}
75
+ @mouseleave=${this.stopDrawing}
76
+ @touchstart=${this.startDrawing}
77
+ @touchend=${this.stopDrawing}
78
+ @touchmove=${this.draw}
79
+ ></canvas>
80
+ <div class="controls">
81
+ <button @click="${this.clearCanvas}">Clear</button>
82
+ <div class="filler"></div>
83
+ <button @click="${this.saveSignature}">Save</button>
84
+ <button @click="${this.closeDialog}">Close</button>
85
+ </div>
86
+ </dialog>
87
+ `
88
+ }
89
+
90
+ firstUpdated() {
91
+ this.ctx = this.canvas.getContext('2d')!
92
+ this.ctx.strokeStyle = '#000'
93
+ this.ctx.lineWidth = 2
94
+
95
+ // 처음 로딩 시 서명 데이터를 미리보기 div에 표시
96
+ if (this.value) {
97
+ this.loadSignature(this.value)
98
+ }
99
+ }
100
+
101
+ openDialog() {
102
+ if (this.disabled) return
103
+ this.dialog.showModal()
104
+
105
+ // 다이아로그가 열릴 때 현재 value를 캔버스에 그리기
106
+ if (this.value) {
107
+ const img = new Image()
108
+ img.onload = () => {
109
+ this.ctx.drawImage(img, 0, 0, this.canvas.width, this.canvas.height)
110
+ }
111
+ img.src = this.value
112
+ }
113
+ }
114
+
115
+ closeDialog() {
116
+ this.dialog.close()
117
+ }
118
+
119
+ startDrawing(event: MouseEvent | TouchEvent) {
120
+ if (this.disabled) {
121
+ return
122
+ }
123
+
124
+ this.isDrawing = true
125
+ this.ctx.beginPath()
126
+ const position = this.getEventPosition(event)
127
+ this.ctx.moveTo(position.x, position.y)
128
+ }
129
+
130
+ stopDrawing() {
131
+ this.isDrawing = false
132
+ }
133
+
134
+ draw(event: MouseEvent | TouchEvent) {
135
+ if (!this.isDrawing) return
136
+ const position = this.getEventPosition(event)
137
+ this.ctx.lineTo(position.x, position.y)
138
+ this.ctx.stroke()
139
+ }
140
+
141
+ clearCanvas() {
142
+ this.ctx.clearRect(0, 0, this.canvas.width, this.canvas.height)
143
+ }
144
+
145
+ saveSignature() {
146
+ this.value = this.canvas.toDataURL()
147
+ this._notifyChange()
148
+ this.previewDiv.style.backgroundImage = `url(${this.value})`
149
+ this.closeDialog()
150
+ }
151
+
152
+ loadSignature(dataUrl: string) {
153
+ this.previewDiv.style.backgroundImage = `url(${dataUrl})`
154
+ }
155
+
156
+ getEventPosition(event: MouseEvent | TouchEvent) {
157
+ const rect = this.canvas.getBoundingClientRect()
158
+
159
+ // 캔버스의 실제 크기와 CSS 크기 간의 비율을 계산
160
+ const scaleX = this.canvas.width / rect.width
161
+ const scaleY = this.canvas.height / rect.height
162
+
163
+ // 실제 좌표 계산
164
+ const isTouchEvent = event instanceof TouchEvent
165
+ const x = (isTouchEvent ? event.touches[0].clientX - rect.left : (event as MouseEvent).clientX - rect.left) * scaleX
166
+ const y = (isTouchEvent ? event.touches[0].clientY - rect.top : (event as MouseEvent).clientY - rect.top) * scaleY
167
+
168
+ return { x, y }
169
+ }
170
+
171
+ _notifyChange() {
172
+ this.dispatchEvent(
173
+ new CustomEvent('change', {
174
+ bubbles: true,
175
+ composed: true,
176
+ detail: this.value
177
+ })
178
+ )
179
+ }
180
+ }
@@ -0,0 +1,75 @@
1
+ import '../src/ox-input-signature.js'
2
+
3
+ import { html, TemplateResult } from 'lit'
4
+ import { styles as MDTypeScaleStyles } from '@material/web/typography/md-typescale-styles'
5
+
6
+ export default {
7
+ title: 'ox-input-signature',
8
+ component: 'ox-input-signature',
9
+ argTypes: {
10
+ disabled: { control: 'boolean' },
11
+ value: { control: 'text' }
12
+ }
13
+ }
14
+
15
+ interface Story<T> {
16
+ (args: T): TemplateResult
17
+ args?: Partial<T>
18
+ argTypes?: Record<string, unknown>
19
+ }
20
+
21
+ interface ArgTypes {
22
+ value?: string
23
+ disabled?: boolean
24
+ }
25
+
26
+ const Template: Story<ArgTypes> = ({ value, disabled }: ArgTypes) => html`
27
+ <link href="https://fonts.googleapis.com/css2?family=Roboto:wght@400;500;700&display=swap" rel="stylesheet" />
28
+
29
+ <link href="/themes/light.css" rel="stylesheet" />
30
+ <link href="/themes/dark.css" rel="stylesheet" />
31
+ <link href="/themes/spacing.css" rel="stylesheet" />
32
+
33
+ <link
34
+ href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL@20..48,100..700,0..1"
35
+ rel="stylesheet"
36
+ />
37
+ <link
38
+ href="https://fonts.googleapis.com/css2?family=Material+Symbols+Rounded:opsz,wght,FILL@20..48,100..700,0..1"
39
+ rel="stylesheet"
40
+ />
41
+ <link
42
+ href="https://fonts.googleapis.com/css2?family=Material+Symbols+Sharp:opsz,wght,FILL@20..48,100..700,0..1"
43
+ rel="stylesheet"
44
+ />
45
+
46
+ <style>
47
+ ${MDTypeScaleStyles.cssText}
48
+ </style>
49
+
50
+ <style>
51
+ .container {
52
+ height: 600px;
53
+ text-align: center;
54
+ padding: 20px;
55
+
56
+ background-color: var(--md-sys-color-primary-container);
57
+ color: var(--md-sys-color-on-primary-container);
58
+ }
59
+ </style>
60
+
61
+ <div class="container md-typescale-body-large-prominent">
62
+ <ox-input-signature
63
+ .value=${value || ''}
64
+ ?disabled=${disabled}
65
+ @change=${(e: Event) => console.log(((e as CustomEvent).target as any)!.value)}
66
+ >
67
+ </ox-input-signature>
68
+ </div>
69
+ `
70
+
71
+ export const Regular = Template.bind({})
72
+ Regular.args = {
73
+ value:
74
+ ''
75
+ }