@rpg-engine/long-bow 0.4.2 → 0.4.3

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 (163) hide show
  1. package/LICENSE +20 -20
  2. package/README.md +181 -181
  3. package/dist/long-bow.cjs.development.js +1060 -1065
  4. package/dist/long-bow.cjs.development.js.map +1 -1
  5. package/dist/long-bow.cjs.production.min.js +1 -1
  6. package/dist/long-bow.cjs.production.min.js.map +1 -1
  7. package/dist/long-bow.esm.js +1066 -1071
  8. package/dist/long-bow.esm.js.map +1 -1
  9. package/package.json +100 -100
  10. package/src/components/Abstractions/ModalPortal.tsx +22 -22
  11. package/src/components/Abstractions/SlotsContainer.tsx +62 -62
  12. package/src/components/Arrow/SelectArrow.tsx +69 -69
  13. package/src/components/Arrow/img/arrow01-left-clicked.png +0 -0
  14. package/src/components/Arrow/img/arrow01-left.png +0 -0
  15. package/src/components/Arrow/img/arrow01-right-clicked.png +0 -0
  16. package/src/components/Arrow/img/arrow01-right.png +0 -0
  17. package/src/components/Arrow/img/arrow02-left-clicked.png +0 -0
  18. package/src/components/Arrow/img/arrow02-left.png +0 -0
  19. package/src/components/Arrow/img/arrow02-right-clicked.png +0 -0
  20. package/src/components/Arrow/img/arrow02-right.png +0 -0
  21. package/src/components/Button.tsx +40 -40
  22. package/src/components/Character/CharacterSelection.tsx +98 -98
  23. package/src/components/CharacterStatus/CharacterStatus.tsx +120 -120
  24. package/src/components/Chat/Chat.tsx +196 -196
  25. package/src/components/Chatdeprecated/ChatDeprecated.tsx +198 -198
  26. package/src/components/CheckButton.tsx +65 -65
  27. package/src/components/CircularController/CircularController.tsx +282 -282
  28. package/src/components/CraftBook/CraftBook.tsx +286 -286
  29. package/src/components/CraftBook/CraftingRecipe.tsx +161 -161
  30. package/src/components/CraftBook/MockItems.ts +101 -101
  31. package/src/components/DraggableContainer.tsx +181 -180
  32. package/src/components/DropdownSelectorContainer.tsx +42 -42
  33. package/src/components/Equipment/EquipmentSet.tsx +199 -199
  34. package/src/components/HistoryDialog.tsx +104 -104
  35. package/src/components/Input.tsx +15 -15
  36. package/src/components/InputRadio.tsx +41 -41
  37. package/src/components/Item/Cards/ItemInfo.tsx +298 -298
  38. package/src/components/Item/Cards/ItemInfoDisplay.tsx +135 -135
  39. package/src/components/Item/Cards/ItemInfoWrapper.tsx +62 -62
  40. package/src/components/Item/Cards/ItemTooltip.tsx +83 -83
  41. package/src/components/Item/Cards/MobileItemTooltip.tsx +149 -149
  42. package/src/components/Item/Inventory/ErrorBoundary.tsx +42 -42
  43. package/src/components/Item/Inventory/ItemContainer.tsx +231 -231
  44. package/src/components/Item/Inventory/ItemContainerTypes.ts +6 -6
  45. package/src/components/Item/Inventory/ItemQuantitySelector.tsx +138 -138
  46. package/src/components/Item/Inventory/ItemSlot.tsx +580 -580
  47. package/src/components/Item/Inventory/itemContainerHelper.ts +175 -175
  48. package/src/components/ListMenu.tsx +63 -63
  49. package/src/components/Marketplace/Marketplace.tsx +132 -132
  50. package/src/components/Marketplace/MarketplaceRows.tsx +171 -171
  51. package/src/components/Marketplace/__mocks__/index.tsx +65 -65
  52. package/src/components/Multitab/Tab.tsx +66 -66
  53. package/src/components/Multitab/TabBody.tsx +13 -13
  54. package/src/components/Multitab/TabsContainer.tsx +97 -97
  55. package/src/components/NPCDialog/NPCDialog.tsx +121 -121
  56. package/src/components/NPCDialog/NPCDialogText.tsx +113 -113
  57. package/src/components/NPCDialog/NPCMultiDialog.tsx +159 -159
  58. package/src/components/NPCDialog/QuestionDialog/QuestionDialog.tsx +237 -237
  59. package/src/components/ProgressBar.tsx +92 -95
  60. package/src/components/PropertySelect/PropertySelect.tsx +106 -106
  61. package/src/components/QuestInfo/QuestInfo.tsx +233 -233
  62. package/src/components/QuestList.tsx +135 -135
  63. package/src/components/RPGUIContainer.tsx +47 -47
  64. package/src/components/RPGUIForceRenderStart.tsx +45 -45
  65. package/src/components/RPGUIRoot.tsx +14 -14
  66. package/src/components/RadioButton.tsx +53 -53
  67. package/src/components/RadioInput/RadioButton.tsx +96 -96
  68. package/src/components/RadioInput/RadioInput.tsx +102 -102
  69. package/src/components/RadioInput/instruments.ts +15 -15
  70. package/src/components/RangeSlider.tsx +78 -78
  71. package/src/components/RelativeListMenu.tsx +90 -90
  72. package/src/components/ScrollList.tsx +79 -79
  73. package/src/components/Shortcuts/Shortcuts.tsx +193 -193
  74. package/src/components/Shortcuts/ShortcutsSetter.tsx +139 -139
  75. package/src/components/Shortcuts/SingleShortcut.ts +82 -82
  76. package/src/components/Shortcuts/useShortcutCooldown.ts +23 -23
  77. package/src/components/SimpleProgressBar.tsx +62 -62
  78. package/src/components/SkillProgressBar.tsx +133 -133
  79. package/src/components/SkillsContainer.tsx +217 -211
  80. package/src/components/Spellbook/Spell.tsx +236 -236
  81. package/src/components/Spellbook/Spellbook.tsx +161 -161
  82. package/src/components/Spellbook/cards/MobileSpellTooltip.tsx +137 -137
  83. package/src/components/Spellbook/cards/SpellInfo.tsx +134 -134
  84. package/src/components/Spellbook/cards/SpellInfoDisplay.tsx +31 -31
  85. package/src/components/Spellbook/cards/SpellInfoWrapper.tsx +48 -48
  86. package/src/components/Spellbook/cards/SpellTooltip.tsx +70 -70
  87. package/src/components/Spellbook/constants.ts +7 -7
  88. package/src/components/Spellbook/mockSpells.ts +84 -84
  89. package/src/components/StaticBook/StaticBook.tsx +103 -103
  90. package/src/components/TextArea.tsx +11 -11
  91. package/src/components/TimeWidget/DayNightPeriod/DayNightPeriod.tsx +35 -35
  92. package/src/components/TimeWidget/TimeWidget.tsx +65 -65
  93. package/src/components/TradingMenu/TradingItemRow.tsx +199 -199
  94. package/src/components/TradingMenu/TradingMenu.tsx +219 -219
  95. package/src/components/TradingMenu/items.mock.ts +48 -48
  96. package/src/components/Truncate.tsx +25 -25
  97. package/src/components/itemSelector/ItemSelector.tsx +136 -136
  98. package/src/components/shared/Column.tsx +16 -16
  99. package/src/components/shared/Ellipsis.tsx +68 -68
  100. package/src/components/shared/SpriteFromAtlas.tsx +104 -104
  101. package/src/components/typography/DynamicText.tsx +49 -49
  102. package/src/constants/uiColors.ts +20 -20
  103. package/src/constants/uiDevices.ts +3 -3
  104. package/src/constants/uiFonts.ts +12 -12
  105. package/src/hooks/useEventListener.ts +21 -21
  106. package/src/hooks/useOutsideAlerter.ts +25 -25
  107. package/src/index.tsx +42 -42
  108. package/src/libs/CastingTypeHelper.ts +7 -7
  109. package/src/libs/StringHelpers.ts +3 -3
  110. package/src/libs/itemCounter.ts +21 -21
  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 -391
  115. package/src/mocks/itemContainer.mocks.ts +605 -605
  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 +44 -44
  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 -42
  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 +201 -201
  133. package/src/stories/ItemInfoDisplay.stories.tsx +33 -33
  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/Marketplace.stories.tsx +42 -42
  139. package/src/stories/MarketplaceRows.stories.tsx +28 -28
  140. package/src/stories/Multitab.stories.tsx +51 -51
  141. package/src/stories/NPCDialog.stories.tsx +130 -130
  142. package/src/stories/NPCMultiDialog.stories.tsx +71 -71
  143. package/src/stories/ProgressBar.stories.tsx +23 -23
  144. package/src/stories/PropertySelect.stories.tsx +40 -40
  145. package/src/stories/QuestInfo.stories.tsx +107 -107
  146. package/src/stories/QuestList.stories.tsx +82 -82
  147. package/src/stories/RPGUIContainers.stories.tsx +42 -42
  148. package/src/stories/RadioButton.stories.tsx +49 -49
  149. package/src/stories/RadioInput.stories.tsx +34 -34
  150. package/src/stories/RangeSlider.stories.tsx +64 -64
  151. package/src/stories/ScrollList.stories.tsx +85 -85
  152. package/src/stories/Shortcuts.stories.tsx +39 -39
  153. package/src/stories/SimpleProgressBar.stories.tsx +22 -22
  154. package/src/stories/SkillProgressBar.stories.tsx +34 -34
  155. package/src/stories/SkillsContainer.stories.tsx +35 -35
  156. package/src/stories/SpellInfoDisplay.stories.tsx +27 -27
  157. package/src/stories/Spellbook.stories.tsx +104 -104
  158. package/src/stories/StaticBook.stories.tsx +32 -32
  159. package/src/stories/Text.stories.tsx +42 -42
  160. package/src/stories/TimeWidget.stories.tsx +27 -27
  161. package/src/stories/TradingMenu.stories.tsx +47 -47
  162. package/src/types/eventTypes.ts +4 -4
  163. package/src/types/index.d.ts +2 -2
@@ -1,286 +1,286 @@
1
- import {
2
- ICraftableItem,
3
- IEquipmentSet,
4
- IItemContainer,
5
- ISkill,
6
- ItemSubType,
7
- } from '@rpg-engine/shared';
8
- import React, { useEffect, useState } from 'react';
9
- import styled from 'styled-components';
10
- import { uiColors } from '../../constants/uiColors';
11
- import { Button, ButtonTypes } from '../Button';
12
- import { DraggableContainer } from '../DraggableContainer';
13
- import { InputRadio } from '../InputRadio';
14
- import { RPGUIContainerTypes } from '../RPGUIContainer';
15
- import { CraftingRecipe } from './CraftingRecipe';
16
-
17
- export interface IItemCraftSelectorProps {
18
- atlasJSON: any;
19
- atlasIMG: any;
20
- onClose: () => void;
21
- onSelect: (value: string) => void;
22
- onCraftItem: (value: string | undefined) => void;
23
- craftablesItems: ICraftableItem[];
24
- equipmentSet?: IEquipmentSet | null;
25
- inventory?: IItemContainer | null;
26
- scale?: number;
27
- skills?: ISkill | null;
28
- savedSelectedType?: string;
29
- }
30
-
31
- export interface ShowMessage {
32
- show: boolean;
33
- index: number;
34
- }
35
-
36
- const desktop = {
37
- width: 'min(900px, 80%)',
38
- height: 'min(700px, 80%)',
39
- };
40
-
41
- const mobileLanscape = {
42
- width: '800px',
43
- height: '500px',
44
- };
45
-
46
- const mobilePortrait = {
47
- width: '500px',
48
- height: '700px',
49
- };
50
-
51
- export const CraftBook: React.FC<IItemCraftSelectorProps> = ({
52
- atlasIMG,
53
- atlasJSON,
54
- onClose,
55
- onSelect,
56
- onCraftItem,
57
- craftablesItems,
58
- equipmentSet,
59
- scale,
60
- inventory,
61
- skills,
62
- savedSelectedType,
63
- }) => {
64
- const [craftItemKey, setCraftItemKey] = useState<string>();
65
- const [selectedType, setSelectedType] = useState<string>(
66
- savedSelectedType ?? Object.keys(ItemSubType)[0]
67
- );
68
- const [size, setSize] = useState<{ width: string; height: string }>();
69
-
70
- useEffect(() => {
71
- const handleResize = (): void => {
72
- if (
73
- window.innerWidth < 500 &&
74
- size?.width !== mobilePortrait.width &&
75
- (!scale || scale < 1)
76
- ) {
77
- setSize(mobilePortrait);
78
- } else if (
79
- (!scale || scale < 1) &&
80
- size?.width !== mobileLanscape.width
81
- ) {
82
- setSize(mobileLanscape);
83
- } else if (size?.width !== desktop.width) {
84
- setSize(desktop);
85
- }
86
- };
87
- handleResize();
88
-
89
- window.addEventListener('resize', handleResize);
90
-
91
- return () => window.removeEventListener('resize', handleResize);
92
- }, [scale]);
93
-
94
- const renderItemTypes = () => {
95
- const itemTypes = ['Suggested', ...Object.keys(ItemSubType)]
96
- .filter(type => type !== 'DeadBody')
97
- .sort((a, b) => {
98
- if (a === 'Suggested') return -1;
99
- if (b === 'Suggested') return 1;
100
- return a.localeCompare(b);
101
- });
102
-
103
- if (window.innerWidth > parseInt(mobilePortrait.width)) {
104
- return itemTypes.map(type => {
105
- return (
106
- <InputRadio
107
- key={type}
108
- value={type}
109
- label={type}
110
- name={type}
111
- isChecked={selectedType === type}
112
- onRadioSelect={value => {
113
- setSelectedType(value);
114
- onSelect(value);
115
- }}
116
- />
117
- );
118
- });
119
- }
120
-
121
- const rows: JSX.Element[][] = [[], []];
122
-
123
- itemTypes.forEach((type, index) => {
124
- let row = 0;
125
-
126
- if (index % 2 === 1) row = 1;
127
-
128
- rows[row].push(
129
- <InputRadio
130
- key={type}
131
- value={type}
132
- label={type}
133
- name={type}
134
- isChecked={selectedType === type}
135
- onRadioSelect={value => {
136
- setSelectedType(value);
137
- onSelect(value);
138
- }}
139
- />
140
- );
141
- });
142
-
143
- return rows.map((row, index) => (
144
- <div key={index} style={{ display: 'flex', gap: '10px' }}>
145
- {row}
146
- </div>
147
- ));
148
- };
149
-
150
- if (!size) return null;
151
-
152
- return (
153
- <DraggableContainer
154
- type={RPGUIContainerTypes.Framed}
155
- width={size.width}
156
- height={size.height}
157
- cancelDrag=".inputRadioCraftBook"
158
- onCloseButton={() => {
159
- if (onClose) {
160
- onClose();
161
- }
162
- }}
163
- scale={scale}
164
- >
165
- <Wrapper>
166
- <div style={{ width: '100%' }}>
167
- <Title>Craftbook</Title>
168
- <Subtitle>Select an item to craft</Subtitle>
169
- <hr className="golden" />
170
- </div>
171
-
172
- <ContentContainer>
173
- <ItemTypes className="inputRadioCraftBook">
174
- {renderItemTypes()}
175
- </ItemTypes>
176
-
177
- <RadioInputScroller className="inputRadioCraftBook">
178
- {craftablesItems?.map(item => (
179
- <CraftingRecipe
180
- key={item.key}
181
- atlasIMG={atlasIMG}
182
- atlasJSON={atlasJSON}
183
- equipmentSet={equipmentSet}
184
- recipe={item}
185
- scale={scale}
186
- handleRecipeSelect={setCraftItemKey.bind(null, item.key)}
187
- selectedCraftItemKey={craftItemKey}
188
- inventory={inventory}
189
- skills={skills}
190
- />
191
- ))}
192
- </RadioInputScroller>
193
- </ContentContainer>
194
- <ButtonWrapper>
195
- <Button buttonType={ButtonTypes.RPGUIButton} onPointerDown={onClose}>
196
- Cancel
197
- </Button>
198
- <Button
199
- disabled={!craftItemKey}
200
- buttonType={ButtonTypes.RPGUIButton}
201
- onPointerDown={() => onCraftItem(craftItemKey)}
202
- >
203
- Craft
204
- </Button>
205
- </ButtonWrapper>
206
- </Wrapper>
207
- </DraggableContainer>
208
- );
209
- };
210
-
211
- const Wrapper = styled.div`
212
- display: flex;
213
- flex-direction: column;
214
- width: 100%;
215
- height: 100%;
216
- `;
217
-
218
- const Title = styled.h1`
219
- font-size: 0.6rem;
220
- color: ${uiColors.yellow} !important;
221
- `;
222
-
223
- const Subtitle = styled.h1`
224
- font-size: 0.4rem;
225
- color: ${uiColors.yellow} !important;
226
- `;
227
-
228
- const RadioInputScroller = styled.div`
229
- padding-left: 15px;
230
- padding-top: 10px;
231
- margin-top: 1rem;
232
- align-items: center;
233
- align-items: flex-start;
234
- overflow-y: scroll;
235
- min-height: 0;
236
- flex: 1;
237
- margin-left: 10px;
238
- -webkit-overflow-scrolling: touch;
239
-
240
- @media (max-width: ${mobilePortrait.width}) {
241
- margin-left: 0;
242
- }
243
- `;
244
-
245
- const ButtonWrapper = styled.div`
246
- display: flex;
247
- justify-content: flex-end;
248
- margin-top: 10px;
249
- width: 100%;
250
-
251
- button {
252
- padding: 0px 50px;
253
- margin: 5px;
254
- }
255
-
256
- @media (max-width: ${mobilePortrait.width}) {
257
- justify-content: center;
258
- }
259
- `;
260
-
261
- const ContentContainer = styled.div`
262
- display: flex;
263
- width: 100%;
264
- min-height: 0;
265
- flex: 1;
266
-
267
- @media (max-width: ${mobilePortrait.width}) {
268
- flex-direction: column;
269
- }
270
- `;
271
-
272
- const ItemTypes = styled.div`
273
- display: flex;
274
- overflow-y: scroll;
275
- overflow-x: hidden;
276
- width: max-content;
277
- flex-direction: column;
278
- padding-right: 5px;
279
-
280
- @media (max-width: ${mobilePortrait.width}) {
281
- overflow-x: scroll;
282
- overflow-y: hidden;
283
- padding-right: 0;
284
- width: 100%;
285
- }
286
- `;
1
+ import {
2
+ ICraftableItem,
3
+ IEquipmentSet,
4
+ IItemContainer,
5
+ ISkill,
6
+ ItemSubType,
7
+ } from '@rpg-engine/shared';
8
+ import React, { useEffect, useState } from 'react';
9
+ import styled from 'styled-components';
10
+ import { uiColors } from '../../constants/uiColors';
11
+ import { Button, ButtonTypes } from '../Button';
12
+ import { DraggableContainer } from '../DraggableContainer';
13
+ import { InputRadio } from '../InputRadio';
14
+ import { RPGUIContainerTypes } from '../RPGUIContainer';
15
+ import { CraftingRecipe } from './CraftingRecipe';
16
+
17
+ export interface IItemCraftSelectorProps {
18
+ atlasJSON: any;
19
+ atlasIMG: any;
20
+ onClose: () => void;
21
+ onSelect: (value: string) => void;
22
+ onCraftItem: (value: string | undefined) => void;
23
+ craftablesItems: ICraftableItem[];
24
+ equipmentSet?: IEquipmentSet | null;
25
+ inventory?: IItemContainer | null;
26
+ scale?: number;
27
+ skills?: ISkill | null;
28
+ savedSelectedType?: string;
29
+ }
30
+
31
+ export interface ShowMessage {
32
+ show: boolean;
33
+ index: number;
34
+ }
35
+
36
+ const desktop = {
37
+ width: 'min(900px, 80%)',
38
+ height: 'min(700px, 80%)',
39
+ };
40
+
41
+ const mobileLanscape = {
42
+ width: '800px',
43
+ height: '500px',
44
+ };
45
+
46
+ const mobilePortrait = {
47
+ width: '500px',
48
+ height: '700px',
49
+ };
50
+
51
+ export const CraftBook: React.FC<IItemCraftSelectorProps> = ({
52
+ atlasIMG,
53
+ atlasJSON,
54
+ onClose,
55
+ onSelect,
56
+ onCraftItem,
57
+ craftablesItems,
58
+ equipmentSet,
59
+ scale,
60
+ inventory,
61
+ skills,
62
+ savedSelectedType,
63
+ }) => {
64
+ const [craftItemKey, setCraftItemKey] = useState<string>();
65
+ const [selectedType, setSelectedType] = useState<string>(
66
+ savedSelectedType ?? Object.keys(ItemSubType)[0]
67
+ );
68
+ const [size, setSize] = useState<{ width: string; height: string }>();
69
+
70
+ useEffect(() => {
71
+ const handleResize = (): void => {
72
+ if (
73
+ window.innerWidth < 500 &&
74
+ size?.width !== mobilePortrait.width &&
75
+ (!scale || scale < 1)
76
+ ) {
77
+ setSize(mobilePortrait);
78
+ } else if (
79
+ (!scale || scale < 1) &&
80
+ size?.width !== mobileLanscape.width
81
+ ) {
82
+ setSize(mobileLanscape);
83
+ } else if (size?.width !== desktop.width) {
84
+ setSize(desktop);
85
+ }
86
+ };
87
+ handleResize();
88
+
89
+ window.addEventListener('resize', handleResize);
90
+
91
+ return () => window.removeEventListener('resize', handleResize);
92
+ }, [scale]);
93
+
94
+ const renderItemTypes = () => {
95
+ const itemTypes = ['Suggested', ...Object.keys(ItemSubType)]
96
+ .filter(type => type !== 'DeadBody')
97
+ .sort((a, b) => {
98
+ if (a === 'Suggested') return -1;
99
+ if (b === 'Suggested') return 1;
100
+ return a.localeCompare(b);
101
+ });
102
+
103
+ if (window.innerWidth > parseInt(mobilePortrait.width)) {
104
+ return itemTypes.map(type => {
105
+ return (
106
+ <InputRadio
107
+ key={type}
108
+ value={type}
109
+ label={type}
110
+ name={type}
111
+ isChecked={selectedType === type}
112
+ onRadioSelect={value => {
113
+ setSelectedType(value);
114
+ onSelect(value);
115
+ }}
116
+ />
117
+ );
118
+ });
119
+ }
120
+
121
+ const rows: JSX.Element[][] = [[], []];
122
+
123
+ itemTypes.forEach((type, index) => {
124
+ let row = 0;
125
+
126
+ if (index % 2 === 1) row = 1;
127
+
128
+ rows[row].push(
129
+ <InputRadio
130
+ key={type}
131
+ value={type}
132
+ label={type}
133
+ name={type}
134
+ isChecked={selectedType === type}
135
+ onRadioSelect={value => {
136
+ setSelectedType(value);
137
+ onSelect(value);
138
+ }}
139
+ />
140
+ );
141
+ });
142
+
143
+ return rows.map((row, index) => (
144
+ <div key={index} style={{ display: 'flex', gap: '10px' }}>
145
+ {row}
146
+ </div>
147
+ ));
148
+ };
149
+
150
+ if (!size) return null;
151
+
152
+ return (
153
+ <DraggableContainer
154
+ type={RPGUIContainerTypes.Framed}
155
+ width={size.width}
156
+ height={size.height}
157
+ cancelDrag=".inputRadioCraftBook"
158
+ onCloseButton={() => {
159
+ if (onClose) {
160
+ onClose();
161
+ }
162
+ }}
163
+ scale={scale}
164
+ >
165
+ <Wrapper>
166
+ <div style={{ width: '100%' }}>
167
+ <Title>Craftbook</Title>
168
+ <Subtitle>Select an item to craft</Subtitle>
169
+ <hr className="golden" />
170
+ </div>
171
+
172
+ <ContentContainer>
173
+ <ItemTypes className="inputRadioCraftBook">
174
+ {renderItemTypes()}
175
+ </ItemTypes>
176
+
177
+ <RadioInputScroller className="inputRadioCraftBook">
178
+ {craftablesItems?.map(item => (
179
+ <CraftingRecipe
180
+ key={item.key}
181
+ atlasIMG={atlasIMG}
182
+ atlasJSON={atlasJSON}
183
+ equipmentSet={equipmentSet}
184
+ recipe={item}
185
+ scale={scale}
186
+ handleRecipeSelect={setCraftItemKey.bind(null, item.key)}
187
+ selectedCraftItemKey={craftItemKey}
188
+ inventory={inventory}
189
+ skills={skills}
190
+ />
191
+ ))}
192
+ </RadioInputScroller>
193
+ </ContentContainer>
194
+ <ButtonWrapper>
195
+ <Button buttonType={ButtonTypes.RPGUIButton} onPointerDown={onClose}>
196
+ Cancel
197
+ </Button>
198
+ <Button
199
+ disabled={!craftItemKey}
200
+ buttonType={ButtonTypes.RPGUIButton}
201
+ onPointerDown={() => onCraftItem(craftItemKey)}
202
+ >
203
+ Craft
204
+ </Button>
205
+ </ButtonWrapper>
206
+ </Wrapper>
207
+ </DraggableContainer>
208
+ );
209
+ };
210
+
211
+ const Wrapper = styled.div`
212
+ display: flex;
213
+ flex-direction: column;
214
+ width: 100%;
215
+ height: 100%;
216
+ `;
217
+
218
+ const Title = styled.h1`
219
+ font-size: 0.6rem;
220
+ color: ${uiColors.yellow} !important;
221
+ `;
222
+
223
+ const Subtitle = styled.h1`
224
+ font-size: 0.4rem;
225
+ color: ${uiColors.yellow} !important;
226
+ `;
227
+
228
+ const RadioInputScroller = styled.div`
229
+ padding-left: 15px;
230
+ padding-top: 10px;
231
+ margin-top: 1rem;
232
+ align-items: center;
233
+ align-items: flex-start;
234
+ overflow-y: scroll;
235
+ min-height: 0;
236
+ flex: 1;
237
+ margin-left: 10px;
238
+ -webkit-overflow-scrolling: touch;
239
+
240
+ @media (max-width: ${mobilePortrait.width}) {
241
+ margin-left: 0;
242
+ }
243
+ `;
244
+
245
+ const ButtonWrapper = styled.div`
246
+ display: flex;
247
+ justify-content: flex-end;
248
+ margin-top: 10px;
249
+ width: 100%;
250
+
251
+ button {
252
+ padding: 0px 50px;
253
+ margin: 5px;
254
+ }
255
+
256
+ @media (max-width: ${mobilePortrait.width}) {
257
+ justify-content: center;
258
+ }
259
+ `;
260
+
261
+ const ContentContainer = styled.div`
262
+ display: flex;
263
+ width: 100%;
264
+ min-height: 0;
265
+ flex: 1;
266
+
267
+ @media (max-width: ${mobilePortrait.width}) {
268
+ flex-direction: column;
269
+ }
270
+ `;
271
+
272
+ const ItemTypes = styled.div`
273
+ display: flex;
274
+ overflow-y: scroll;
275
+ overflow-x: hidden;
276
+ width: max-content;
277
+ flex-direction: column;
278
+ padding-right: 5px;
279
+
280
+ @media (max-width: ${mobilePortrait.width}) {
281
+ overflow-x: scroll;
282
+ overflow-y: hidden;
283
+ padding-right: 0;
284
+ width: 100%;
285
+ }
286
+ `;