@operato/input 8.0.0-alpha.1 → 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.1",
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",
@@ -212,8 +212,8 @@
212
212
  "@material/web": "^2.0.0",
213
213
  "@operato/color-picker": "^8.0.0-alpha.0",
214
214
  "@operato/i18n": "^8.0.0-alpha.0",
215
- "@operato/popup": "^8.0.0-alpha.0",
216
- "@operato/styles": "^8.0.0-alpha.0",
215
+ "@operato/popup": "^8.0.0-alpha.4",
216
+ "@operato/styles": "^8.0.0-alpha.4",
217
217
  "@operato/utils": "^8.0.0-alpha.0",
218
218
  "@polymer/paper-dropdown-menu": "^3.2.0",
219
219
  "@polymer/paper-item": "^3.0.1",
@@ -258,5 +258,5 @@
258
258
  "prettier --write"
259
259
  ]
260
260
  },
261
- "gitHead": "7e2725314b1cb79c18f27b5a8db9dccd2461a1a3"
261
+ "gitHead": "f937b76c9831c46e26acde8e94cc71514068a961"
262
262
  }
@@ -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 {
@@ -1,7 +1,7 @@
1
1
  import '@material/web/icon/icon.js'
2
2
 
3
3
  import { css, html, nothing } from 'lit'
4
- import { customElement, property } from 'lit/decorators.js'
4
+ import { customElement, property, query, state } from 'lit/decorators.js'
5
5
 
6
6
  import { OxFormField } from './ox-form-field.js'
7
7
 
@@ -10,83 +10,112 @@ export class OxInputSignature extends OxFormField {
10
10
  static styles = [
11
11
  css`
12
12
  :host {
13
- position: relative;
14
- box-sizing: border-box;
15
-
16
13
  display: flex;
17
14
  flex-direction: column;
18
- place-content: center;
19
- border-radius: var(--border-radius);
20
- padding: var(--padding-default, 9px);
21
- min-height: 100px;
22
- text-transform: capitalize;
15
+ min-height: var(--signature-min-height, 80px);
16
+ min-width: var(--signature-min-width, 120px);
23
17
 
24
- border: var(--file-uploader-border);
25
- background-color: var(--md-sys-color-background);
26
- font: var(--file-uploader-font) !important;
27
- color: var(--file-uploader-color);
18
+ background-color: var(--signature-background-color, white);
28
19
 
29
20
  overflow: hidden;
30
21
  }
31
22
 
32
- canvas {
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 {
33
34
  width: 100%;
34
- border: 1px solid #000;
35
+ height: 100%;
36
+ border: 1px solid var(--md-sys-color-outline);
35
37
  }
36
38
 
37
39
  .controls {
38
40
  margin-top: 10px;
41
+ display: flex;
42
+ flex-direction: row;
43
+ gap: var(--spacing-medium);
44
+ }
45
+
46
+ .filler {
47
+ flex: 1;
39
48
  }
40
49
  `
41
50
  ]
42
51
 
43
- @property({ type: Boolean }) isDrawing = false
44
52
  @property({ type: String }) value: string | null = null
45
53
 
46
- private ctx!: CanvasRenderingContext2D
54
+ @query('.signature-preview')
55
+ private previewDiv!: HTMLDivElement
56
+ @query('dialog')
57
+ private dialog!: HTMLDialogElement
58
+ @query('canvas')
47
59
  private canvas!: HTMLCanvasElement
48
60
 
61
+ private ctx!: CanvasRenderingContext2D
62
+ private isDrawing = false
63
+
49
64
  render() {
50
65
  return html`
51
- <canvas
52
- width="400"
53
- height="200"
54
- @mousedown=${this.startDrawing}
55
- @mouseup=${this.stopDrawing}
56
- @mousemove=${this.draw}
57
- @mouseleave=${this.stopDrawing}
58
- @touchstart=${this.startDrawing}
59
- @touchend=${this.stopDrawing}
60
- @touchmove=${this.draw}
61
- ></canvas>
62
-
63
- ${!this.disabled
64
- ? html` <div class="controls">
65
- <button @click="${this.clearCanvas}">Clear</button>
66
- <button @click="${this.saveSignature}">Save</button>
67
- </div>`
68
- : nothing}
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>
69
87
  `
70
88
  }
71
89
 
72
90
  firstUpdated() {
73
- this.canvas = this.shadowRoot!.querySelector('canvas')!
74
91
  this.ctx = this.canvas.getContext('2d')!
75
92
  this.ctx.strokeStyle = '#000'
76
93
  this.ctx.lineWidth = 2
77
94
 
78
- // 처음 로딩 시 서명 데이터를 캔버스에 표시
95
+ // 처음 로딩 시 서명 데이터를 미리보기 div에 표시
79
96
  if (this.value) {
80
97
  this.loadSignature(this.value)
81
98
  }
82
99
  }
83
100
 
84
- updated(changedProperties: Map<string | number | symbol, unknown>) {
85
- if (changedProperties.has('value') && this.value) {
86
- this.loadSignature(this.value)
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
87
112
  }
88
113
  }
89
114
 
115
+ closeDialog() {
116
+ this.dialog.close()
117
+ }
118
+
90
119
  startDrawing(event: MouseEvent | TouchEvent) {
91
120
  if (this.disabled) {
92
121
  return
@@ -116,15 +145,12 @@ export class OxInputSignature extends OxFormField {
116
145
  saveSignature() {
117
146
  this.value = this.canvas.toDataURL()
118
147
  this._notifyChange()
148
+ this.previewDiv.style.backgroundImage = `url(${this.value})`
149
+ this.closeDialog()
119
150
  }
120
151
 
121
152
  loadSignature(dataUrl: string) {
122
- const image = new Image()
123
- image.onload = () => {
124
- this.ctx.clearRect(0, 0, this.canvas.width, this.canvas.height) // 이전 내용을 지움
125
- this.ctx.drawImage(image, 0, 0) // 이미지를 캔버스에 그림
126
- }
127
- image.src = dataUrl
153
+ this.previewDiv.style.backgroundImage = `url(${dataUrl})`
128
154
  }
129
155
 
130
156
  getEventPosition(event: MouseEvent | TouchEvent) {
@@ -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
+ }
@@ -1,56 +0,0 @@
1
- import '../src/ox-input-signature.js';
2
- import { html } from 'lit';
3
- import { styles as MDTypeScaleStyles } from '@material/web/typography/md-typescale-styles';
4
- export default {
5
- title: 'ox-input-signature',
6
- component: 'ox-input-signature',
7
- argTypes: {
8
- disabled: { control: 'boolean' }
9
- }
10
- };
11
- const Template = ({ label = 'signature', name = 'signature', disabled }) => html `
12
- <link href="https://fonts.googleapis.com/css2?family=Roboto:wght@400;500;700&display=swap" rel="stylesheet" />
13
-
14
- <link href="/themes/light.css" rel="stylesheet" />
15
- <link href="/themes/dark.css" rel="stylesheet" />
16
- <link href="/themes/spacing.css" rel="stylesheet" />
17
-
18
- <link
19
- href="https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL@20..48,100..700,0..1"
20
- rel="stylesheet"
21
- />
22
- <link
23
- href="https://fonts.googleapis.com/css2?family=Material+Symbols+Rounded:opsz,wght,FILL@20..48,100..700,0..1"
24
- rel="stylesheet"
25
- />
26
- <link
27
- href="https://fonts.googleapis.com/css2?family=Material+Symbols+Sharp:opsz,wght,FILL@20..48,100..700,0..1"
28
- rel="stylesheet"
29
- />
30
-
31
- <style>
32
- ${MDTypeScaleStyles.cssText}
33
- </style>
34
-
35
- <style>
36
- .container {
37
- height: 500px;
38
- text-align: center;
39
- padding: 20px;
40
-
41
- background-color: var(--md-sys-color-primary-container);
42
- color: var(--md-sys-color-on-primary-container);
43
- }
44
- </style>
45
-
46
- <div class="container md-typescale-body-large-prominent">
47
- <ox-input-signature
48
- ?disabled=${disabled}
49
- @change=${(e) => console.log(e.target.value)}
50
- >
51
- </ox-input-signature>
52
- </div>
53
- `;
54
- export const Regular = Template.bind({});
55
- Regular.args = {};
56
- //# sourceMappingURL=ox-input-%08signature.stories.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"ox-input-\bsignature.stories.js","sourceRoot":"","sources":["../../stories/ox-input-\bsignature.stories.ts"],"names":[],"mappings":"AAAA,OAAO,8BAA8B,CAAA;AAErC,OAAO,EAAE,IAAI,EAAkB,MAAM,KAAK,CAAA;AAC1C,OAAO,EAAE,MAAM,IAAI,iBAAiB,EAAE,MAAM,8CAA8C,CAAA;AAE1F,eAAe;IACb,KAAK,EAAE,oBAAoB;IAC3B,SAAS,EAAE,oBAAoB;IAC/B,QAAQ,EAAE;QACR,QAAQ,EAAE,EAAE,OAAO,EAAE,SAAS,EAAE;KACjC;CACF,CAAA;AAcD,MAAM,QAAQ,GAAoB,CAAC,EAAE,KAAK,GAAG,WAAW,EAAE,IAAI,GAAG,WAAW,EAAE,QAAQ,EAAY,EAAE,EAAE,CAAC,IAAI,CAAA;;;;;;;;;;;;;;;;;;;;;MAqBrG,iBAAiB,CAAC,OAAO;;;;;;;;;;;;;;;;kBAgBb,QAAQ;gBACV,CAAC,CAAQ,EAAE,EAAE,CAAC,OAAO,CAAC,GAAG,CAAG,CAAiB,CAAC,MAAe,CAAC,KAAK,CAAC;;;;CAInF,CAAA;AAED,MAAM,CAAC,MAAM,OAAO,GAAG,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAA;AACxC,OAAO,CAAC,IAAI,GAAG,EAAE,CAAA","sourcesContent":["import '../src/ox-input-signature.js'\n\nimport { html, TemplateResult } from 'lit'\nimport { styles as MDTypeScaleStyles } from '@material/web/typography/md-typescale-styles'\n\nexport default {\n title: 'ox-input-signature',\n component: 'ox-input-signature',\n argTypes: {\n disabled: { control: 'boolean' }\n }\n}\n\ninterface Story<T> {\n (args: T): TemplateResult\n args?: Partial<T>\n argTypes?: Record<string, unknown>\n}\n\ninterface ArgTypes {\n label?: string\n name?: string\n disabled?: boolean\n}\n\nconst Template: Story<ArgTypes> = ({ label = 'signature', name = 'signature', disabled }: ArgTypes) => html`\n <link href=\"https://fonts.googleapis.com/css2?family=Roboto:wght@400;500;700&display=swap\" rel=\"stylesheet\" />\n\n <link href=\"/themes/light.css\" rel=\"stylesheet\" />\n <link href=\"/themes/dark.css\" rel=\"stylesheet\" />\n <link href=\"/themes/spacing.css\" rel=\"stylesheet\" />\n\n <link\n href=\"https://fonts.googleapis.com/css2?family=Material+Symbols+Outlined:opsz,wght,FILL@20..48,100..700,0..1\"\n rel=\"stylesheet\"\n />\n <link\n href=\"https://fonts.googleapis.com/css2?family=Material+Symbols+Rounded:opsz,wght,FILL@20..48,100..700,0..1\"\n rel=\"stylesheet\"\n />\n <link\n href=\"https://fonts.googleapis.com/css2?family=Material+Symbols+Sharp:opsz,wght,FILL@20..48,100..700,0..1\"\n rel=\"stylesheet\"\n />\n\n <style>\n ${MDTypeScaleStyles.cssText}\n </style>\n\n <style>\n .container {\n height: 500px;\n text-align: center;\n padding: 20px;\n\n background-color: var(--md-sys-color-primary-container);\n color: var(--md-sys-color-on-primary-container);\n }\n </style>\n\n <div class=\"container md-typescale-body-large-prominent\">\n <ox-input-signature\n ?disabled=${disabled}\n @change=${(e: Event) => console.log(((e as CustomEvent).target as any)!.value)}\n >\n </ox-input-signature>\n </div>\n`\n\nexport const Regular = Template.bind({})\nRegular.args = {}\n"]}
@@ -1,71 +0,0 @@
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
- }
12
- }
13
-
14
- interface Story<T> {
15
- (args: T): TemplateResult
16
- args?: Partial<T>
17
- argTypes?: Record<string, unknown>
18
- }
19
-
20
- interface ArgTypes {
21
- label?: string
22
- name?: string
23
- disabled?: boolean
24
- }
25
-
26
- const Template: Story<ArgTypes> = ({ label = 'signature', name = 'signature', 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: 500px;
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
- ?disabled=${disabled}
64
- @change=${(e: Event) => console.log(((e as CustomEvent).target as any)!.value)}
65
- >
66
- </ox-input-signature>
67
- </div>
68
- `
69
-
70
- export const Regular = Template.bind({})
71
- Regular.args = {}