@rpg-engine/long-bow 0.4.7 → 0.4.8

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.
Files changed (177) hide show
  1. package/LICENSE +20 -20
  2. package/README.md +181 -181
  3. package/dist/components/ConfirmModal.d.ts +8 -0
  4. package/dist/components/DraggableContainer.d.ts +1 -0
  5. package/dist/components/Item/Inventory/ItemSlot.d.ts +7 -7
  6. package/dist/components/Marketplace/BuyPanel.d.ts +22 -0
  7. package/dist/components/Marketplace/ManagmentPanel.d.ts +18 -0
  8. package/dist/components/Marketplace/Marketplace.d.ts +22 -9
  9. package/dist/components/Marketplace/MarketplaceRows.d.ts +3 -1
  10. package/dist/components/Marketplace/{__mocks__ → filters}/index.d.ts +1 -3
  11. package/dist/components/Pager.d.ts +9 -0
  12. package/dist/long-bow.cjs.development.js +1540 -1163
  13. package/dist/long-bow.cjs.development.js.map +1 -1
  14. package/dist/long-bow.cjs.production.min.js +1 -1
  15. package/dist/long-bow.cjs.production.min.js.map +1 -1
  16. package/dist/long-bow.esm.js +1594 -1217
  17. package/dist/long-bow.esm.js.map +1 -1
  18. package/package.json +100 -100
  19. package/src/components/Abstractions/ModalPortal.tsx +22 -22
  20. package/src/components/Abstractions/SlotsContainer.tsx +62 -62
  21. package/src/components/Arrow/SelectArrow.tsx +69 -69
  22. package/src/components/Arrow/img/arrow01-left-clicked.png +0 -0
  23. package/src/components/Arrow/img/arrow01-left.png +0 -0
  24. package/src/components/Arrow/img/arrow01-right-clicked.png +0 -0
  25. package/src/components/Arrow/img/arrow01-right.png +0 -0
  26. package/src/components/Arrow/img/arrow02-left-clicked.png +0 -0
  27. package/src/components/Arrow/img/arrow02-left.png +0 -0
  28. package/src/components/Arrow/img/arrow02-right-clicked.png +0 -0
  29. package/src/components/Arrow/img/arrow02-right.png +0 -0
  30. package/src/components/Button.tsx +40 -40
  31. package/src/components/Character/CharacterSelection.tsx +98 -98
  32. package/src/components/CharacterStatus/CharacterStatus.tsx +120 -120
  33. package/src/components/Chat/Chat.tsx +196 -196
  34. package/src/components/Chatdeprecated/ChatDeprecated.tsx +198 -198
  35. package/src/components/CheckButton.tsx +65 -65
  36. package/src/components/CircularController/CircularController.tsx +282 -282
  37. package/src/components/ConfirmModal.tsx +87 -0
  38. package/src/components/CraftBook/CraftBook.tsx +286 -286
  39. package/src/components/CraftBook/CraftingRecipe.tsx +161 -161
  40. package/src/components/CraftBook/MockItems.ts +101 -101
  41. package/src/components/DraggableContainer.tsx +183 -180
  42. package/src/components/Dropdown.tsx +14 -6
  43. package/src/components/DropdownSelectorContainer.tsx +42 -42
  44. package/src/components/Equipment/EquipmentSet.tsx +199 -199
  45. package/src/components/HistoryDialog.tsx +104 -104
  46. package/src/components/Input.tsx +15 -15
  47. package/src/components/InputRadio.tsx +41 -41
  48. package/src/components/Item/Cards/ItemInfo.tsx +298 -298
  49. package/src/components/Item/Cards/ItemInfoDisplay.tsx +135 -135
  50. package/src/components/Item/Cards/ItemInfoWrapper.tsx +62 -62
  51. package/src/components/Item/Cards/ItemTooltip.tsx +83 -83
  52. package/src/components/Item/Cards/MobileItemTooltip.tsx +149 -149
  53. package/src/components/Item/Inventory/ErrorBoundary.tsx +42 -42
  54. package/src/components/Item/Inventory/ItemContainer.tsx +231 -231
  55. package/src/components/Item/Inventory/ItemContainerTypes.ts +6 -6
  56. package/src/components/Item/Inventory/ItemQuantitySelector.tsx +138 -138
  57. package/src/components/Item/Inventory/ItemSlot.tsx +595 -580
  58. package/src/components/Item/Inventory/itemContainerHelper.ts +175 -175
  59. package/src/components/ListMenu.tsx +63 -63
  60. package/src/components/Marketplace/BuyPanel.tsx +296 -0
  61. package/src/components/Marketplace/ManagmentPanel.tsx +247 -0
  62. package/src/components/Marketplace/Marketplace.tsx +106 -132
  63. package/src/components/Marketplace/MarketplaceRows.tsx +161 -171
  64. package/src/components/Marketplace/{__mocks__ → filters}/index.tsx +67 -65
  65. package/src/components/Multitab/Tab.tsx +66 -66
  66. package/src/components/Multitab/TabBody.tsx +13 -13
  67. package/src/components/Multitab/TabsContainer.tsx +97 -97
  68. package/src/components/NPCDialog/NPCDialog.tsx +121 -121
  69. package/src/components/NPCDialog/NPCDialogText.tsx +113 -113
  70. package/src/components/NPCDialog/NPCMultiDialog.tsx +159 -159
  71. package/src/components/NPCDialog/QuestionDialog/QuestionDialog.tsx +237 -237
  72. package/src/components/Pager.tsx +94 -0
  73. package/src/components/ProgressBar.tsx +95 -95
  74. package/src/components/PropertySelect/PropertySelect.tsx +106 -106
  75. package/src/components/QuestInfo/QuestInfo.tsx +233 -233
  76. package/src/components/QuestList.tsx +135 -135
  77. package/src/components/RPGUIContainer.tsx +47 -47
  78. package/src/components/RPGUIForceRenderStart.tsx +45 -45
  79. package/src/components/RPGUIRoot.tsx +14 -14
  80. package/src/components/RadioButton.tsx +53 -53
  81. package/src/components/RadioInput/RadioButton.tsx +96 -96
  82. package/src/components/RadioInput/RadioInput.tsx +102 -102
  83. package/src/components/RadioInput/instruments.ts +15 -15
  84. package/src/components/RangeSlider.tsx +78 -78
  85. package/src/components/RelativeListMenu.tsx +90 -90
  86. package/src/components/ScrollList.tsx +79 -79
  87. package/src/components/Shortcuts/Shortcuts.tsx +193 -193
  88. package/src/components/Shortcuts/ShortcutsSetter.tsx +139 -139
  89. package/src/components/Shortcuts/SingleShortcut.ts +82 -82
  90. package/src/components/Shortcuts/useShortcutCooldown.ts +23 -23
  91. package/src/components/SimpleProgressBar.tsx +62 -62
  92. package/src/components/SkillProgressBar.tsx +198 -198
  93. package/src/components/SkillsContainer.tsx +213 -213
  94. package/src/components/Spellbook/Spell.tsx +235 -236
  95. package/src/components/Spellbook/Spellbook.tsx +145 -161
  96. package/src/components/Spellbook/cards/MobileSpellTooltip.tsx +137 -137
  97. package/src/components/Spellbook/cards/SpellInfo.tsx +134 -134
  98. package/src/components/Spellbook/cards/SpellInfoDisplay.tsx +31 -31
  99. package/src/components/Spellbook/cards/SpellInfoWrapper.tsx +48 -48
  100. package/src/components/Spellbook/cards/SpellTooltip.tsx +70 -70
  101. package/src/components/Spellbook/constants.ts +7 -7
  102. package/src/components/Spellbook/mockSpells.ts +84 -84
  103. package/src/components/StaticBook/StaticBook.tsx +103 -103
  104. package/src/components/TextArea.tsx +11 -11
  105. package/src/components/TimeWidget/DayNightPeriod/DayNightPeriod.tsx +35 -35
  106. package/src/components/TimeWidget/TimeWidget.tsx +65 -65
  107. package/src/components/TradingMenu/TradingItemRow.tsx +199 -199
  108. package/src/components/TradingMenu/TradingMenu.tsx +219 -219
  109. package/src/components/TradingMenu/items.mock.ts +48 -48
  110. package/src/components/Truncate.tsx +25 -25
  111. package/src/components/itemSelector/ItemSelector.tsx +136 -136
  112. package/src/components/shared/Column.tsx +16 -16
  113. package/src/components/shared/Ellipsis.tsx +76 -68
  114. package/src/components/shared/SpriteFromAtlas.tsx +104 -104
  115. package/src/components/typography/DynamicText.tsx +49 -49
  116. package/src/constants/uiColors.ts +20 -20
  117. package/src/constants/uiDevices.ts +3 -3
  118. package/src/constants/uiFonts.ts +12 -12
  119. package/src/hooks/useEventListener.ts +21 -21
  120. package/src/hooks/useOutsideAlerter.ts +25 -25
  121. package/src/index.tsx +42 -42
  122. package/src/libs/CastingTypeHelper.ts +7 -7
  123. package/src/libs/StringHelpers.ts +3 -3
  124. package/src/libs/itemCounter.ts +21 -21
  125. package/src/mocks/atlas/entities/entities.json +20215 -20215
  126. package/src/mocks/atlas/icons/icons.json +735 -735
  127. package/src/mocks/atlas/items/items.json +12086 -12086
  128. package/src/mocks/equipmentSet.mocks.ts +391 -391
  129. package/src/mocks/itemContainer.mocks.ts +605 -605
  130. package/src/mocks/skills.mocks.ts +130 -130
  131. package/src/stories/Arrow.stories.tsx +26 -26
  132. package/src/stories/Button.stories.tsx +36 -36
  133. package/src/stories/CharacterSelection.stories.tsx +44 -44
  134. package/src/stories/CharacterStatus.stories.tsx +29 -29
  135. package/src/stories/Chat.stories.tsx +187 -187
  136. package/src/stories/ChatDeprecated.stories.tsx +170 -170
  137. package/src/stories/CheckButton.stories.tsx +48 -48
  138. package/src/stories/CircullarController.stories.tsx +37 -37
  139. package/src/stories/CraftBook.stories.tsx +42 -42
  140. package/src/stories/DayNightPeriod.stories.tsx +27 -27
  141. package/src/stories/DraggableContainer.stories.tsx +28 -28
  142. package/src/stories/Dropdown.stories.tsx +46 -46
  143. package/src/stories/DropdownSelectorContainer.stories.tsx +41 -41
  144. package/src/stories/EquipmentSet.stories.tsx +65 -65
  145. package/src/stories/HistoryDialog.stories.tsx +61 -61
  146. package/src/stories/ItemContainer.stories.tsx +201 -201
  147. package/src/stories/ItemInfoDisplay.stories.tsx +33 -33
  148. package/src/stories/ItemQuantitySelector.stories.tsx +26 -26
  149. package/src/stories/ItemSelector.stories.tsx +77 -77
  150. package/src/stories/ItemTradingComponent.stories.tsx +35 -35
  151. package/src/stories/ListMenu.stories.tsx +56 -56
  152. package/src/stories/Marketplace.stories.tsx +53 -42
  153. package/src/stories/MarketplaceRows.stories.tsx +28 -28
  154. package/src/stories/Multitab.stories.tsx +51 -51
  155. package/src/stories/NPCDialog.stories.tsx +130 -130
  156. package/src/stories/NPCMultiDialog.stories.tsx +71 -71
  157. package/src/stories/ProgressBar.stories.tsx +23 -23
  158. package/src/stories/PropertySelect.stories.tsx +40 -40
  159. package/src/stories/QuestInfo.stories.tsx +107 -107
  160. package/src/stories/QuestList.stories.tsx +82 -82
  161. package/src/stories/RPGUIContainers.stories.tsx +42 -42
  162. package/src/stories/RadioButton.stories.tsx +49 -49
  163. package/src/stories/RadioInput.stories.tsx +34 -34
  164. package/src/stories/RangeSlider.stories.tsx +64 -64
  165. package/src/stories/ScrollList.stories.tsx +85 -85
  166. package/src/stories/Shortcuts.stories.tsx +39 -39
  167. package/src/stories/SimpleProgressBar.stories.tsx +22 -22
  168. package/src/stories/SkillProgressBar.stories.tsx +34 -34
  169. package/src/stories/SkillsContainer.stories.tsx +35 -35
  170. package/src/stories/SpellInfoDisplay.stories.tsx +27 -27
  171. package/src/stories/Spellbook.stories.tsx +104 -104
  172. package/src/stories/StaticBook.stories.tsx +32 -32
  173. package/src/stories/Text.stories.tsx +42 -42
  174. package/src/stories/TimeWidget.stories.tsx +27 -27
  175. package/src/stories/TradingMenu.stories.tsx +47 -47
  176. package/src/types/eventTypes.ts +4 -4
  177. package/src/types/index.d.ts +2 -2
@@ -1,282 +1,282 @@
1
- import {
2
- getItemTextureKeyPath,
3
- IItem,
4
- IItemContainer,
5
- IRawSpell,
6
- IShortcut,
7
- ShortcutType,
8
- } from '@rpg-engine/shared';
9
- import React from 'react';
10
- import styled from 'styled-components';
11
- import { uiColors } from '../../constants/uiColors';
12
- import { SpriteFromAtlas } from '../shared/SpriteFromAtlas';
13
- import { SingleShortcut } from '../Shortcuts/SingleShortcut';
14
- import { useShortcutCooldown } from '../Shortcuts/useShortcutCooldown';
15
-
16
- export type CircularControllerProps = {
17
- onActionClick: () => void;
18
- onCancelClick: () => void;
19
- onShortcutClick: (index: number) => void;
20
- mana: number;
21
- shortcuts: IShortcut[];
22
- inventory?: IItemContainer | null;
23
- atlasIMG: any;
24
- atlasJSON: any;
25
- spellCooldowns?: Record<string, number>;
26
- };
27
-
28
- export const CircularController: React.FC<CircularControllerProps> = ({
29
- onActionClick,
30
- onCancelClick,
31
- onShortcutClick,
32
- mana,
33
- shortcuts,
34
- inventory,
35
- atlasIMG,
36
- atlasJSON,
37
- spellCooldowns,
38
- }) => {
39
- const { handleShortcutCast, shortcutCooldown } = useShortcutCooldown(
40
- onShortcutClick
41
- );
42
-
43
- const onTouchStart = (e: React.TouchEvent<HTMLButtonElement>) => {
44
- const target = e.target as HTMLButtonElement;
45
- target?.classList.add('active');
46
- };
47
-
48
- const onTouchEnd = (
49
- action: () => void,
50
- e: React.TouchEvent<HTMLButtonElement>
51
- ) => {
52
- const target = e.target as HTMLButtonElement;
53
- setTimeout(() => {
54
- target?.classList.remove('active');
55
- }, 100);
56
- action();
57
- };
58
-
59
- const renderShortcut = (i: number) => {
60
- const buildClassName = (classBase: string, isOnCooldown: boolean) => {
61
- return `${classBase} ${isOnCooldown ? 'onCooldown' : ''}`;
62
- };
63
-
64
- let variant = '';
65
-
66
- if (i === 0) variant = 'top';
67
- else if (i >= 3) variant = `bottom-${i - 3}`;
68
-
69
- const onShortcutClickBinded =
70
- shortcuts[i]?.type !== ShortcutType.None
71
- ? handleShortcutCast.bind(null, i)
72
- : () => {};
73
-
74
- const isOnShortcutCooldown = shortcutCooldown > 0;
75
-
76
- if (shortcuts[i]?.type === ShortcutType.Item) {
77
- const payload = shortcuts[i]?.payload as IItem | undefined;
78
-
79
- let itemsFromEquipment: (IItem | undefined | null)[] = [];
80
-
81
- if (inventory) {
82
- Object.keys(inventory.slots).forEach(i => {
83
- const index = parseInt(i);
84
-
85
- if (inventory.slots[index]?.key === payload?.key) {
86
- itemsFromEquipment.push(inventory.slots[index]);
87
- }
88
- });
89
- }
90
-
91
- const totalQty = itemsFromEquipment.reduce(
92
- (acc, item) => acc + (item?.stackQty || 1),
93
- 0
94
- );
95
-
96
- return (
97
- <StyledShortcut
98
- key={i}
99
- onTouchStart={onTouchStart}
100
- onTouchEnd={onTouchEnd.bind(null, onShortcutClickBinded)}
101
- disabled={false}
102
- className={variant}
103
- >
104
- {isOnShortcutCooldown && (
105
- <span className="cooldown">{shortcutCooldown.toFixed(1)}</span>
106
- )}
107
- {payload && (
108
- <SpriteFromAtlas
109
- atlasIMG={atlasIMG}
110
- atlasJSON={atlasJSON}
111
- spriteKey={getItemTextureKeyPath(
112
- {
113
- key: payload.texturePath,
114
- texturePath: payload.texturePath,
115
- stackQty: payload.stackQty || 1,
116
- isStackable: payload.isStackable,
117
- },
118
- atlasJSON
119
- )}
120
- width={32}
121
- height={32}
122
- imgScale={1.4}
123
- imgStyle={{ left: '4px' }}
124
- containerStyle={{ pointerEvents: 'none' }}
125
- />
126
- )}
127
- <span className={buildClassName('qty', isOnShortcutCooldown)}>
128
- {totalQty}
129
- </span>
130
- </StyledShortcut>
131
- );
132
- }
133
-
134
- const payload = shortcuts[i]?.payload as IRawSpell | undefined;
135
-
136
- const spellCooldown = !payload
137
- ? 0
138
- : spellCooldowns?.[payload.magicWords.replaceAll(' ', '_')] ??
139
- shortcutCooldown;
140
- const cooldown =
141
- spellCooldown > shortcutCooldown ? spellCooldown : shortcutCooldown;
142
- const isOnCooldown = cooldown > 0 && !!payload;
143
-
144
- return (
145
- <StyledShortcut
146
- key={i}
147
- onTouchStart={onTouchStart}
148
- onTouchEnd={onTouchEnd.bind(null, onShortcutClickBinded)}
149
- disabled={mana < (payload?.manaCost ?? 0)}
150
- className={variant}
151
- >
152
- {isOnCooldown && (
153
- <span className="cooldown">
154
- {cooldown.toFixed(cooldown < 10 ? 1 : 0)}
155
- </span>
156
- )}
157
- <span className={buildClassName('mana', isOnCooldown)}>
158
- {payload && payload.manaCost}
159
- </span>
160
- <span className="magicWords">
161
- {payload?.magicWords.split(' ').map(word => word[0])}
162
- </span>
163
- </StyledShortcut>
164
- );
165
- };
166
-
167
- return (
168
- <ButtonsContainer>
169
- <ShortcutsContainer>
170
- {Array.from({ length: 6 }).map((_, i) => renderShortcut(i))}
171
- </ShortcutsContainer>
172
- <Button
173
- onTouchStart={onTouchStart}
174
- onTouchEnd={onTouchEnd.bind(null, onActionClick)}
175
- >
176
- <div className="rpgui-icon sword" />
177
- </Button>
178
-
179
- <CancelButton
180
- onTouchStart={onTouchStart}
181
- onTouchEnd={onTouchEnd.bind(null, onCancelClick)}
182
- >
183
- <span>X</span>
184
- </CancelButton>
185
- </ButtonsContainer>
186
- );
187
- };
188
-
189
- const Button = styled.button`
190
- width: 4.3rem;
191
- height: 4.3rem;
192
- background-color: ${uiColors.lightGray};
193
- border: 2px solid ${uiColors.darkGray};
194
- border-radius: 50%;
195
- text-transform: uppercase;
196
- font-size: 0.7rem;
197
- font-weight: bold;
198
- display: flex;
199
- align-items: center;
200
- justify-content: center;
201
- position: relative;
202
- transition: all 0.1s;
203
- margin-top: -3rem;
204
-
205
- &.active {
206
- background-color: ${uiColors.gray};
207
- border-color: ${uiColors.yellow};
208
- }
209
-
210
- .sword {
211
- transform: rotate(-45deg);
212
- height: 2.5rem;
213
- width: 1.9rem;
214
- pointer-events: none;
215
- }
216
- `;
217
-
218
- const CancelButton = styled(Button)`
219
- width: 3rem;
220
- height: 3rem;
221
- font-size: 0.8rem;
222
-
223
- span {
224
- margin-top: 4px;
225
- margin-left: 2px;
226
- pointer-events: none;
227
- }
228
- `;
229
-
230
- const ButtonsContainer = styled.div`
231
- display: flex;
232
- align-items: center;
233
- justify-content: center;
234
- gap: 0.5rem;
235
- `;
236
-
237
- const ShortcutsContainer = styled.div`
238
- display: flex;
239
- align-items: center;
240
- justify-content: center;
241
- gap: 0.5rem;
242
- flex-direction: column;
243
- margin-top: 3rem;
244
-
245
- .top {
246
- transform: translate(93%, 25%);
247
- }
248
-
249
- .bottom-0 {
250
- transform: translate(93%, -25%);
251
- }
252
-
253
- .bottom-1 {
254
- transform: translate(-120%, calc(-5.5rem));
255
- }
256
-
257
- .bottom-2 {
258
- transform: translate(-30%, calc(-5.5rem - 25%));
259
- }
260
- `;
261
-
262
- const StyledShortcut = styled(SingleShortcut)`
263
- width: 2.5rem;
264
- height: 2.5rem;
265
- transition: all 0.1s;
266
-
267
- .mana,
268
- .qty {
269
- font-size: 0.5rem;
270
- }
271
-
272
- &:hover,
273
- &:focus,
274
- &:active {
275
- background-color: ${uiColors.lightGray};
276
- }
277
-
278
- &.active {
279
- background-color: ${uiColors.gray};
280
- border-color: ${uiColors.yellow};
281
- }
282
- `;
1
+ import {
2
+ getItemTextureKeyPath,
3
+ IItem,
4
+ IItemContainer,
5
+ IRawSpell,
6
+ IShortcut,
7
+ ShortcutType,
8
+ } from '@rpg-engine/shared';
9
+ import React from 'react';
10
+ import styled from 'styled-components';
11
+ import { uiColors } from '../../constants/uiColors';
12
+ import { SpriteFromAtlas } from '../shared/SpriteFromAtlas';
13
+ import { SingleShortcut } from '../Shortcuts/SingleShortcut';
14
+ import { useShortcutCooldown } from '../Shortcuts/useShortcutCooldown';
15
+
16
+ export type CircularControllerProps = {
17
+ onActionClick: () => void;
18
+ onCancelClick: () => void;
19
+ onShortcutClick: (index: number) => void;
20
+ mana: number;
21
+ shortcuts: IShortcut[];
22
+ inventory?: IItemContainer | null;
23
+ atlasIMG: any;
24
+ atlasJSON: any;
25
+ spellCooldowns?: Record<string, number>;
26
+ };
27
+
28
+ export const CircularController: React.FC<CircularControllerProps> = ({
29
+ onActionClick,
30
+ onCancelClick,
31
+ onShortcutClick,
32
+ mana,
33
+ shortcuts,
34
+ inventory,
35
+ atlasIMG,
36
+ atlasJSON,
37
+ spellCooldowns,
38
+ }) => {
39
+ const { handleShortcutCast, shortcutCooldown } = useShortcutCooldown(
40
+ onShortcutClick
41
+ );
42
+
43
+ const onTouchStart = (e: React.TouchEvent<HTMLButtonElement>) => {
44
+ const target = e.target as HTMLButtonElement;
45
+ target?.classList.add('active');
46
+ };
47
+
48
+ const onTouchEnd = (
49
+ action: () => void,
50
+ e: React.TouchEvent<HTMLButtonElement>
51
+ ) => {
52
+ const target = e.target as HTMLButtonElement;
53
+ setTimeout(() => {
54
+ target?.classList.remove('active');
55
+ }, 100);
56
+ action();
57
+ };
58
+
59
+ const renderShortcut = (i: number) => {
60
+ const buildClassName = (classBase: string, isOnCooldown: boolean) => {
61
+ return `${classBase} ${isOnCooldown ? 'onCooldown' : ''}`;
62
+ };
63
+
64
+ let variant = '';
65
+
66
+ if (i === 0) variant = 'top';
67
+ else if (i >= 3) variant = `bottom-${i - 3}`;
68
+
69
+ const onShortcutClickBinded =
70
+ shortcuts[i]?.type !== ShortcutType.None
71
+ ? handleShortcutCast.bind(null, i)
72
+ : () => {};
73
+
74
+ const isOnShortcutCooldown = shortcutCooldown > 0;
75
+
76
+ if (shortcuts[i]?.type === ShortcutType.Item) {
77
+ const payload = shortcuts[i]?.payload as IItem | undefined;
78
+
79
+ let itemsFromEquipment: (IItem | undefined | null)[] = [];
80
+
81
+ if (inventory) {
82
+ Object.keys(inventory.slots).forEach(i => {
83
+ const index = parseInt(i);
84
+
85
+ if (inventory.slots[index]?.key === payload?.key) {
86
+ itemsFromEquipment.push(inventory.slots[index]);
87
+ }
88
+ });
89
+ }
90
+
91
+ const totalQty = itemsFromEquipment.reduce(
92
+ (acc, item) => acc + (item?.stackQty || 1),
93
+ 0
94
+ );
95
+
96
+ return (
97
+ <StyledShortcut
98
+ key={i}
99
+ onTouchStart={onTouchStart}
100
+ onTouchEnd={onTouchEnd.bind(null, onShortcutClickBinded)}
101
+ disabled={false}
102
+ className={variant}
103
+ >
104
+ {isOnShortcutCooldown && (
105
+ <span className="cooldown">{shortcutCooldown.toFixed(1)}</span>
106
+ )}
107
+ {payload && (
108
+ <SpriteFromAtlas
109
+ atlasIMG={atlasIMG}
110
+ atlasJSON={atlasJSON}
111
+ spriteKey={getItemTextureKeyPath(
112
+ {
113
+ key: payload.texturePath,
114
+ texturePath: payload.texturePath,
115
+ stackQty: payload.stackQty || 1,
116
+ isStackable: payload.isStackable,
117
+ },
118
+ atlasJSON
119
+ )}
120
+ width={32}
121
+ height={32}
122
+ imgScale={1.4}
123
+ imgStyle={{ left: '4px' }}
124
+ containerStyle={{ pointerEvents: 'none' }}
125
+ />
126
+ )}
127
+ <span className={buildClassName('qty', isOnShortcutCooldown)}>
128
+ {totalQty}
129
+ </span>
130
+ </StyledShortcut>
131
+ );
132
+ }
133
+
134
+ const payload = shortcuts[i]?.payload as IRawSpell | undefined;
135
+
136
+ const spellCooldown = !payload
137
+ ? 0
138
+ : spellCooldowns?.[payload.magicWords.replaceAll(' ', '_')] ??
139
+ shortcutCooldown;
140
+ const cooldown =
141
+ spellCooldown > shortcutCooldown ? spellCooldown : shortcutCooldown;
142
+ const isOnCooldown = cooldown > 0 && !!payload;
143
+
144
+ return (
145
+ <StyledShortcut
146
+ key={i}
147
+ onTouchStart={onTouchStart}
148
+ onTouchEnd={onTouchEnd.bind(null, onShortcutClickBinded)}
149
+ disabled={mana < (payload?.manaCost ?? 0)}
150
+ className={variant}
151
+ >
152
+ {isOnCooldown && (
153
+ <span className="cooldown">
154
+ {cooldown.toFixed(cooldown < 10 ? 1 : 0)}
155
+ </span>
156
+ )}
157
+ <span className={buildClassName('mana', isOnCooldown)}>
158
+ {payload && payload.manaCost}
159
+ </span>
160
+ <span className="magicWords">
161
+ {payload?.magicWords.split(' ').map(word => word[0])}
162
+ </span>
163
+ </StyledShortcut>
164
+ );
165
+ };
166
+
167
+ return (
168
+ <ButtonsContainer>
169
+ <ShortcutsContainer>
170
+ {Array.from({ length: 6 }).map((_, i) => renderShortcut(i))}
171
+ </ShortcutsContainer>
172
+ <Button
173
+ onTouchStart={onTouchStart}
174
+ onTouchEnd={onTouchEnd.bind(null, onActionClick)}
175
+ >
176
+ <div className="rpgui-icon sword" />
177
+ </Button>
178
+
179
+ <CancelButton
180
+ onTouchStart={onTouchStart}
181
+ onTouchEnd={onTouchEnd.bind(null, onCancelClick)}
182
+ >
183
+ <span>X</span>
184
+ </CancelButton>
185
+ </ButtonsContainer>
186
+ );
187
+ };
188
+
189
+ const Button = styled.button`
190
+ width: 4.3rem;
191
+ height: 4.3rem;
192
+ background-color: ${uiColors.lightGray};
193
+ border: 2px solid ${uiColors.darkGray};
194
+ border-radius: 50%;
195
+ text-transform: uppercase;
196
+ font-size: 0.7rem;
197
+ font-weight: bold;
198
+ display: flex;
199
+ align-items: center;
200
+ justify-content: center;
201
+ position: relative;
202
+ transition: all 0.1s;
203
+ margin-top: -3rem;
204
+
205
+ &.active {
206
+ background-color: ${uiColors.gray};
207
+ border-color: ${uiColors.yellow};
208
+ }
209
+
210
+ .sword {
211
+ transform: rotate(-45deg);
212
+ height: 2.5rem;
213
+ width: 1.9rem;
214
+ pointer-events: none;
215
+ }
216
+ `;
217
+
218
+ const CancelButton = styled(Button)`
219
+ width: 3rem;
220
+ height: 3rem;
221
+ font-size: 0.8rem;
222
+
223
+ span {
224
+ margin-top: 4px;
225
+ margin-left: 2px;
226
+ pointer-events: none;
227
+ }
228
+ `;
229
+
230
+ const ButtonsContainer = styled.div`
231
+ display: flex;
232
+ align-items: center;
233
+ justify-content: center;
234
+ gap: 0.5rem;
235
+ `;
236
+
237
+ const ShortcutsContainer = styled.div`
238
+ display: flex;
239
+ align-items: center;
240
+ justify-content: center;
241
+ gap: 0.5rem;
242
+ flex-direction: column;
243
+ margin-top: 3rem;
244
+
245
+ .top {
246
+ transform: translate(93%, 25%);
247
+ }
248
+
249
+ .bottom-0 {
250
+ transform: translate(93%, -25%);
251
+ }
252
+
253
+ .bottom-1 {
254
+ transform: translate(-120%, calc(-5.5rem));
255
+ }
256
+
257
+ .bottom-2 {
258
+ transform: translate(-30%, calc(-5.5rem - 25%));
259
+ }
260
+ `;
261
+
262
+ const StyledShortcut = styled(SingleShortcut)`
263
+ width: 2.5rem;
264
+ height: 2.5rem;
265
+ transition: all 0.1s;
266
+
267
+ .mana,
268
+ .qty {
269
+ font-size: 0.5rem;
270
+ }
271
+
272
+ &:hover,
273
+ &:focus,
274
+ &:active {
275
+ background-color: ${uiColors.lightGray};
276
+ }
277
+
278
+ &.active {
279
+ background-color: ${uiColors.gray};
280
+ border-color: ${uiColors.yellow};
281
+ }
282
+ `;
@@ -0,0 +1,87 @@
1
+ import React from 'react';
2
+ import styled from 'styled-components';
3
+ import ModalPortal from './Abstractions/ModalPortal';
4
+ import { Button, ButtonTypes } from './Button';
5
+ import { DraggableContainer } from './DraggableContainer';
6
+
7
+ interface IConfirmModalProps {
8
+ onConfirm: () => void;
9
+ onClose: () => void;
10
+ message?: string;
11
+ }
12
+
13
+ export const ConfirmModal: React.FC<IConfirmModalProps> = ({
14
+ onConfirm,
15
+ onClose,
16
+ message,
17
+ }) => {
18
+ return (
19
+ <ModalPortal>
20
+ <Background />
21
+ <Container onPointerDown={onClose}>
22
+ <DraggableContainer width="auto" dragDisabled>
23
+ <Wrapper onPointerDown={e => e.stopPropagation()}>
24
+ <p>{message ?? 'Are you sure?'}</p>
25
+
26
+ <ButtonsWrapper>
27
+ <div className="cancel-button">
28
+ <Button
29
+ buttonType={ButtonTypes.RPGUIButton}
30
+ onPointerDown={onClose}
31
+ >
32
+ No
33
+ </Button>
34
+ </div>
35
+ <Button
36
+ buttonType={ButtonTypes.RPGUIButton}
37
+ onPointerDown={onConfirm}
38
+ >
39
+ Yes
40
+ </Button>
41
+ </ButtonsWrapper>
42
+ </Wrapper>
43
+ </DraggableContainer>
44
+ </Container>
45
+ </ModalPortal>
46
+ );
47
+ };
48
+
49
+ const Background = styled.div`
50
+ position: absolute;
51
+ width: 100%;
52
+ height: 100%;
53
+ background-color: #000000;
54
+ opacity: 0.5;
55
+ left: 0;
56
+ top: 0;
57
+ z-index: 1000;
58
+ `;
59
+
60
+ const Container = styled.div`
61
+ position: absolute;
62
+ width: 100%;
63
+ height: 100%;
64
+ left: 0;
65
+ top: 0;
66
+ display: flex;
67
+ justify-content: center;
68
+ align-items: center;
69
+ z-index: 1001;
70
+ `;
71
+
72
+ const Wrapper = styled.div`
73
+ p {
74
+ margin: 0;
75
+ }
76
+ `;
77
+
78
+ const ButtonsWrapper = styled.div`
79
+ display: flex;
80
+ justify-content: flex-end;
81
+ gap: 5px;
82
+ margin-top: 5px;
83
+
84
+ .cancel-button {
85
+ filter: grayscale(0.7);
86
+ }
87
+ `;