cx 24.8.6 → 24.9.0

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.
@@ -1,55 +1,55 @@
1
- @mixin cx-textfield(
2
- $name: "textfield",
3
- $state-style-map: $cx-std-field-state-style-map,
4
- $placeholder: $cx-input-placeholder,
5
- $empty-text: $cx-empty-text,
6
- $left-icon-state-style-map: $cx-input-left-icon-state-style-map,
7
- $clear-state-style-map: $cx-clear-state-style-map,
8
- $icon-size: $cx-default-input-icon-size,
9
- $width: $cx-default-input-width,
10
- $besm: $cx-besm
11
- ) {
12
- $block: map-get($besm, block);
13
- $element: map-get($besm, element);
14
- $state: map-get($besm, state);
15
- $mod: map-get($besm, mod);
16
-
17
- .#{$block}#{$name} {
18
- @include cxb-field($besm, $state-style-map, $width: $width, $input: true);
19
- }
20
-
21
- $padding: cx-get-state-rule($state-style-map, default, "padding");
22
-
23
- .#{$element}#{$name}-input {
24
- @include cxe-field-input($besm, $state-style-map, $placeholder: $placeholder);
25
-
26
- .#{$state}icon > & {
27
- padding-left: cx-calc(2 * cx-left($padding), $icon-size);
28
- }
29
-
30
- .#{$state}clear > & {
31
- padding-right: cx-calc(cx-right($padding), $cx-default-clear-size, $cx-default-clear-spacing);
32
- }
33
- }
34
-
35
- .#{$element}#{$name}-left-icon {
36
- @include cxe-field-button($besm, $left-icon-state-style-map);
37
- pointer-events: none;
38
- }
39
-
40
- .#{$element}#{$name}-clear {
41
- @include cxe-field-button($besm, $clear-state-style-map);
42
- }
43
-
44
- .#{$element}#{$name}-icon {
45
- @include cxe-field-button-icon($besm, $icon-size);
46
- }
47
-
48
- .#{$element}#{$name}-empty-text {
49
- @include cxe-field-empty-text($empty-text);
50
- }
51
- }
52
-
53
- @if (cx-should-include("cx/widgets/TextField")) {
54
- @include cx-textfield();
55
- }
1
+ @mixin cx-textfield(
2
+ $name: "textfield",
3
+ $state-style-map: $cx-std-field-state-style-map,
4
+ $placeholder: $cx-input-placeholder,
5
+ $empty-text: $cx-empty-text,
6
+ $left-icon-state-style-map: $cx-input-left-icon-state-style-map,
7
+ $clear-state-style-map: $cx-clear-state-style-map,
8
+ $icon-size: $cx-default-input-icon-size,
9
+ $width: $cx-default-input-width,
10
+ $besm: $cx-besm
11
+ ) {
12
+ $block: map-get($besm, block);
13
+ $element: map-get($besm, element);
14
+ $state: map-get($besm, state);
15
+ $mod: map-get($besm, mod);
16
+
17
+ .#{$block}#{$name} {
18
+ @include cxb-field($besm, $state-style-map, $width: $width, $input: true);
19
+ }
20
+
21
+ $padding: cx-get-state-rule($state-style-map, default, "padding");
22
+
23
+ .#{$element}#{$name}-input {
24
+ @include cxe-field-input($besm, $state-style-map, $placeholder: $placeholder);
25
+
26
+ .#{$state}icon > & {
27
+ padding-left: cx-calc(2 * cx-left($padding), $icon-size);
28
+ }
29
+
30
+ .#{$state}clear > & {
31
+ padding-right: cx-calc(cx-right($padding), $cx-default-clear-size, $cx-default-clear-spacing);
32
+ }
33
+ }
34
+
35
+ .#{$element}#{$name}-left-icon {
36
+ pointer-events: none;
37
+ @include cxe-field-button($besm, $left-icon-state-style-map);
38
+ }
39
+
40
+ .#{$element}#{$name}-clear {
41
+ @include cxe-field-button($besm, $clear-state-style-map);
42
+ }
43
+
44
+ .#{$element}#{$name}-icon {
45
+ @include cxe-field-button-icon($besm, $icon-size);
46
+ }
47
+
48
+ .#{$element}#{$name}-empty-text {
49
+ @include cxe-field-empty-text($empty-text);
50
+ }
51
+ }
52
+
53
+ @if (cx-should-include("cx/widgets/TextField")) {
54
+ @include cx-textfield();
55
+ }
@@ -1,89 +1,88 @@
1
- @mixin cx-treenode(
2
- $name: "treenode",
3
- $icon-size: $cx-default-icon-size,
4
- $handle-size: $cx-default-input-tool-size,
5
- $handle-state-style-map: $cx-input-right-icon-state-style-map,
6
- $besm: $cx-besm
7
- ) {
8
- $block: map-get($besm, block);
9
- $element: map-get($besm, element);
10
- $state: map-get($besm, state);
11
- $mod: map-get($besm, mod);
12
-
13
- .#{$block}#{$name} {
14
- padding-left: $handle-size + $icon-size * 0.25;
15
-
16
- &.#{$state}icon {
17
- padding-left: $handle-size + $icon-size + $icon-size * 0.25;
18
- }
19
-
20
- position: relative;
21
-
22
- @for $i from 1 through 20 {
23
- &.#{$state}level-#{$i} {
24
- margin-left: #{($i * 20)}px;
25
- }
26
- }
27
- }
28
-
29
- .#{$element}#{$name}-handle {
30
- position: absolute;
31
-
32
- @include cx-add-state-rules($handle-state-style-map, default);
33
-
34
- left: 0;
35
- top: 50%;
36
- margin-top: -$handle-size * 0.5;
37
- bottom: 0;
38
- width: $handle-size + $icon-size;
39
- height: $handle-size;
40
-
41
- &:hover {
42
- @include cx-add-state-rules($handle-state-style-map, hover);
43
- }
44
-
45
- &:active {
46
- @include cx-add-state-rules($handle-state-style-map, active);
47
- }
48
- }
49
-
50
- .#{$element}#{$name}-arrow {
51
- display: block;
52
- position: absolute;
53
- left: 0;
54
- top: 50%;
55
- margin-top: -$icon-size * 0.5;
56
- margin-left: ($handle-size - $icon-size) * 0.5;
57
- height: $icon-size;
58
- width: $icon-size;
59
- line-height: $icon-size;
60
- transform: rotate(-90deg);
61
- transition: transform 0.2s;
62
-
63
- .#{$state}expanded & {
64
- transform: rotate(0);
65
- }
66
- }
67
-
68
- .#{$element}#{$name}-icon {
69
- display: block;
70
- position: absolute;
71
- left: 0;
72
- top: 50%;
73
- margin-top: -$icon-size * 0.5;
74
- margin-left: ($handle-size + $icon-size) * 0.5;
75
- height: $icon-size;
76
- width: $icon-size;
77
- line-height: $icon-size;
78
- }
79
-
80
- .#{$block}grid.#{$mod}tree {
81
- td {
82
- border-top: none;
83
- }
84
- }
85
- }
86
-
87
- @if (cx-should-include("cx/widgets/TreeNode")) {
88
- @include cx-treenode();
89
- }
1
+ @mixin cx-treenode(
2
+ $name: "treenode",
3
+ $icon-size: $cx-default-icon-size,
4
+ $handle-size: $cx-default-input-tool-size,
5
+ $handle-state-style-map: $cx-input-right-icon-state-style-map,
6
+ $besm: $cx-besm
7
+ ) {
8
+ $block: map-get($besm, block);
9
+ $element: map-get($besm, element);
10
+ $state: map-get($besm, state);
11
+ $mod: map-get($besm, mod);
12
+
13
+ .#{$block}#{$name} {
14
+ padding-left: $handle-size + $icon-size * 0.25;
15
+ position: relative;
16
+
17
+ &.#{$state}icon {
18
+ padding-left: $handle-size + $icon-size + $icon-size * 0.25;
19
+ }
20
+
21
+ @for $i from 1 through 20 {
22
+ &.#{$state}level-#{$i} {
23
+ margin-left: #{($i * 20)}px;
24
+ }
25
+ }
26
+ }
27
+
28
+ .#{$element}#{$name}-handle {
29
+ position: absolute;
30
+
31
+ @include cx-add-state-rules($handle-state-style-map, default);
32
+
33
+ left: 0;
34
+ top: 50%;
35
+ margin-top: -$handle-size * 0.5;
36
+ bottom: 0;
37
+ width: $handle-size + $icon-size;
38
+ height: $handle-size;
39
+
40
+ &:hover {
41
+ @include cx-add-state-rules($handle-state-style-map, hover);
42
+ }
43
+
44
+ &:active {
45
+ @include cx-add-state-rules($handle-state-style-map, active);
46
+ }
47
+ }
48
+
49
+ .#{$element}#{$name}-arrow {
50
+ display: block;
51
+ position: absolute;
52
+ left: 0;
53
+ top: 50%;
54
+ margin-top: -$icon-size * 0.5;
55
+ margin-left: ($handle-size - $icon-size) * 0.5;
56
+ height: $icon-size;
57
+ width: $icon-size;
58
+ line-height: $icon-size;
59
+ transform: rotate(-90deg);
60
+ transition: transform 0.2s;
61
+
62
+ .#{$state}expanded & {
63
+ transform: rotate(0);
64
+ }
65
+ }
66
+
67
+ .#{$element}#{$name}-icon {
68
+ display: block;
69
+ position: absolute;
70
+ left: 0;
71
+ top: 50%;
72
+ margin-top: -$icon-size * 0.5;
73
+ margin-left: ($handle-size + $icon-size) * 0.5;
74
+ height: $icon-size;
75
+ width: $icon-size;
76
+ line-height: $icon-size;
77
+ }
78
+
79
+ .#{$block}grid.#{$mod}tree {
80
+ td {
81
+ border-top: none;
82
+ }
83
+ }
84
+ }
85
+
86
+ @if (cx-should-include("cx/widgets/TreeNode")) {
87
+ @include cx-treenode();
88
+ }
@@ -1,74 +1,74 @@
1
- @mixin cx-menu($name: "menu", $state-style-map: $cx-menu-state-style-map, $besm: $cx-besm) {
2
- $block: map-get($besm, block);
3
- $element: map-get($besm, element);
4
- $state: map-get($besm, state);
5
- $mod: map-get($besm, mod);
6
-
7
- .#{$block}#{$name} {
8
- box-sizing: border-box;
9
- margin: 0;
10
- padding: 0;
11
- list-style: none;
12
-
13
- &.#{$state}horizontal {
14
- white-space: nowrap;
15
-
16
- & > .#{$element}#{$name}-item {
17
- display: inline-block;
18
- user-select: none;
19
-
20
- @each $size, $value in $cx-menu-padding-options {
21
- &.#{$state}size-#{$size} > .#{$mod}#{$name} {
22
- margin: 0 cx-right($value) 0 cx-left($value);
23
- }
24
- }
25
- }
26
- }
27
-
28
- @each $size, $value in $cx-menu-padding-options {
29
- &.#{$state}vertical.#{$state}#{$size}-item-padding .#{$mod}#{$name} {
30
- margin: $value;
31
- display: block;
32
- }
33
- }
34
-
35
- @include cx-add-state-rules($state-style-map, default);
36
-
37
- .#{$element}#{$name}-item {
38
- & > hr {
39
- border: none;
40
- border-top: 1px solid $cx-default-menu-separator-color;
41
- margin: map-get($cx-menu-padding-options, medium);
42
- }
43
-
44
- &.#{$state}hidden {
45
- visibility: hidden;
46
- position: absolute;
47
- }
48
- }
49
-
50
- &.#{$state}overflow {
51
- overflow: hidden;
52
- display: flex;
53
-
54
- & > .#{$element}#{$name}-item:last-child {
55
- visibility: hidden;
56
- position: absolute;
57
- max-height: 100%;
58
- }
59
-
60
- &.#{$state}pack > .#{$element}#{$name}-item:last-child {
61
- visibility: visible;
62
- position: relative;
63
- }
64
- }
65
- }
66
-
67
- .#{$element}#{$name}-spacer {
68
- flex: 1 1 0%;
69
- }
70
- }
71
-
72
- @if (cx-should-include("cx/widgets/Menu")) {
73
- @include cx-menu();
74
- }
1
+ @mixin cx-menu($name: "menu", $state-style-map: $cx-menu-state-style-map, $besm: $cx-besm) {
2
+ $block: map-get($besm, block);
3
+ $element: map-get($besm, element);
4
+ $state: map-get($besm, state);
5
+ $mod: map-get($besm, mod);
6
+
7
+ .#{$block}#{$name} {
8
+ box-sizing: border-box;
9
+ margin: 0;
10
+ padding: 0;
11
+ list-style: none;
12
+
13
+ @include cx-add-state-rules($state-style-map, default);
14
+
15
+ &.#{$state}horizontal {
16
+ white-space: nowrap;
17
+
18
+ & > .#{$element}#{$name}-item {
19
+ display: inline-block;
20
+ user-select: none;
21
+
22
+ @each $size, $value in $cx-menu-padding-options {
23
+ &.#{$state}size-#{$size} > .#{$mod}#{$name} {
24
+ margin: 0 cx-right($value) 0 cx-left($value);
25
+ }
26
+ }
27
+ }
28
+ }
29
+
30
+ @each $size, $value in $cx-menu-padding-options {
31
+ &.#{$state}vertical.#{$state}#{$size}-item-padding .#{$mod}#{$name} {
32
+ margin: $value;
33
+ display: block;
34
+ }
35
+ }
36
+
37
+ .#{$element}#{$name}-item {
38
+ & > hr {
39
+ border: none;
40
+ border-top: 1px solid $cx-default-menu-separator-color;
41
+ margin: map-get($cx-menu-padding-options, medium);
42
+ }
43
+
44
+ &.#{$state}hidden {
45
+ visibility: hidden;
46
+ position: absolute;
47
+ }
48
+ }
49
+
50
+ &.#{$state}overflow {
51
+ overflow: hidden;
52
+ display: flex;
53
+
54
+ & > .#{$element}#{$name}-item:last-child {
55
+ visibility: hidden;
56
+ position: absolute;
57
+ max-height: 100%;
58
+ }
59
+
60
+ &.#{$state}pack > .#{$element}#{$name}-item:last-child {
61
+ visibility: visible;
62
+ position: relative;
63
+ }
64
+ }
65
+ }
66
+
67
+ .#{$element}#{$name}-spacer {
68
+ flex: 1 1 0%;
69
+ }
70
+ }
71
+
72
+ @if (cx-should-include("cx/widgets/Menu")) {
73
+ @include cx-menu();
74
+ }
@@ -1,36 +1,39 @@
1
- import { Widget } from "../../ui/Widget";
2
- import { VDOM } from "../../ui/VDOM";
3
- import { tooltipMouseMove, tooltipMouseLeave } from "./tooltip-ops";
4
- import { closest, isObject } from "../../util";
5
-
6
- export class FlyweightTooltipTracker extends Widget {
7
- initInstance(context, instance) {
8
- let handler = (e) => this.handleMouseMove(e, instance);
9
- document.addEventListener("mousemove", handler);
10
- instance.subscribeOnDestroy(() => {
11
- document.removeEventListener("mousemove", handler);
12
- });
13
- }
14
-
15
- render(context, instance, key) {
16
- return null;
17
- }
18
-
19
- handleMouseMove(e, instance) {
20
- if (!this.onGetTooltip) return;
21
- if (instance.lastTarget != e.target) {
22
- instance.lastTarget = e.target;
23
- let tooltip = null;
24
- instance.parentEl = closest(e.target, (element) => {
25
- tooltip = instance.invoke("onGetTooltip", element, instance);
26
- if (tooltip) return true;
27
- });
28
- instance.tooltip = tooltip;
29
- }
30
- if (!instance.parentEl) tooltipMouseMove(e, instance, null);
31
- else
32
- tooltipMouseMove(e, instance, instance.tooltip, {
33
- target: instance.parentEl,
34
- });
35
- }
36
- }
1
+ import { Widget } from "../../ui/Widget";
2
+ import { VDOM } from "../../ui/VDOM";
3
+ import { tooltipMouseMove, tooltipMouseLeave } from "./tooltip-ops";
4
+ import { closest, isObject } from "../../util";
5
+
6
+ export class FlyweightTooltipTracker extends Widget {
7
+ initInstance(context, instance) {
8
+ let handler = (e) => this.handleMouseMove(e, instance);
9
+ document.addEventListener("mousemove", handler);
10
+ instance.subscribeOnDestroy(() => {
11
+ document.removeEventListener("mousemove", handler);
12
+ });
13
+ }
14
+
15
+ render(context, instance, key) {
16
+ return null;
17
+ }
18
+
19
+ handleMouseMove(e, instance) {
20
+ if (!this.onGetTooltip) return;
21
+ let parentEl, tooltip;
22
+ if (instance.lastTarget == e.target) return;
23
+
24
+ instance.lastTarget = e.target;
25
+ parentEl = closest(e.target, (element) => {
26
+ tooltip = instance.invoke("onGetTooltip", element, instance);
27
+ if (tooltip) return true;
28
+ });
29
+
30
+ if (!parentEl) tooltipMouseLeave(e, instance, instance.tooltip, { target: instance.parentEl });
31
+ else {
32
+ instance.tooltip = tooltip;
33
+ instance.parentEl = parentEl;
34
+ tooltipMouseMove(e, instance, instance.tooltip, {
35
+ target: parentEl,
36
+ });
37
+ }
38
+ }
39
+ }
@@ -166,7 +166,7 @@ export function getTooltipInstance(e, parentInstance, tooltip, options = {}) {
166
166
  if (
167
167
  tooltipInstance &&
168
168
  (tooltipInstance.widget.relatedElement != target ||
169
- tooltipInstance.config != tooltip ||
169
+ !shallowEquals(tooltipInstance.config, tooltip) ||
170
170
  tooltipInstance.store.store != parentInstance.store)
171
171
  ) {
172
172
  if (tooltipInstance.dismissTooltip) tooltipInstance.dismissTooltip();
@@ -189,6 +189,7 @@ export function getTooltipInstance(e, parentInstance, tooltip, options = {}) {
189
189
  });
190
190
  tooltipInstance = parentInstance.tooltips[name] = parentInstance.getDetachedChild(tooltipWidget, name, store);
191
191
  tooltipInstance.config = tooltip;
192
+ tooltipInstance.tooltipName = name;
192
193
 
193
194
  if (tooltip.alwaysVisible || tooltip.trackMouse || tooltip.trackMouseX || tooltip.trackMouseY) {
194
195
  tooltipInstance.init(new RenderingContext());
@@ -219,6 +220,8 @@ function tooltipMouseMove(e, parentInstance, tooltip, options = {}) {
219
220
  dismiss();
220
221
  });
221
222
  instance.dismissTooltip = () => {
223
+ if (instance.parent.tooltips[instance.tooltipName] === instance)
224
+ delete instance.parent.tooltips[instance.tooltipName];
222
225
  unsubscribeDismiss();
223
226
  dismiss();
224
227
  };
@@ -244,7 +247,9 @@ function tooltipMouseMove(e, parentInstance, tooltip, options = {}) {
244
247
 
245
248
  function tooltipMouseLeave(e, parentInstance, tooltip, options) {
246
249
  let instance = getTooltipInstance(e, parentInstance, tooltip, options);
247
- if (instance) {
250
+
251
+ // do not process leave events twice even if called multiple times
252
+ if (instance && instance.mouseOverTarget) {
248
253
  instance.mouseOverTarget = false;
249
254
  instance.widget.handleMouseLeavesParent(instance);
250
255
  }