@rpg-engine/long-bow 0.4.9 → 0.4.81

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/Marketplace/BuyPanel.d.ts +0 -1
  4. package/dist/components/Marketplace/ManagmentPanel.d.ts +0 -1
  5. package/dist/components/ProgressBar.d.ts +1 -0
  6. package/dist/long-bow.cjs.development.js +1566 -1595
  7. package/dist/long-bow.cjs.development.js.map +1 -1
  8. package/dist/long-bow.cjs.production.min.js +1 -1
  9. package/dist/long-bow.cjs.production.min.js.map +1 -1
  10. package/dist/long-bow.esm.js +1472 -1501
  11. package/dist/long-bow.esm.js.map +1 -1
  12. package/package.json +100 -100
  13. package/src/.DS_Store +0 -0
  14. package/src/components/.DS_Store +0 -0
  15. package/src/components/Abstractions/ModalPortal.tsx +22 -22
  16. package/src/components/Abstractions/SlotsContainer.tsx +62 -62
  17. package/src/components/Arrow/SelectArrow.tsx +69 -69
  18. package/src/components/Arrow/img/arrow01-left-clicked.png +0 -0
  19. package/src/components/Arrow/img/arrow01-left.png +0 -0
  20. package/src/components/Arrow/img/arrow01-right-clicked.png +0 -0
  21. package/src/components/Arrow/img/arrow01-right.png +0 -0
  22. package/src/components/Arrow/img/arrow02-left-clicked.png +0 -0
  23. package/src/components/Arrow/img/arrow02-left.png +0 -0
  24. package/src/components/Arrow/img/arrow02-right-clicked.png +0 -0
  25. package/src/components/Arrow/img/arrow02-right.png +0 -0
  26. package/src/components/Button.tsx +40 -40
  27. package/src/components/Character/CharacterSelection.tsx +98 -98
  28. package/src/components/CharacterStatus/CharacterStatus.tsx +120 -120
  29. package/src/components/Chat/Chat.tsx +196 -196
  30. package/src/components/Chatdeprecated/ChatDeprecated.tsx +198 -198
  31. package/src/components/CheckButton.tsx +65 -65
  32. package/src/components/CircularController/CircularController.tsx +282 -282
  33. package/src/components/ConfirmModal.tsx +87 -87
  34. package/src/components/CraftBook/CraftBook.tsx +286 -286
  35. package/src/components/CraftBook/CraftingRecipe.tsx +161 -161
  36. package/src/components/CraftBook/MockItems.ts +101 -101
  37. package/src/components/DraggableContainer.tsx +183 -183
  38. package/src/components/Dropdown.tsx +114 -114
  39. package/src/components/DropdownSelectorContainer.tsx +42 -42
  40. package/src/components/Equipment/EquipmentSet.tsx +199 -199
  41. package/src/components/HistoryDialog.tsx +104 -104
  42. package/src/components/Input.tsx +15 -15
  43. package/src/components/InputRadio.tsx +41 -41
  44. package/src/components/Item/Cards/ItemInfo.tsx +298 -298
  45. package/src/components/Item/Cards/ItemInfoDisplay.tsx +135 -135
  46. package/src/components/Item/Cards/ItemInfoWrapper.tsx +62 -62
  47. package/src/components/Item/Cards/ItemTooltip.tsx +83 -83
  48. package/src/components/Item/Cards/MobileItemTooltip.tsx +149 -149
  49. package/src/components/Item/Inventory/ErrorBoundary.tsx +42 -42
  50. package/src/components/Item/Inventory/ItemContainer.tsx +231 -231
  51. package/src/components/Item/Inventory/ItemContainerTypes.ts +6 -6
  52. package/src/components/Item/Inventory/ItemQuantitySelector.tsx +138 -138
  53. package/src/components/Item/Inventory/ItemSlot.tsx +595 -595
  54. package/src/components/Item/Inventory/itemContainerHelper.ts +175 -175
  55. package/src/components/ListMenu.tsx +63 -63
  56. package/src/components/Marketplace/BuyPanel.tsx +296 -304
  57. package/src/components/Marketplace/ManagmentPanel.tsx +247 -255
  58. package/src/components/Marketplace/Marketplace.tsx +106 -106
  59. package/src/components/Marketplace/MarketplaceRows.tsx +161 -177
  60. package/src/components/Marketplace/filters/index.tsx +67 -67
  61. package/src/components/Multitab/Tab.tsx +66 -66
  62. package/src/components/Multitab/TabBody.tsx +13 -13
  63. package/src/components/Multitab/TabsContainer.tsx +97 -97
  64. package/src/components/NPCDialog/.DS_Store +0 -0
  65. package/src/components/NPCDialog/NPCDialog.tsx +121 -121
  66. package/src/components/NPCDialog/NPCDialogText.tsx +113 -113
  67. package/src/components/NPCDialog/NPCMultiDialog.tsx +159 -159
  68. package/src/components/NPCDialog/QuestionDialog/QuestionDialog.tsx +237 -237
  69. package/src/components/NPCDialog/img/.DS_Store +0 -0
  70. package/src/components/Pager.tsx +94 -94
  71. package/src/components/ProgressBar.tsx +102 -95
  72. package/src/components/PropertySelect/PropertySelect.tsx +106 -106
  73. package/src/components/QuestInfo/QuestInfo.tsx +233 -233
  74. package/src/components/QuestList.tsx +135 -135
  75. package/src/components/RPGUIContainer.tsx +47 -47
  76. package/src/components/RPGUIForceRenderStart.tsx +45 -45
  77. package/src/components/RPGUIRoot.tsx +14 -14
  78. package/src/components/RadioButton.tsx +53 -53
  79. package/src/components/RadioInput/RadioButton.tsx +96 -96
  80. package/src/components/RadioInput/RadioInput.tsx +102 -102
  81. package/src/components/RadioInput/instruments.ts +15 -15
  82. package/src/components/RangeSlider.tsx +78 -78
  83. package/src/components/RelativeListMenu.tsx +90 -90
  84. package/src/components/ScrollList.tsx +79 -79
  85. package/src/components/Shortcuts/Shortcuts.tsx +193 -193
  86. package/src/components/Shortcuts/ShortcutsSetter.tsx +139 -139
  87. package/src/components/Shortcuts/SingleShortcut.ts +82 -82
  88. package/src/components/Shortcuts/useShortcutCooldown.ts +23 -23
  89. package/src/components/SimpleProgressBar.tsx +62 -62
  90. package/src/components/SkillProgressBar.tsx +198 -198
  91. package/src/components/SkillsContainer.tsx +213 -213
  92. package/src/components/Spellbook/Spell.tsx +235 -235
  93. package/src/components/Spellbook/Spellbook.tsx +145 -145
  94. package/src/components/Spellbook/cards/MobileSpellTooltip.tsx +137 -137
  95. package/src/components/Spellbook/cards/SpellInfo.tsx +134 -134
  96. package/src/components/Spellbook/cards/SpellInfoDisplay.tsx +31 -31
  97. package/src/components/Spellbook/cards/SpellInfoWrapper.tsx +48 -48
  98. package/src/components/Spellbook/cards/SpellTooltip.tsx +70 -70
  99. package/src/components/Spellbook/constants.ts +7 -7
  100. package/src/components/Spellbook/mockSpells.ts +84 -84
  101. package/src/components/StaticBook/StaticBook.tsx +103 -103
  102. package/src/components/TextArea.tsx +11 -11
  103. package/src/components/TimeWidget/DayNightPeriod/DayNightPeriod.tsx +35 -35
  104. package/src/components/TimeWidget/TimeWidget.tsx +65 -65
  105. package/src/components/TradingMenu/TradingItemRow.tsx +199 -199
  106. package/src/components/TradingMenu/TradingMenu.tsx +219 -219
  107. package/src/components/TradingMenu/items.mock.ts +48 -48
  108. package/src/components/Truncate.tsx +25 -25
  109. package/src/components/itemSelector/ItemSelector.tsx +136 -136
  110. package/src/components/shared/Column.tsx +16 -16
  111. package/src/components/shared/Ellipsis.tsx +76 -76
  112. package/src/components/shared/SpriteFromAtlas.tsx +104 -104
  113. package/src/components/typography/DynamicText.tsx +49 -49
  114. package/src/constants/uiColors.ts +20 -20
  115. package/src/constants/uiDevices.ts +3 -3
  116. package/src/constants/uiFonts.ts +12 -12
  117. package/src/hooks/useEventListener.ts +21 -21
  118. package/src/hooks/useOutsideAlerter.ts +25 -25
  119. package/src/index.tsx +42 -42
  120. package/src/libs/CastingTypeHelper.ts +7 -7
  121. package/src/libs/StringHelpers.ts +3 -3
  122. package/src/libs/itemCounter.ts +21 -21
  123. package/src/mocks/.DS_Store +0 -0
  124. package/src/mocks/atlas/.DS_Store +0 -0
  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 -57
  153. package/src/stories/MarketplaceRows.stories.tsx +27 -27
  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 +24 -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,41 +1,41 @@
1
- import React from 'react';
2
-
3
- interface IProps
4
- extends React.DetailedHTMLProps<
5
- React.InputHTMLAttributes<HTMLInputElement>,
6
- HTMLInputElement
7
- > {
8
- label: string;
9
- name: string;
10
- value: string;
11
- isChecked: boolean;
12
- onRadioSelect: (value: string) => void;
13
- }
14
-
15
- export const InputRadio: React.FC<IProps> = ({
16
- label,
17
- name,
18
- value,
19
- isChecked,
20
- onRadioSelect,
21
- }) => {
22
- const onRadioClick = (): void => {
23
- onRadioSelect(value);
24
- };
25
-
26
- return (
27
- <div onPointerUp={onRadioClick}>
28
- <input
29
- className="rpgui-radio"
30
- name={name}
31
- value={value}
32
- type="radio"
33
- data-rpguitype="radio"
34
- checked={isChecked}
35
- // rpgui breaks onChange on this input (doesn't work). That's why I had to wrap it with a div and a onClick listener.
36
- readOnly
37
- ></input>
38
- <label>{label}</label>
39
- </div>
40
- );
41
- };
1
+ import React from 'react';
2
+
3
+ interface IProps
4
+ extends React.DetailedHTMLProps<
5
+ React.InputHTMLAttributes<HTMLInputElement>,
6
+ HTMLInputElement
7
+ > {
8
+ label: string;
9
+ name: string;
10
+ value: string;
11
+ isChecked: boolean;
12
+ onRadioSelect: (value: string) => void;
13
+ }
14
+
15
+ export const InputRadio: React.FC<IProps> = ({
16
+ label,
17
+ name,
18
+ value,
19
+ isChecked,
20
+ onRadioSelect,
21
+ }) => {
22
+ const onRadioClick = (): void => {
23
+ onRadioSelect(value);
24
+ };
25
+
26
+ return (
27
+ <div onPointerUp={onRadioClick}>
28
+ <input
29
+ className="rpgui-radio"
30
+ name={name}
31
+ value={value}
32
+ type="radio"
33
+ data-rpguitype="radio"
34
+ checked={isChecked}
35
+ // rpgui breaks onChange on this input (doesn't work). That's why I had to wrap it with a div and a onClick listener.
36
+ readOnly
37
+ ></input>
38
+ <label>{label}</label>
39
+ </div>
40
+ );
41
+ };
@@ -1,298 +1,298 @@
1
- import { IItem } from '@rpg-engine/shared';
2
- import React from 'react';
3
- import styled from 'styled-components';
4
- import { uiColors } from '../../../constants/uiColors';
5
- import { uiFonts } from '../../../constants/uiFonts';
6
- import { SpriteFromAtlas } from '../../shared/SpriteFromAtlas';
7
- import { ErrorBoundary } from '../Inventory/ErrorBoundary';
8
- import { EquipmentSlotSpriteByType, rarityColor } from '../Inventory/ItemSlot';
9
-
10
- interface IItemInfoProps {
11
- item: IItem;
12
- itemToCompare?: IItem;
13
- atlasIMG: any;
14
- atlasJSON: any;
15
- }
16
-
17
- interface IItemStat {
18
- key: keyof IItem;
19
- label?: string;
20
- higherIsWorse?: boolean;
21
- }
22
-
23
- const statisticsToDisplay: IItemStat[] = [
24
- { key: 'attack' },
25
- { key: 'defense' },
26
- { key: 'maxRange', label: 'Range' },
27
- { key: 'weight', higherIsWorse: true },
28
- ];
29
-
30
- export const ItemInfo: React.FC<IItemInfoProps> = ({
31
- item,
32
- itemToCompare,
33
- atlasIMG,
34
- atlasJSON,
35
- }) => {
36
- const renderStatistics = () => {
37
- const statistics = [];
38
-
39
- for (const stat of statisticsToDisplay) {
40
- const itemStatistic = item[stat.key];
41
-
42
- if (itemStatistic) {
43
- const label =
44
- stat.label || stat.key[0].toUpperCase() + stat.key.slice(1);
45
-
46
- const isItemToCompare = !!itemToCompare;
47
-
48
- const isOnlyInOneItem = isItemToCompare && !itemToCompare?.[stat.key];
49
- const statDiff =
50
- parseInt(itemStatistic.toString()) -
51
- parseInt(itemToCompare?.[stat.key]?.toString() ?? '0');
52
-
53
- const isDifference = isItemToCompare && statDiff !== 0;
54
- const isBetter =
55
- (statDiff > 0 && !stat.higherIsWorse) ||
56
- (statDiff < 0 && stat.higherIsWorse);
57
-
58
- statistics.push(
59
- <Statistic key={stat.key} className={isOnlyInOneItem ? 'better' : ''}>
60
- <div className="label">{label}:</div>
61
- <div
62
- className={`value ${
63
- isDifference ? (isBetter ? 'better' : 'worse') : ''
64
- }`}
65
- >
66
- {`${itemStatistic.toString()} ${
67
- isDifference ? `(${statDiff > 0 ? '+' : ''}${statDiff})` : ''
68
- }`}
69
- </div>
70
- </Statistic>
71
- );
72
- }
73
- }
74
-
75
- return statistics;
76
- };
77
-
78
- const renderMissingStatistic = () => {
79
- const statistics = [];
80
-
81
- for (const stat of statisticsToDisplay) {
82
- const itemToCompareStatistic = itemToCompare?.[stat.key];
83
-
84
- if (itemToCompareStatistic && !item[stat.key]) {
85
- const label =
86
- stat.label || stat.key[0].toUpperCase() + stat.key.slice(1);
87
-
88
- statistics.push(
89
- <Statistic key={stat.key} className="worse">
90
- <div className="label">{label}:</div>
91
- <div className="value worse">
92
- {itemToCompareStatistic.toString()}
93
- </div>
94
- </Statistic>
95
- );
96
- }
97
- }
98
-
99
- return statistics;
100
- };
101
-
102
- const renderEntityEffects = () => {
103
- if (!item.entityEffects || !item.entityEffectChance) return null;
104
-
105
- return item.entityEffects.map((effect, index) => (
106
- <Statistic key={index} $isSpecial>
107
- {effect[0].toUpperCase() + effect.slice(1)} ({item.entityEffectChance}%)
108
- </Statistic>
109
- ));
110
- };
111
-
112
- const renderAvaibleSlots = () => {
113
- if (!item.allowedEquipSlotType) return null;
114
-
115
- return item.allowedEquipSlotType.map((slotType, index) => (
116
- <ErrorBoundary key={index}>
117
- <SpriteFromAtlas
118
- atlasIMG={atlasIMG}
119
- atlasJSON={atlasJSON}
120
- spriteKey={EquipmentSlotSpriteByType[slotType]}
121
- imgScale={2}
122
- grayScale={true}
123
- opacity={0.4}
124
- containerStyle={{ width: '32px', height: '32px' }}
125
- />
126
- </ErrorBoundary>
127
- ));
128
- };
129
-
130
- return (
131
- <Container item={item}>
132
- <Header>
133
- <div>
134
- <Title>{item.name}</Title>
135
- {item.rarity !== 'Common' && (
136
- <Rarity item={item}>{item.rarity}</Rarity>
137
- )}
138
- <Type>{item.subType}</Type>
139
- </div>
140
- <AllowedSlots>{renderAvaibleSlots()}</AllowedSlots>
141
- </Header>
142
-
143
- {item.minRequirements && (
144
- <LevelRequirement>
145
- <div className="title">Requirements:</div>
146
- <div>- Level: {item.minRequirements.level}</div>
147
- <div>
148
- -{' '}
149
- {item.minRequirements.skill.name[0].toUpperCase() +
150
- item.minRequirements.skill.name.slice(1)}
151
- : {item.minRequirements.skill.level}
152
- </div>
153
- </LevelRequirement>
154
- )}
155
-
156
- {renderStatistics()}
157
- {renderEntityEffects()}
158
- {item.usableEffectDescription && (
159
- <Statistic $isSpecial>{item.usableEffectDescription}</Statistic>
160
- )}
161
- {item.equippedBuffDescription && (
162
- <Statistic $isSpecial>{item.equippedBuffDescription}</Statistic>
163
- )}
164
- {item.isTwoHanded && <Statistic $isSpecial>Two handed</Statistic>}
165
-
166
- <Description>{item.description}</Description>
167
-
168
- {item.maxStackSize && item.maxStackSize !== 1 && (
169
- <StackInfo>
170
- x{Math.round((item.stackQty ?? 1) * 100) / 100}({item.maxStackSize})
171
- </StackInfo>
172
- )}
173
-
174
- {renderMissingStatistic().length > 0 && (
175
- <MissingStatistics>
176
- <Statistic>Equipped Diff</Statistic>
177
- {itemToCompare && renderMissingStatistic()}
178
- </MissingStatistics>
179
- )}
180
- </Container>
181
- );
182
- };
183
-
184
- const Container = styled.div<{ item: IItem }>`
185
- color: white;
186
- background-color: #222;
187
- border-radius: 5px;
188
- padding: 0.5rem;
189
- font-size: ${uiFonts.size.small};
190
- border: 3px solid ${({ item }) => rarityColor(item) ?? uiColors.lightGray};
191
- height: max-content;
192
- width: 18rem;
193
-
194
- @media (max-width: 640px) {
195
- width: 80vw;
196
- }
197
- `;
198
-
199
- const Title = styled.div`
200
- font-size: ${uiFonts.size.medium};
201
- font-weight: bold;
202
- margin-bottom: 0.5rem;
203
- display: flex;
204
- align-items: center;
205
- margin: 0;
206
- `;
207
-
208
- const Rarity = styled.div<{ item: IItem }>`
209
- font-size: ${uiFonts.size.small};
210
- font-weight: normal;
211
- margin-top: 0.2rem;
212
- color: ${({ item }) => rarityColor(item)};
213
- filter: brightness(1.5);
214
- `;
215
-
216
- const Type = styled.div`
217
- font-size: ${uiFonts.size.small};
218
- margin-top: 0.2rem;
219
- color: ${uiColors.lightGray};
220
- `;
221
-
222
- const LevelRequirement = styled.div`
223
- font-size: ${uiFonts.size.small};
224
- margin-top: 0.2rem;
225
- margin-bottom: 1rem;
226
- color: ${uiColors.orange};
227
-
228
- .title {
229
- margin-bottom: 4px;
230
- }
231
-
232
- div {
233
- margin-bottom: 2px;
234
- }
235
- `;
236
-
237
- const Statistic = styled.div<{ $isSpecial?: boolean }>`
238
- margin-bottom: 0.4rem;
239
- width: 100%;
240
- color: ${({ $isSpecial }) => ($isSpecial ? uiColors.darkYellow : 'inherit')};
241
-
242
- .label {
243
- display: inline-block;
244
- margin-right: 0.5rem;
245
- color: inherit;
246
- }
247
-
248
- .value {
249
- display: inline-block;
250
- color: inherit;
251
- }
252
-
253
- &.better,
254
- .better {
255
- color: ${uiColors.lightGreen};
256
- }
257
-
258
- &.worse,
259
- .worse {
260
- color: ${uiColors.cardinal};
261
- }
262
- `;
263
-
264
- const Description = styled.div`
265
- margin-top: 1.5rem;
266
- font-size: ${uiFonts.size.small};
267
- color: ${uiColors.lightGray};
268
- font-style: italic;
269
- `;
270
-
271
- const Header = styled.div`
272
- display: flex;
273
- align-items: center;
274
- justify-content: space-between;
275
- margin-bottom: 0.5rem;
276
- `;
277
-
278
- const AllowedSlots = styled.div`
279
- display: flex;
280
- align-items: center;
281
- justify-content: center;
282
- gap: 0.5rem;
283
- margin-left: auto;
284
- align-self: flex-start;
285
- `;
286
-
287
- const StackInfo = styled.div`
288
- width: 100%;
289
- text-align: right;
290
- font-size: ${uiFonts.size.small};
291
- color: ${uiColors.orange};
292
- margin-top: 1rem;
293
- `;
294
-
295
- const MissingStatistics = styled.div`
296
- margin-top: 1rem;
297
- color: ${uiColors.cardinal};
298
- `;
1
+ import { IItem } from '@rpg-engine/shared';
2
+ import React from 'react';
3
+ import styled from 'styled-components';
4
+ import { uiColors } from '../../../constants/uiColors';
5
+ import { uiFonts } from '../../../constants/uiFonts';
6
+ import { SpriteFromAtlas } from '../../shared/SpriteFromAtlas';
7
+ import { ErrorBoundary } from '../Inventory/ErrorBoundary';
8
+ import { EquipmentSlotSpriteByType, rarityColor } from '../Inventory/ItemSlot';
9
+
10
+ interface IItemInfoProps {
11
+ item: IItem;
12
+ itemToCompare?: IItem;
13
+ atlasIMG: any;
14
+ atlasJSON: any;
15
+ }
16
+
17
+ interface IItemStat {
18
+ key: keyof IItem;
19
+ label?: string;
20
+ higherIsWorse?: boolean;
21
+ }
22
+
23
+ const statisticsToDisplay: IItemStat[] = [
24
+ { key: 'attack' },
25
+ { key: 'defense' },
26
+ { key: 'maxRange', label: 'Range' },
27
+ { key: 'weight', higherIsWorse: true },
28
+ ];
29
+
30
+ export const ItemInfo: React.FC<IItemInfoProps> = ({
31
+ item,
32
+ itemToCompare,
33
+ atlasIMG,
34
+ atlasJSON,
35
+ }) => {
36
+ const renderStatistics = () => {
37
+ const statistics = [];
38
+
39
+ for (const stat of statisticsToDisplay) {
40
+ const itemStatistic = item[stat.key];
41
+
42
+ if (itemStatistic) {
43
+ const label =
44
+ stat.label || stat.key[0].toUpperCase() + stat.key.slice(1);
45
+
46
+ const isItemToCompare = !!itemToCompare;
47
+
48
+ const isOnlyInOneItem = isItemToCompare && !itemToCompare?.[stat.key];
49
+ const statDiff =
50
+ parseInt(itemStatistic.toString()) -
51
+ parseInt(itemToCompare?.[stat.key]?.toString() ?? '0');
52
+
53
+ const isDifference = isItemToCompare && statDiff !== 0;
54
+ const isBetter =
55
+ (statDiff > 0 && !stat.higherIsWorse) ||
56
+ (statDiff < 0 && stat.higherIsWorse);
57
+
58
+ statistics.push(
59
+ <Statistic key={stat.key} className={isOnlyInOneItem ? 'better' : ''}>
60
+ <div className="label">{label}:</div>
61
+ <div
62
+ className={`value ${
63
+ isDifference ? (isBetter ? 'better' : 'worse') : ''
64
+ }`}
65
+ >
66
+ {`${itemStatistic.toString()} ${
67
+ isDifference ? `(${statDiff > 0 ? '+' : ''}${statDiff})` : ''
68
+ }`}
69
+ </div>
70
+ </Statistic>
71
+ );
72
+ }
73
+ }
74
+
75
+ return statistics;
76
+ };
77
+
78
+ const renderMissingStatistic = () => {
79
+ const statistics = [];
80
+
81
+ for (const stat of statisticsToDisplay) {
82
+ const itemToCompareStatistic = itemToCompare?.[stat.key];
83
+
84
+ if (itemToCompareStatistic && !item[stat.key]) {
85
+ const label =
86
+ stat.label || stat.key[0].toUpperCase() + stat.key.slice(1);
87
+
88
+ statistics.push(
89
+ <Statistic key={stat.key} className="worse">
90
+ <div className="label">{label}:</div>
91
+ <div className="value worse">
92
+ {itemToCompareStatistic.toString()}
93
+ </div>
94
+ </Statistic>
95
+ );
96
+ }
97
+ }
98
+
99
+ return statistics;
100
+ };
101
+
102
+ const renderEntityEffects = () => {
103
+ if (!item.entityEffects || !item.entityEffectChance) return null;
104
+
105
+ return item.entityEffects.map((effect, index) => (
106
+ <Statistic key={index} $isSpecial>
107
+ {effect[0].toUpperCase() + effect.slice(1)} ({item.entityEffectChance}%)
108
+ </Statistic>
109
+ ));
110
+ };
111
+
112
+ const renderAvaibleSlots = () => {
113
+ if (!item.allowedEquipSlotType) return null;
114
+
115
+ return item.allowedEquipSlotType.map((slotType, index) => (
116
+ <ErrorBoundary key={index}>
117
+ <SpriteFromAtlas
118
+ atlasIMG={atlasIMG}
119
+ atlasJSON={atlasJSON}
120
+ spriteKey={EquipmentSlotSpriteByType[slotType]}
121
+ imgScale={2}
122
+ grayScale={true}
123
+ opacity={0.4}
124
+ containerStyle={{ width: '32px', height: '32px' }}
125
+ />
126
+ </ErrorBoundary>
127
+ ));
128
+ };
129
+
130
+ return (
131
+ <Container item={item}>
132
+ <Header>
133
+ <div>
134
+ <Title>{item.name}</Title>
135
+ {item.rarity !== 'Common' && (
136
+ <Rarity item={item}>{item.rarity}</Rarity>
137
+ )}
138
+ <Type>{item.subType}</Type>
139
+ </div>
140
+ <AllowedSlots>{renderAvaibleSlots()}</AllowedSlots>
141
+ </Header>
142
+
143
+ {item.minRequirements && (
144
+ <LevelRequirement>
145
+ <div className="title">Requirements:</div>
146
+ <div>- Level: {item.minRequirements.level}</div>
147
+ <div>
148
+ -{' '}
149
+ {item.minRequirements.skill.name[0].toUpperCase() +
150
+ item.minRequirements.skill.name.slice(1)}
151
+ : {item.minRequirements.skill.level}
152
+ </div>
153
+ </LevelRequirement>
154
+ )}
155
+
156
+ {renderStatistics()}
157
+ {renderEntityEffects()}
158
+ {item.usableEffectDescription && (
159
+ <Statistic $isSpecial>{item.usableEffectDescription}</Statistic>
160
+ )}
161
+ {item.equippedBuffDescription && (
162
+ <Statistic $isSpecial>{item.equippedBuffDescription}</Statistic>
163
+ )}
164
+ {item.isTwoHanded && <Statistic $isSpecial>Two handed</Statistic>}
165
+
166
+ <Description>{item.description}</Description>
167
+
168
+ {item.maxStackSize && item.maxStackSize !== 1 && (
169
+ <StackInfo>
170
+ x{Math.round((item.stackQty ?? 1) * 100) / 100}({item.maxStackSize})
171
+ </StackInfo>
172
+ )}
173
+
174
+ {renderMissingStatistic().length > 0 && (
175
+ <MissingStatistics>
176
+ <Statistic>Equipped Diff</Statistic>
177
+ {itemToCompare && renderMissingStatistic()}
178
+ </MissingStatistics>
179
+ )}
180
+ </Container>
181
+ );
182
+ };
183
+
184
+ const Container = styled.div<{ item: IItem }>`
185
+ color: white;
186
+ background-color: #222;
187
+ border-radius: 5px;
188
+ padding: 0.5rem;
189
+ font-size: ${uiFonts.size.small};
190
+ border: 3px solid ${({ item }) => rarityColor(item) ?? uiColors.lightGray};
191
+ height: max-content;
192
+ width: 18rem;
193
+
194
+ @media (max-width: 640px) {
195
+ width: 80vw;
196
+ }
197
+ `;
198
+
199
+ const Title = styled.div`
200
+ font-size: ${uiFonts.size.medium};
201
+ font-weight: bold;
202
+ margin-bottom: 0.5rem;
203
+ display: flex;
204
+ align-items: center;
205
+ margin: 0;
206
+ `;
207
+
208
+ const Rarity = styled.div<{ item: IItem }>`
209
+ font-size: ${uiFonts.size.small};
210
+ font-weight: normal;
211
+ margin-top: 0.2rem;
212
+ color: ${({ item }) => rarityColor(item)};
213
+ filter: brightness(1.5);
214
+ `;
215
+
216
+ const Type = styled.div`
217
+ font-size: ${uiFonts.size.small};
218
+ margin-top: 0.2rem;
219
+ color: ${uiColors.lightGray};
220
+ `;
221
+
222
+ const LevelRequirement = styled.div`
223
+ font-size: ${uiFonts.size.small};
224
+ margin-top: 0.2rem;
225
+ margin-bottom: 1rem;
226
+ color: ${uiColors.orange};
227
+
228
+ .title {
229
+ margin-bottom: 4px;
230
+ }
231
+
232
+ div {
233
+ margin-bottom: 2px;
234
+ }
235
+ `;
236
+
237
+ const Statistic = styled.div<{ $isSpecial?: boolean }>`
238
+ margin-bottom: 0.4rem;
239
+ width: 100%;
240
+ color: ${({ $isSpecial }) => ($isSpecial ? uiColors.darkYellow : 'inherit')};
241
+
242
+ .label {
243
+ display: inline-block;
244
+ margin-right: 0.5rem;
245
+ color: inherit;
246
+ }
247
+
248
+ .value {
249
+ display: inline-block;
250
+ color: inherit;
251
+ }
252
+
253
+ &.better,
254
+ .better {
255
+ color: ${uiColors.lightGreen};
256
+ }
257
+
258
+ &.worse,
259
+ .worse {
260
+ color: ${uiColors.cardinal};
261
+ }
262
+ `;
263
+
264
+ const Description = styled.div`
265
+ margin-top: 1.5rem;
266
+ font-size: ${uiFonts.size.small};
267
+ color: ${uiColors.lightGray};
268
+ font-style: italic;
269
+ `;
270
+
271
+ const Header = styled.div`
272
+ display: flex;
273
+ align-items: center;
274
+ justify-content: space-between;
275
+ margin-bottom: 0.5rem;
276
+ `;
277
+
278
+ const AllowedSlots = styled.div`
279
+ display: flex;
280
+ align-items: center;
281
+ justify-content: center;
282
+ gap: 0.5rem;
283
+ margin-left: auto;
284
+ align-self: flex-start;
285
+ `;
286
+
287
+ const StackInfo = styled.div`
288
+ width: 100%;
289
+ text-align: right;
290
+ font-size: ${uiFonts.size.small};
291
+ color: ${uiColors.orange};
292
+ margin-top: 1rem;
293
+ `;
294
+
295
+ const MissingStatistics = styled.div`
296
+ margin-top: 1rem;
297
+ color: ${uiColors.cardinal};
298
+ `;