@rpg-engine/long-bow 0.3.77 → 0.3.78

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 (160) hide show
  1. package/LICENSE +20 -20
  2. package/README.md +181 -181
  3. package/dist/components/CraftBook/CraftBook.d.ts +3 -1
  4. package/dist/components/Item/Cards/ItemInfo.d.ts +10 -0
  5. package/dist/components/Item/Cards/ItemInfoDisplay.d.ts +10 -0
  6. package/dist/components/Item/Cards/ItemInfoWrapper.d.ts +12 -0
  7. package/dist/components/Item/Cards/ItemTooltip.d.ts +7 -4
  8. package/dist/components/Item/Cards/MobileItemTooltip.d.ts +18 -0
  9. package/dist/components/Item/Inventory/ItemContainer.d.ts +2 -1
  10. package/dist/components/Item/Inventory/ItemSlot.d.ts +4 -1
  11. package/dist/components/TradingMenu/TradingItemRow.d.ts +3 -1
  12. package/dist/components/TradingMenu/TradingMenu.d.ts +6 -4
  13. package/dist/components/shared/SpriteFromAtlas.d.ts +1 -0
  14. package/dist/long-bow.cjs.development.js +1404 -1009
  15. package/dist/long-bow.cjs.development.js.map +1 -1
  16. package/dist/long-bow.cjs.production.min.js +1 -1
  17. package/dist/long-bow.cjs.production.min.js.map +1 -1
  18. package/dist/long-bow.esm.js +1406 -1013
  19. package/dist/long-bow.esm.js.map +1 -1
  20. package/dist/stories/ItemInfoDisplay.stories.d.ts +8 -0
  21. package/dist/stories/TradingMenu.stories.d.ts +2 -2
  22. package/package.json +100 -100
  23. package/src/components/Abstractions/SlotsContainer.tsx +45 -45
  24. package/src/components/Arrow/SelectArrow.tsx +69 -69
  25. package/src/components/Arrow/img/arrow01-left-clicked.png +0 -0
  26. package/src/components/Arrow/img/arrow01-left.png +0 -0
  27. package/src/components/Arrow/img/arrow01-right-clicked.png +0 -0
  28. package/src/components/Arrow/img/arrow01-right.png +0 -0
  29. package/src/components/Arrow/img/arrow02-left-clicked.png +0 -0
  30. package/src/components/Arrow/img/arrow02-left.png +0 -0
  31. package/src/components/Arrow/img/arrow02-right-clicked.png +0 -0
  32. package/src/components/Arrow/img/arrow02-right.png +0 -0
  33. package/src/components/Button.tsx +40 -40
  34. package/src/components/Character/CharacterSelection.tsx +96 -96
  35. package/src/components/CharacterStatus/CharacterStatus.tsx +120 -120
  36. package/src/components/Chat/Chat.tsx +195 -195
  37. package/src/components/Chatdeprecated/ChatDeprecated.tsx +198 -198
  38. package/src/components/CheckButton.tsx +65 -65
  39. package/src/components/CircularController/CircularController.tsx +248 -248
  40. package/src/components/CraftBook/CraftBook.tsx +240 -227
  41. package/src/components/CraftBook/MockItems.ts +77 -251
  42. package/src/components/DraggableContainer.tsx +153 -153
  43. package/src/components/Dropdown.tsx +90 -90
  44. package/src/components/DropdownSelectorContainer.tsx +42 -42
  45. package/src/components/Equipment/EquipmentSet.tsx +190 -190
  46. package/src/components/HistoryDialog.tsx +104 -104
  47. package/src/components/Input.tsx +15 -15
  48. package/src/components/Item/Cards/ItemInfo.tsx +252 -0
  49. package/src/components/Item/Cards/ItemInfoDisplay.tsx +128 -0
  50. package/src/components/Item/Cards/ItemInfoWrapper.tsx +62 -0
  51. package/src/components/Item/Cards/ItemTooltip.tsx +85 -33
  52. package/src/components/Item/Cards/MobileItemTooltip.tsx +134 -0
  53. package/src/components/Item/Inventory/ErrorBoundary.tsx +42 -42
  54. package/src/components/Item/Inventory/ItemContainer.tsx +214 -210
  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 +540 -501
  58. package/src/components/Item/Inventory/itemContainerHelper.ts +156 -156
  59. package/src/components/ListMenu.tsx +63 -63
  60. package/src/components/Multitab/Tab.tsx +66 -66
  61. package/src/components/Multitab/TabBody.tsx +13 -13
  62. package/src/components/Multitab/TabsContainer.tsx +97 -97
  63. package/src/components/NPCDialog/NPCDialog.tsx +121 -121
  64. package/src/components/NPCDialog/NPCDialogText.tsx +113 -113
  65. package/src/components/NPCDialog/NPCMultiDialog.tsx +159 -159
  66. package/src/components/NPCDialog/QuestionDialog/QuestionDialog.tsx +237 -237
  67. package/src/components/ProgressBar.tsx +92 -92
  68. package/src/components/PropertySelect/PropertySelect.tsx +106 -106
  69. package/src/components/QuestInfo/QuestInfo.tsx +230 -230
  70. package/src/components/QuestList.tsx +129 -129
  71. package/src/components/RPGUIContainer.tsx +47 -47
  72. package/src/components/RPGUIForceRenderStart.tsx +45 -45
  73. package/src/components/RPGUIRoot.tsx +14 -14
  74. package/src/components/RadioButton.tsx +53 -53
  75. package/src/components/RadioInput/RadioButton.tsx +96 -96
  76. package/src/components/RadioInput/RadioInput.tsx +102 -102
  77. package/src/components/RadioInput/instruments.ts +15 -15
  78. package/src/components/RangeSlider.tsx +78 -78
  79. package/src/components/RelativeListMenu.tsx +83 -83
  80. package/src/components/ScrollList.tsx +79 -79
  81. package/src/components/Shortcuts/Shortcuts.tsx +151 -151
  82. package/src/components/Shortcuts/ShortcutsSetter.tsx +132 -132
  83. package/src/components/Shortcuts/SingleShortcut.ts +62 -62
  84. package/src/components/SimpleProgressBar.tsx +62 -62
  85. package/src/components/SkillProgressBar.tsx +133 -133
  86. package/src/components/SkillsContainer.tsx +200 -200
  87. package/src/components/Spellbook/Spell.tsx +201 -201
  88. package/src/components/Spellbook/Spellbook.tsx +150 -150
  89. package/src/components/Spellbook/constants.ts +8 -8
  90. package/src/components/Spellbook/mockSpells.ts +60 -60
  91. package/src/components/StaticBook/StaticBook.tsx +103 -103
  92. package/src/components/TextArea.tsx +11 -11
  93. package/src/components/TimeWidget/DayNightPeriod/DayNightPeriod.tsx +35 -35
  94. package/src/components/TimeWidget/TimeWidget.tsx +63 -63
  95. package/src/components/TradingMenu/TradingItemRow.tsx +198 -181
  96. package/src/components/TradingMenu/TradingMenu.tsx +215 -203
  97. package/src/components/TradingMenu/items.mock.ts +48 -96
  98. package/src/components/Truncate.tsx +25 -25
  99. package/src/components/itemSelector/ItemSelector.tsx +136 -136
  100. package/src/components/shared/Column.tsx +16 -16
  101. package/src/components/shared/Ellipsis.tsx +65 -65
  102. package/src/components/shared/SpriteFromAtlas.tsx +104 -102
  103. package/src/components/typography/DynamicText.tsx +49 -49
  104. package/src/constants/uiColors.ts +20 -20
  105. package/src/constants/uiDevices.ts +3 -3
  106. package/src/constants/uiFonts.ts +12 -12
  107. package/src/hooks/useEventListener.ts +21 -21
  108. package/src/hooks/useOutsideAlerter.ts +25 -25
  109. package/src/index.tsx +40 -40
  110. package/src/libs/StringHelpers.ts +3 -3
  111. package/src/mocks/atlas/entities/entities.json +20215 -20215
  112. package/src/mocks/atlas/icons/icons.json +735 -735
  113. package/src/mocks/atlas/items/items.json +12086 -12086
  114. package/src/mocks/equipmentSet.mocks.ts +391 -393
  115. package/src/mocks/itemContainer.mocks.ts +563 -562
  116. package/src/mocks/skills.mocks.ts +128 -128
  117. package/src/stories/Arrow.stories.tsx +26 -26
  118. package/src/stories/Button.stories.tsx +36 -36
  119. package/src/stories/CharacterSelection.stories.tsx +45 -45
  120. package/src/stories/CharacterStatus.stories.tsx +29 -29
  121. package/src/stories/Chat.stories.tsx +187 -187
  122. package/src/stories/ChatDeprecated.stories.tsx +170 -170
  123. package/src/stories/CheckButton.stories.tsx +48 -48
  124. package/src/stories/CircullarController.stories.tsx +37 -37
  125. package/src/stories/CraftBook.stories.tsx +42 -40
  126. package/src/stories/DayNightPeriod.stories.tsx +27 -27
  127. package/src/stories/DraggableContainer.stories.tsx +28 -28
  128. package/src/stories/Dropdown.stories.tsx +46 -46
  129. package/src/stories/DropdownSelectorContainer.stories.tsx +41 -41
  130. package/src/stories/EquipmentSet.stories.tsx +65 -65
  131. package/src/stories/HistoryDialog.stories.tsx +61 -61
  132. package/src/stories/ItemContainer.stories.tsx +200 -198
  133. package/src/stories/ItemInfoDisplay.stories.tsx +33 -0
  134. package/src/stories/ItemQuantitySelector.stories.tsx +26 -26
  135. package/src/stories/ItemSelector.stories.tsx +77 -77
  136. package/src/stories/ItemTradingComponent.stories.tsx +35 -35
  137. package/src/stories/ListMenu.stories.tsx +56 -56
  138. package/src/stories/Multitab.stories.tsx +51 -51
  139. package/src/stories/NPCDialog.stories.tsx +130 -130
  140. package/src/stories/NPCMultiDialog.stories.tsx +71 -71
  141. package/src/stories/ProgressBar.stories.tsx +23 -23
  142. package/src/stories/PropertySelect.stories.tsx +40 -40
  143. package/src/stories/QuestInfo.stories.tsx +107 -107
  144. package/src/stories/QuestList.stories.tsx +82 -82
  145. package/src/stories/RPGUIContainers.stories.tsx +42 -42
  146. package/src/stories/RadioButton.stories.tsx +49 -49
  147. package/src/stories/RadioInput.stories.tsx +34 -34
  148. package/src/stories/RangeSlider.stories.tsx +64 -64
  149. package/src/stories/ScrollList.stories.tsx +85 -85
  150. package/src/stories/Shortcuts.stories.tsx +39 -39
  151. package/src/stories/SimpleProgressBar.stories.tsx +22 -22
  152. package/src/stories/SkillProgressBar.stories.tsx +34 -34
  153. package/src/stories/SkillsContainer.stories.tsx +35 -35
  154. package/src/stories/Spellbook.stories.tsx +104 -104
  155. package/src/stories/StaticBook.stories.tsx +32 -32
  156. package/src/stories/Text.stories.tsx +42 -42
  157. package/src/stories/TimeWidget.stories.tsx +27 -27
  158. package/src/stories/TradingMenu.stories.tsx +47 -45
  159. package/src/types/eventTypes.ts +4 -4
  160. package/src/types/index.d.ts +2 -2
@@ -1,104 +1,104 @@
1
- import React, { useEffect, useState } from 'react';
2
- import styled from 'styled-components';
3
- import { NPCDialog, NPCDialogType } from './NPCDialog/NPCDialog';
4
- import { NPCMultiDialog, NPCMultiDialogType } from './NPCDialog/NPCMultiDialog';
5
- import {
6
- IQuestionDialog,
7
- IQuestionDialogAnswer,
8
- QuestionDialog,
9
- } from './NPCDialog/QuestionDialog/QuestionDialog';
10
-
11
- export interface IHistoryDialogProps {
12
- backgroundImgPath: string[];
13
- fullCoverBackground: boolean;
14
- questions?: IQuestionDialog[];
15
- answers?: IQuestionDialogAnswer[];
16
- text?: string;
17
- imagePath?: string;
18
- textAndTypeArray?: NPCMultiDialogType[];
19
- onClose: () => void;
20
- }
21
-
22
- export const HistoryDialog: React.FC<IHistoryDialogProps> = ({
23
- backgroundImgPath,
24
- fullCoverBackground,
25
- questions,
26
- answers,
27
- text,
28
- imagePath,
29
- textAndTypeArray,
30
- onClose,
31
- }) => {
32
- const [img, setImage] = useState<number>(0);
33
- const onHandleSpacePress = (event: KeyboardEvent) => {
34
- if (event.code === 'Space') {
35
- if (img < backgroundImgPath?.length - 1) {
36
- setImage(prev => prev + 1);
37
- } else {
38
- // if there's no more text chunks, close the dialog
39
- onClose();
40
- }
41
- }
42
- };
43
-
44
- useEffect(() => {
45
- document.addEventListener('keydown', onHandleSpacePress);
46
-
47
- return () => document.removeEventListener('keydown', onHandleSpacePress);
48
- }, [backgroundImgPath]);
49
- return (
50
- <BackgroundContainer
51
- imgPath={backgroundImgPath[img]}
52
- fullImg={fullCoverBackground}
53
- >
54
- <DialogContainer>
55
- {textAndTypeArray ? (
56
- <NPCMultiDialog
57
- textAndTypeArray={textAndTypeArray}
58
- onClose={onClose}
59
- />
60
- ) : questions && answers ? (
61
- <QuestionDialog
62
- questions={questions}
63
- answers={answers}
64
- onClose={onClose}
65
- />
66
- ) : text && imagePath ? (
67
- <NPCDialog
68
- text={text}
69
- imagePath={imagePath}
70
- onClose={onClose}
71
- type={NPCDialogType.TextAndThumbnail}
72
- />
73
- ) : (
74
- <NPCDialog
75
- text={text}
76
- onClose={onClose}
77
- type={NPCDialogType.TextOnly}
78
- />
79
- )}
80
- </DialogContainer>
81
- </BackgroundContainer>
82
- );
83
- };
84
-
85
- interface IImgContainerProps {
86
- imgPath: string;
87
- fullImg: boolean;
88
- }
89
-
90
- const BackgroundContainer = styled.div<IImgContainerProps>`
91
- width: 100%;
92
- height: 100%;
93
- background-image: url(${props => props.imgPath});
94
- background-size: ${props => (props.imgPath ? 'cover' : 'auto')};
95
- display: flex;
96
- justify-content: space-evenly;
97
- align-items: center;
98
- `;
99
-
100
- const DialogContainer = styled.div`
101
- display: flex;
102
- justify-content: center;
103
- padding-top: 200px;
104
- `;
1
+ import React, { useEffect, useState } from 'react';
2
+ import styled from 'styled-components';
3
+ import { NPCDialog, NPCDialogType } from './NPCDialog/NPCDialog';
4
+ import { NPCMultiDialog, NPCMultiDialogType } from './NPCDialog/NPCMultiDialog';
5
+ import {
6
+ IQuestionDialog,
7
+ IQuestionDialogAnswer,
8
+ QuestionDialog,
9
+ } from './NPCDialog/QuestionDialog/QuestionDialog';
10
+
11
+ export interface IHistoryDialogProps {
12
+ backgroundImgPath: string[];
13
+ fullCoverBackground: boolean;
14
+ questions?: IQuestionDialog[];
15
+ answers?: IQuestionDialogAnswer[];
16
+ text?: string;
17
+ imagePath?: string;
18
+ textAndTypeArray?: NPCMultiDialogType[];
19
+ onClose: () => void;
20
+ }
21
+
22
+ export const HistoryDialog: React.FC<IHistoryDialogProps> = ({
23
+ backgroundImgPath,
24
+ fullCoverBackground,
25
+ questions,
26
+ answers,
27
+ text,
28
+ imagePath,
29
+ textAndTypeArray,
30
+ onClose,
31
+ }) => {
32
+ const [img, setImage] = useState<number>(0);
33
+ const onHandleSpacePress = (event: KeyboardEvent) => {
34
+ if (event.code === 'Space') {
35
+ if (img < backgroundImgPath?.length - 1) {
36
+ setImage(prev => prev + 1);
37
+ } else {
38
+ // if there's no more text chunks, close the dialog
39
+ onClose();
40
+ }
41
+ }
42
+ };
43
+
44
+ useEffect(() => {
45
+ document.addEventListener('keydown', onHandleSpacePress);
46
+
47
+ return () => document.removeEventListener('keydown', onHandleSpacePress);
48
+ }, [backgroundImgPath]);
49
+ return (
50
+ <BackgroundContainer
51
+ imgPath={backgroundImgPath[img]}
52
+ fullImg={fullCoverBackground}
53
+ >
54
+ <DialogContainer>
55
+ {textAndTypeArray ? (
56
+ <NPCMultiDialog
57
+ textAndTypeArray={textAndTypeArray}
58
+ onClose={onClose}
59
+ />
60
+ ) : questions && answers ? (
61
+ <QuestionDialog
62
+ questions={questions}
63
+ answers={answers}
64
+ onClose={onClose}
65
+ />
66
+ ) : text && imagePath ? (
67
+ <NPCDialog
68
+ text={text}
69
+ imagePath={imagePath}
70
+ onClose={onClose}
71
+ type={NPCDialogType.TextAndThumbnail}
72
+ />
73
+ ) : (
74
+ <NPCDialog
75
+ text={text}
76
+ onClose={onClose}
77
+ type={NPCDialogType.TextOnly}
78
+ />
79
+ )}
80
+ </DialogContainer>
81
+ </BackgroundContainer>
82
+ );
83
+ };
84
+
85
+ interface IImgContainerProps {
86
+ imgPath: string;
87
+ fullImg: boolean;
88
+ }
89
+
90
+ const BackgroundContainer = styled.div<IImgContainerProps>`
91
+ width: 100%;
92
+ height: 100%;
93
+ background-image: url(${props => props.imgPath});
94
+ background-size: ${props => (props.imgPath ? 'cover' : 'auto')};
95
+ display: flex;
96
+ justify-content: space-evenly;
97
+ align-items: center;
98
+ `;
99
+
100
+ const DialogContainer = styled.div`
101
+ display: flex;
102
+ justify-content: center;
103
+ padding-top: 200px;
104
+ `;
@@ -1,15 +1,15 @@
1
- import React from 'react';
2
-
3
- export interface IInputProps
4
- extends React.DetailedHTMLProps<
5
- React.InputHTMLAttributes<HTMLInputElement>,
6
- HTMLInputElement
7
- > {
8
- innerRef?: React.Ref<HTMLInputElement>;
9
- }
10
-
11
- export const Input: React.FC<IInputProps> = ({ ...props }) => {
12
- const { innerRef, ...rest } = props;
13
-
14
- return <input {...rest} ref={props.innerRef} />;
15
- };
1
+ import React from 'react';
2
+
3
+ export interface IInputProps
4
+ extends React.DetailedHTMLProps<
5
+ React.InputHTMLAttributes<HTMLInputElement>,
6
+ HTMLInputElement
7
+ > {
8
+ innerRef?: React.Ref<HTMLInputElement>;
9
+ }
10
+
11
+ export const Input: React.FC<IInputProps> = ({ ...props }) => {
12
+ const { innerRef, ...rest } = props;
13
+
14
+ return <input {...rest} ref={props.innerRef} />;
15
+ };
@@ -0,0 +1,252 @@
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 renderAvaibleSlots = () => {
103
+ if (!item.allowedEquipSlotType) return null;
104
+
105
+ return item.allowedEquipSlotType.map((slotType, index) => (
106
+ <ErrorBoundary key={index}>
107
+ <SpriteFromAtlas
108
+ atlasIMG={atlasIMG}
109
+ atlasJSON={atlasJSON}
110
+ spriteKey={EquipmentSlotSpriteByType[slotType]}
111
+ imgScale={2}
112
+ grayScale={true}
113
+ opacity={0.4}
114
+ containerStyle={{ width: '32px', height: '32px' }}
115
+ />
116
+ </ErrorBoundary>
117
+ ));
118
+ };
119
+
120
+ return (
121
+ <Container item={item}>
122
+ <Header>
123
+ <div>
124
+ <Title>{item.name}</Title>
125
+ {item.rarity !== 'Common' && (
126
+ <Rarity item={item}>{item.rarity}</Rarity>
127
+ )}
128
+ <Type>{item.subType}</Type>
129
+ </div>
130
+ <AllowedSlots>{renderAvaibleSlots()}</AllowedSlots>
131
+ </Header>
132
+
133
+ {renderStatistics()}
134
+ {item.isTwoHanded && <Statistic>Two handed</Statistic>}
135
+
136
+ <Description>{item.description}</Description>
137
+
138
+ {item.maxStackSize && item.maxStackSize !== 1 && (
139
+ <StackInfo>
140
+ x{item.stackQty ?? 1}({item.maxStackSize})
141
+ </StackInfo>
142
+ )}
143
+
144
+ {renderMissingStatistic().length > 0 && (
145
+ <MissingStatistics>
146
+ <Statistic>Equipped Diff</Statistic>
147
+ {itemToCompare && renderMissingStatistic()}
148
+ </MissingStatistics>
149
+ )}
150
+ </Container>
151
+ );
152
+ };
153
+
154
+ const Container = styled.div<{ item: IItem }>`
155
+ color: white;
156
+ background-color: #222;
157
+ border-radius: 5px;
158
+ padding: 0.5rem;
159
+ font-size: ${uiFonts.size.small};
160
+ border: 3px solid ${({ item }) => rarityColor(item) ?? uiColors.lightGray};
161
+ height: max-content;
162
+ width: 15rem;
163
+
164
+ @media (max-width: 580px) {
165
+ width: 80vw;
166
+ }
167
+ `;
168
+
169
+ const Title = styled.div`
170
+ font-size: ${uiFonts.size.medium};
171
+ font-weight: bold;
172
+ margin-bottom: 0.5rem;
173
+ display: flex;
174
+ align-items: center;
175
+ margin: 0;
176
+ `;
177
+
178
+ const Rarity = styled.div<{ item: IItem }>`
179
+ font-size: ${uiFonts.size.small};
180
+ font-weight: normal;
181
+ margin-top: 0.2rem;
182
+ color: ${({ item }) => rarityColor(item)};
183
+ filter: brightness(1.5);
184
+ `;
185
+
186
+ const Type = styled.div`
187
+ font-size: ${uiFonts.size.small};
188
+ margin-top: 0.2rem;
189
+ color: ${uiColors.lightGray};
190
+ `;
191
+
192
+ const Statistic = styled.div`
193
+ margin-bottom: 0.4rem;
194
+ width: max-content;
195
+
196
+ .label {
197
+ display: inline-block;
198
+ margin-right: 0.5rem;
199
+ color: inherit;
200
+ }
201
+
202
+ .value {
203
+ display: inline-block;
204
+ color: inherit;
205
+ }
206
+
207
+ &.better,
208
+ .better {
209
+ color: ${uiColors.lightGreen};
210
+ }
211
+
212
+ &.worse,
213
+ .worse {
214
+ color: ${uiColors.cardinal};
215
+ }
216
+ `;
217
+
218
+ const Description = styled.div`
219
+ margin-top: 1.5rem;
220
+ font-size: ${uiFonts.size.small};
221
+ color: ${uiColors.lightGray};
222
+ font-style: italic;
223
+ `;
224
+
225
+ const Header = styled.div`
226
+ display: flex;
227
+ align-items: center;
228
+ justify-content: space-between;
229
+ margin-bottom: 0.5rem;
230
+ `;
231
+
232
+ const AllowedSlots = styled.div`
233
+ display: flex;
234
+ align-items: center;
235
+ justify-content: center;
236
+ gap: 0.5rem;
237
+ margin-left: auto;
238
+ align-self: flex-start;
239
+ `;
240
+
241
+ const StackInfo = styled.div`
242
+ width: 100%;
243
+ text-align: right;
244
+ font-size: ${uiFonts.size.small};
245
+ color: ${uiColors.orange};
246
+ margin-top: 1rem;
247
+ `;
248
+
249
+ const MissingStatistics = styled.div`
250
+ margin-top: 1rem;
251
+ color: ${uiColors.cardinal};
252
+ `;
@@ -0,0 +1,128 @@
1
+ import { IEquipmentSet, IItem } from '@rpg-engine/shared';
2
+ import { camelCase } from 'lodash';
3
+ import React, { useMemo } from 'react';
4
+ import styled from 'styled-components';
5
+ import { ItemInfo } from './ItemInfo';
6
+
7
+ export interface IItemInfoDisplayProps {
8
+ item: IItem;
9
+ atlasIMG: any;
10
+ atlasJSON: any;
11
+ equipmentSet?: IEquipmentSet | null;
12
+ isMobile?: boolean;
13
+ }
14
+
15
+ type EquipmentSlotTypes =
16
+ | 'head'
17
+ | 'neck'
18
+ | 'leftHand'
19
+ | 'rightHand'
20
+ | 'ring'
21
+ | 'legs'
22
+ | 'boot'
23
+ | 'accessory'
24
+ | 'armor'
25
+ | 'inventory';
26
+
27
+ const itemSlotTypes: EquipmentSlotTypes[] = [
28
+ 'head',
29
+ 'neck',
30
+ 'leftHand',
31
+ 'rightHand',
32
+ 'ring',
33
+ 'legs',
34
+ 'boot',
35
+ 'accessory',
36
+ 'armor',
37
+ 'inventory',
38
+ ];
39
+
40
+ const getSlotType = (
41
+ itemSlotTypes: string[],
42
+ slotType: string,
43
+ subType: string
44
+ ): keyof IEquipmentSet => {
45
+ if (!itemSlotTypes.includes(slotType)) {
46
+ return subType as keyof IEquipmentSet;
47
+ }
48
+ return slotType as keyof IEquipmentSet;
49
+ };
50
+
51
+ export const ItemInfoDisplay: React.FC<IItemInfoDisplayProps> = ({
52
+ item,
53
+ atlasIMG,
54
+ atlasJSON,
55
+ equipmentSet,
56
+ isMobile,
57
+ }) => {
58
+ const itemToCompare = useMemo(() => {
59
+ if (equipmentSet && item.allowedEquipSlotType?.length) {
60
+ const allowedSlotTypeCamelCase = camelCase(item.allowedEquipSlotType[0]);
61
+ const itemSubTypeCamelCase = camelCase(item.subType);
62
+
63
+ const slotType = getSlotType(
64
+ itemSlotTypes,
65
+ allowedSlotTypeCamelCase,
66
+ itemSubTypeCamelCase
67
+ );
68
+
69
+ const itemFromEquipment = equipmentSet[slotType] as IItem;
70
+
71
+ if (
72
+ itemFromEquipment &&
73
+ (!item._id || itemFromEquipment._id !== item._id)
74
+ ) {
75
+ return itemFromEquipment;
76
+ }
77
+ }
78
+
79
+ return undefined;
80
+ }, [equipmentSet, item]);
81
+
82
+ return (
83
+ <Flex $isMobile={isMobile}>
84
+ <ItemInfo
85
+ item={item}
86
+ itemToCompare={itemToCompare}
87
+ atlasIMG={atlasIMG}
88
+ atlasJSON={atlasJSON}
89
+ />
90
+
91
+ {itemToCompare && (
92
+ <CompareContainer>
93
+ <Equipped>
94
+ <span>Equipped</span>
95
+ </Equipped>
96
+ <ItemInfo
97
+ item={itemToCompare}
98
+ itemToCompare={item}
99
+ atlasIMG={atlasIMG}
100
+ atlasJSON={atlasJSON}
101
+ />
102
+ </CompareContainer>
103
+ )}
104
+ </Flex>
105
+ );
106
+ };
107
+
108
+ const Flex = styled.div<{ $isMobile?: boolean }>`
109
+ display: flex;
110
+ gap: 0.5rem;
111
+ flex-direction: ${({ $isMobile }) => ($isMobile ? 'row-reverse' : 'row')};
112
+
113
+ @media (max-width: 580px) {
114
+ flex-direction: column-reverse;
115
+ align-items: center;
116
+ }
117
+ `;
118
+
119
+ const Equipped = styled.div`
120
+ position: absolute;
121
+ bottom: 100%;
122
+ left: 50%;
123
+ transform: translateX(-50%);
124
+ `;
125
+
126
+ const CompareContainer = styled.div`
127
+ position: relative;
128
+ `;
@@ -0,0 +1,62 @@
1
+ import { IEquipmentSet, IItem } from '@rpg-engine/shared';
2
+ import React, { useState } from 'react';
3
+ import { ItemTooltip } from './ItemTooltip';
4
+ import { MobileItemTooltip } from './MobileItemTooltip';
5
+
6
+ interface IItemInfoWrapperProps {
7
+ item: IItem;
8
+ children: React.ReactNode;
9
+ atlasIMG: any;
10
+ atlasJSON: any;
11
+ equipmentSet?: IEquipmentSet | null;
12
+ scale?: number;
13
+ }
14
+
15
+ export const ItemInfoWrapper: React.FC<IItemInfoWrapperProps> = ({
16
+ children,
17
+ atlasIMG,
18
+ atlasJSON,
19
+ item,
20
+ equipmentSet,
21
+ scale,
22
+ }) => {
23
+ const [isTooltipVisible, setIsTooltipVisible] = useState(false);
24
+ const [isTooltipMobileVisible, setIsTooltipMobileVisible] = useState(false);
25
+
26
+ return (
27
+ <div
28
+ onMouseEnter={() => {
29
+ if (!isTooltipMobileVisible) setIsTooltipVisible(true);
30
+ }}
31
+ onMouseLeave={setIsTooltipVisible.bind(null, false)}
32
+ onTouchEnd={() => {
33
+ setIsTooltipMobileVisible(true);
34
+ setIsTooltipVisible(false);
35
+ }}
36
+ >
37
+ {children}
38
+
39
+ {isTooltipVisible && !isTooltipMobileVisible && (
40
+ <ItemTooltip
41
+ atlasIMG={atlasIMG}
42
+ atlasJSON={atlasJSON}
43
+ equipmentSet={equipmentSet}
44
+ item={item}
45
+ />
46
+ )}
47
+ {isTooltipMobileVisible && (
48
+ <MobileItemTooltip
49
+ atlasIMG={atlasIMG}
50
+ atlasJSON={atlasJSON}
51
+ equipmentSet={equipmentSet}
52
+ closeTooltip={() => {
53
+ setIsTooltipMobileVisible(false);
54
+ console.log('close');
55
+ }}
56
+ item={item}
57
+ scale={scale}
58
+ />
59
+ )}
60
+ </div>
61
+ );
62
+ };