@archireport/react-native-drawing 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (323) hide show
  1. package/README.md +181 -0
  2. package/lib/commonjs/DrawingEditor.js +815 -0
  3. package/lib/commonjs/DrawingEditor.js.map +1 -0
  4. package/lib/commonjs/assets/toolbar-icons/arrow-disabled.png +0 -0
  5. package/lib/commonjs/assets/toolbar-icons/arrow-enabled.png +0 -0
  6. package/lib/commonjs/assets/toolbar-icons/arrow.png +0 -0
  7. package/lib/commonjs/assets/toolbar-icons/circle-disabled.png +0 -0
  8. package/lib/commonjs/assets/toolbar-icons/circle-enabled.png +0 -0
  9. package/lib/commonjs/assets/toolbar-icons/circle.png +0 -0
  10. package/lib/commonjs/assets/toolbar-icons/freehand-disabled.png +0 -0
  11. package/lib/commonjs/assets/toolbar-icons/freehand-enabled.png +0 -0
  12. package/lib/commonjs/assets/toolbar-icons/freehand.png +0 -0
  13. package/lib/commonjs/assets/toolbar-icons/line-disabled.png +0 -0
  14. package/lib/commonjs/assets/toolbar-icons/line-enabled.png +0 -0
  15. package/lib/commonjs/assets/toolbar-icons/line.png +0 -0
  16. package/lib/commonjs/assets/toolbar-icons/measure-disabled.png +0 -0
  17. package/lib/commonjs/assets/toolbar-icons/measure-enabled.png +0 -0
  18. package/lib/commonjs/assets/toolbar-icons/measure.png +0 -0
  19. package/lib/commonjs/assets/toolbar-icons/move-disabled.png +0 -0
  20. package/lib/commonjs/assets/toolbar-icons/move-enabled.png +0 -0
  21. package/lib/commonjs/assets/toolbar-icons/move.png +0 -0
  22. package/lib/commonjs/assets/toolbar-icons/polygon-disabled.png +0 -0
  23. package/lib/commonjs/assets/toolbar-icons/polygon-enabled.png +0 -0
  24. package/lib/commonjs/assets/toolbar-icons/polygon.png +0 -0
  25. package/lib/commonjs/assets/toolbar-icons/rectangle-disabled.png +0 -0
  26. package/lib/commonjs/assets/toolbar-icons/rectangle-enabled.png +0 -0
  27. package/lib/commonjs/assets/toolbar-icons/rectangle.png +0 -0
  28. package/lib/commonjs/assets/toolbar-icons/text-disabled.png +0 -0
  29. package/lib/commonjs/assets/toolbar-icons/text-enabled.png +0 -0
  30. package/lib/commonjs/assets/toolbar-icons/text.png +0 -0
  31. package/lib/commonjs/components/ColorPalette.js +379 -0
  32. package/lib/commonjs/components/ColorPalette.js.map +1 -0
  33. package/lib/commonjs/components/LineWidthSlider.js +70 -0
  34. package/lib/commonjs/components/LineWidthSlider.js.map +1 -0
  35. package/lib/commonjs/components/MeasurementEditModal.js +153 -0
  36. package/lib/commonjs/components/MeasurementEditModal.js.map +1 -0
  37. package/lib/commonjs/components/MiniMap.js +244 -0
  38. package/lib/commonjs/components/MiniMap.js.map +1 -0
  39. package/lib/commonjs/components/TextAnnotation.js +162 -0
  40. package/lib/commonjs/components/TextAnnotation.js.map +1 -0
  41. package/lib/commonjs/components/TextEditModal.js +133 -0
  42. package/lib/commonjs/components/TextEditModal.js.map +1 -0
  43. package/lib/commonjs/components/Toolbar.js +198 -0
  44. package/lib/commonjs/components/Toolbar.js.map +1 -0
  45. package/lib/commonjs/components/ZoomBadge.js +161 -0
  46. package/lib/commonjs/components/ZoomBadge.js.map +1 -0
  47. package/lib/commonjs/hooks/useFreehandGesture.js +173 -0
  48. package/lib/commonjs/hooks/useFreehandGesture.js.map +1 -0
  49. package/lib/commonjs/hooks/usePolygonGesture.js +109 -0
  50. package/lib/commonjs/hooks/usePolygonGesture.js.map +1 -0
  51. package/lib/commonjs/hooks/useSelectionGesture.js +236 -0
  52. package/lib/commonjs/hooks/useSelectionGesture.js.map +1 -0
  53. package/lib/commonjs/hooks/useShapeGesture.js +181 -0
  54. package/lib/commonjs/hooks/useShapeGesture.js.map +1 -0
  55. package/lib/commonjs/hooks/useViewportGesture.js +238 -0
  56. package/lib/commonjs/hooks/useViewportGesture.js.map +1 -0
  57. package/lib/commonjs/index.js +104 -0
  58. package/lib/commonjs/index.js.map +1 -0
  59. package/lib/commonjs/package.json +1 -0
  60. package/lib/commonjs/renderers/ArrowRenderer.js +118 -0
  61. package/lib/commonjs/renderers/ArrowRenderer.js.map +1 -0
  62. package/lib/commonjs/renderers/CircleRenderer.js +51 -0
  63. package/lib/commonjs/renderers/CircleRenderer.js.map +1 -0
  64. package/lib/commonjs/renderers/FreehandRenderer.js +31 -0
  65. package/lib/commonjs/renderers/FreehandRenderer.js.map +1 -0
  66. package/lib/commonjs/renderers/InProgressRenderer.js +174 -0
  67. package/lib/commonjs/renderers/InProgressRenderer.js.map +1 -0
  68. package/lib/commonjs/renderers/LineRenderer.js +27 -0
  69. package/lib/commonjs/renderers/LineRenderer.js.map +1 -0
  70. package/lib/commonjs/renderers/MeasurementRenderer.js +134 -0
  71. package/lib/commonjs/renderers/MeasurementRenderer.js.map +1 -0
  72. package/lib/commonjs/renderers/ObjectRenderer.js +65 -0
  73. package/lib/commonjs/renderers/ObjectRenderer.js.map +1 -0
  74. package/lib/commonjs/renderers/PolygonRenderer.js +46 -0
  75. package/lib/commonjs/renderers/PolygonRenderer.js.map +1 -0
  76. package/lib/commonjs/renderers/RectRenderer.js +51 -0
  77. package/lib/commonjs/renderers/RectRenderer.js.map +1 -0
  78. package/lib/commonjs/renderers/SelectedObjectRenderer.js +592 -0
  79. package/lib/commonjs/renderers/SelectedObjectRenderer.js.map +1 -0
  80. package/lib/commonjs/renderers/SelectionOverlay.js +120 -0
  81. package/lib/commonjs/renderers/SelectionOverlay.js.map +1 -0
  82. package/lib/commonjs/store/useDrawingStore.js +354 -0
  83. package/lib/commonjs/store/useDrawingStore.js.map +1 -0
  84. package/lib/commonjs/types.js +6 -0
  85. package/lib/commonjs/types.js.map +1 -0
  86. package/lib/commonjs/utils/colors.js +44 -0
  87. package/lib/commonjs/utils/colors.js.map +1 -0
  88. package/lib/commonjs/utils/coordinates.js +81 -0
  89. package/lib/commonjs/utils/coordinates.js.map +1 -0
  90. package/lib/commonjs/utils/hitTesting.js +181 -0
  91. package/lib/commonjs/utils/hitTesting.js.map +1 -0
  92. package/lib/commonjs/utils/serialization.js +42 -0
  93. package/lib/commonjs/utils/serialization.js.map +1 -0
  94. package/lib/commonjs/utils/shapeDetection.js +151 -0
  95. package/lib/commonjs/utils/shapeDetection.js.map +1 -0
  96. package/lib/commonjs/utils/smoothing.js +85 -0
  97. package/lib/commonjs/utils/smoothing.js.map +1 -0
  98. package/lib/module/DrawingEditor.js +811 -0
  99. package/lib/module/DrawingEditor.js.map +1 -0
  100. package/lib/module/assets/toolbar-icons/arrow-disabled.png +0 -0
  101. package/lib/module/assets/toolbar-icons/arrow-enabled.png +0 -0
  102. package/lib/module/assets/toolbar-icons/arrow.png +0 -0
  103. package/lib/module/assets/toolbar-icons/circle-disabled.png +0 -0
  104. package/lib/module/assets/toolbar-icons/circle-enabled.png +0 -0
  105. package/lib/module/assets/toolbar-icons/circle.png +0 -0
  106. package/lib/module/assets/toolbar-icons/freehand-disabled.png +0 -0
  107. package/lib/module/assets/toolbar-icons/freehand-enabled.png +0 -0
  108. package/lib/module/assets/toolbar-icons/freehand.png +0 -0
  109. package/lib/module/assets/toolbar-icons/line-disabled.png +0 -0
  110. package/lib/module/assets/toolbar-icons/line-enabled.png +0 -0
  111. package/lib/module/assets/toolbar-icons/line.png +0 -0
  112. package/lib/module/assets/toolbar-icons/measure-disabled.png +0 -0
  113. package/lib/module/assets/toolbar-icons/measure-enabled.png +0 -0
  114. package/lib/module/assets/toolbar-icons/measure.png +0 -0
  115. package/lib/module/assets/toolbar-icons/move-disabled.png +0 -0
  116. package/lib/module/assets/toolbar-icons/move-enabled.png +0 -0
  117. package/lib/module/assets/toolbar-icons/move.png +0 -0
  118. package/lib/module/assets/toolbar-icons/polygon-disabled.png +0 -0
  119. package/lib/module/assets/toolbar-icons/polygon-enabled.png +0 -0
  120. package/lib/module/assets/toolbar-icons/polygon.png +0 -0
  121. package/lib/module/assets/toolbar-icons/rectangle-disabled.png +0 -0
  122. package/lib/module/assets/toolbar-icons/rectangle-enabled.png +0 -0
  123. package/lib/module/assets/toolbar-icons/rectangle.png +0 -0
  124. package/lib/module/assets/toolbar-icons/text-disabled.png +0 -0
  125. package/lib/module/assets/toolbar-icons/text-enabled.png +0 -0
  126. package/lib/module/assets/toolbar-icons/text.png +0 -0
  127. package/lib/module/components/ColorPalette.js +374 -0
  128. package/lib/module/components/ColorPalette.js.map +1 -0
  129. package/lib/module/components/LineWidthSlider.js +64 -0
  130. package/lib/module/components/LineWidthSlider.js.map +1 -0
  131. package/lib/module/components/MeasurementEditModal.js +148 -0
  132. package/lib/module/components/MeasurementEditModal.js.map +1 -0
  133. package/lib/module/components/MiniMap.js +239 -0
  134. package/lib/module/components/MiniMap.js.map +1 -0
  135. package/lib/module/components/TextAnnotation.js +157 -0
  136. package/lib/module/components/TextAnnotation.js.map +1 -0
  137. package/lib/module/components/TextEditModal.js +128 -0
  138. package/lib/module/components/TextEditModal.js.map +1 -0
  139. package/lib/module/components/Toolbar.js +193 -0
  140. package/lib/module/components/Toolbar.js.map +1 -0
  141. package/lib/module/components/ZoomBadge.js +155 -0
  142. package/lib/module/components/ZoomBadge.js.map +1 -0
  143. package/lib/module/hooks/useFreehandGesture.js +169 -0
  144. package/lib/module/hooks/useFreehandGesture.js.map +1 -0
  145. package/lib/module/hooks/usePolygonGesture.js +106 -0
  146. package/lib/module/hooks/usePolygonGesture.js.map +1 -0
  147. package/lib/module/hooks/useSelectionGesture.js +232 -0
  148. package/lib/module/hooks/useSelectionGesture.js.map +1 -0
  149. package/lib/module/hooks/useShapeGesture.js +177 -0
  150. package/lib/module/hooks/useShapeGesture.js.map +1 -0
  151. package/lib/module/hooks/useViewportGesture.js +234 -0
  152. package/lib/module/hooks/useViewportGesture.js.map +1 -0
  153. package/lib/module/index.js +20 -0
  154. package/lib/module/index.js.map +1 -0
  155. package/lib/module/package.json +1 -0
  156. package/lib/module/renderers/ArrowRenderer.js +113 -0
  157. package/lib/module/renderers/ArrowRenderer.js.map +1 -0
  158. package/lib/module/renderers/CircleRenderer.js +46 -0
  159. package/lib/module/renderers/CircleRenderer.js.map +1 -0
  160. package/lib/module/renderers/FreehandRenderer.js +26 -0
  161. package/lib/module/renderers/FreehandRenderer.js.map +1 -0
  162. package/lib/module/renderers/InProgressRenderer.js +169 -0
  163. package/lib/module/renderers/InProgressRenderer.js.map +1 -0
  164. package/lib/module/renderers/LineRenderer.js +22 -0
  165. package/lib/module/renderers/LineRenderer.js.map +1 -0
  166. package/lib/module/renderers/MeasurementRenderer.js +129 -0
  167. package/lib/module/renderers/MeasurementRenderer.js.map +1 -0
  168. package/lib/module/renderers/ObjectRenderer.js +60 -0
  169. package/lib/module/renderers/ObjectRenderer.js.map +1 -0
  170. package/lib/module/renderers/PolygonRenderer.js +41 -0
  171. package/lib/module/renderers/PolygonRenderer.js.map +1 -0
  172. package/lib/module/renderers/RectRenderer.js +46 -0
  173. package/lib/module/renderers/RectRenderer.js.map +1 -0
  174. package/lib/module/renderers/SelectedObjectRenderer.js +587 -0
  175. package/lib/module/renderers/SelectedObjectRenderer.js.map +1 -0
  176. package/lib/module/renderers/SelectionOverlay.js +116 -0
  177. package/lib/module/renderers/SelectionOverlay.js.map +1 -0
  178. package/lib/module/store/useDrawingStore.js +350 -0
  179. package/lib/module/store/useDrawingStore.js.map +1 -0
  180. package/lib/module/types.js +4 -0
  181. package/lib/module/types.js.map +1 -0
  182. package/lib/module/utils/colors.js +40 -0
  183. package/lib/module/utils/colors.js.map +1 -0
  184. package/lib/module/utils/coordinates.js +71 -0
  185. package/lib/module/utils/coordinates.js.map +1 -0
  186. package/lib/module/utils/hitTesting.js +171 -0
  187. package/lib/module/utils/hitTesting.js.map +1 -0
  188. package/lib/module/utils/serialization.js +36 -0
  189. package/lib/module/utils/serialization.js.map +1 -0
  190. package/lib/module/utils/shapeDetection.js +147 -0
  191. package/lib/module/utils/shapeDetection.js.map +1 -0
  192. package/lib/module/utils/smoothing.js +80 -0
  193. package/lib/module/utils/smoothing.js.map +1 -0
  194. package/lib/typescript/DrawingEditor.d.ts +3 -0
  195. package/lib/typescript/DrawingEditor.d.ts.map +1 -0
  196. package/lib/typescript/components/ColorPalette.d.ts +9 -0
  197. package/lib/typescript/components/ColorPalette.d.ts.map +1 -0
  198. package/lib/typescript/components/LineWidthSlider.d.ts +11 -0
  199. package/lib/typescript/components/LineWidthSlider.d.ts.map +1 -0
  200. package/lib/typescript/components/MeasurementEditModal.d.ts +11 -0
  201. package/lib/typescript/components/MeasurementEditModal.d.ts.map +1 -0
  202. package/lib/typescript/components/MiniMap.d.ts +23 -0
  203. package/lib/typescript/components/MiniMap.d.ts.map +1 -0
  204. package/lib/typescript/components/TextAnnotation.d.ts +22 -0
  205. package/lib/typescript/components/TextAnnotation.d.ts.map +1 -0
  206. package/lib/typescript/components/TextEditModal.d.ts +10 -0
  207. package/lib/typescript/components/TextEditModal.d.ts.map +1 -0
  208. package/lib/typescript/components/Toolbar.d.ts +13 -0
  209. package/lib/typescript/components/Toolbar.d.ts.map +1 -0
  210. package/lib/typescript/components/ZoomBadge.d.ts +9 -0
  211. package/lib/typescript/components/ZoomBadge.d.ts.map +1 -0
  212. package/lib/typescript/hooks/useFreehandGesture.d.ts +47 -0
  213. package/lib/typescript/hooks/useFreehandGesture.d.ts.map +1 -0
  214. package/lib/typescript/hooks/usePolygonGesture.d.ts +47 -0
  215. package/lib/typescript/hooks/usePolygonGesture.d.ts.map +1 -0
  216. package/lib/typescript/hooks/useSelectionGesture.d.ts +32 -0
  217. package/lib/typescript/hooks/useSelectionGesture.d.ts.map +1 -0
  218. package/lib/typescript/hooks/useShapeGesture.d.ts +54 -0
  219. package/lib/typescript/hooks/useShapeGesture.d.ts.map +1 -0
  220. package/lib/typescript/hooks/useViewportGesture.d.ts +37 -0
  221. package/lib/typescript/hooks/useViewportGesture.d.ts.map +1 -0
  222. package/lib/typescript/index.d.ts +11 -0
  223. package/lib/typescript/index.d.ts.map +1 -0
  224. package/lib/typescript/renderers/ArrowRenderer.d.ts +9 -0
  225. package/lib/typescript/renderers/ArrowRenderer.d.ts.map +1 -0
  226. package/lib/typescript/renderers/CircleRenderer.d.ts +9 -0
  227. package/lib/typescript/renderers/CircleRenderer.d.ts.map +1 -0
  228. package/lib/typescript/renderers/FreehandRenderer.d.ts +9 -0
  229. package/lib/typescript/renderers/FreehandRenderer.d.ts.map +1 -0
  230. package/lib/typescript/renderers/InProgressRenderer.d.ts +32 -0
  231. package/lib/typescript/renderers/InProgressRenderer.d.ts.map +1 -0
  232. package/lib/typescript/renderers/LineRenderer.d.ts +9 -0
  233. package/lib/typescript/renderers/LineRenderer.d.ts.map +1 -0
  234. package/lib/typescript/renderers/MeasurementRenderer.d.ts +9 -0
  235. package/lib/typescript/renderers/MeasurementRenderer.d.ts.map +1 -0
  236. package/lib/typescript/renderers/ObjectRenderer.d.ts +12 -0
  237. package/lib/typescript/renderers/ObjectRenderer.d.ts.map +1 -0
  238. package/lib/typescript/renderers/PolygonRenderer.d.ts +13 -0
  239. package/lib/typescript/renderers/PolygonRenderer.d.ts.map +1 -0
  240. package/lib/typescript/renderers/RectRenderer.d.ts +9 -0
  241. package/lib/typescript/renderers/RectRenderer.d.ts.map +1 -0
  242. package/lib/typescript/renderers/SelectedObjectRenderer.d.ts +18 -0
  243. package/lib/typescript/renderers/SelectedObjectRenderer.d.ts.map +1 -0
  244. package/lib/typescript/renderers/SelectionOverlay.d.ts +21 -0
  245. package/lib/typescript/renderers/SelectionOverlay.d.ts.map +1 -0
  246. package/lib/typescript/store/useDrawingStore.d.ts +30 -0
  247. package/lib/typescript/store/useDrawingStore.d.ts.map +1 -0
  248. package/lib/typescript/types.d.ts +130 -0
  249. package/lib/typescript/types.d.ts.map +1 -0
  250. package/lib/typescript/utils/colors.d.ts +11 -0
  251. package/lib/typescript/utils/colors.d.ts.map +1 -0
  252. package/lib/typescript/utils/coordinates.d.ts +34 -0
  253. package/lib/typescript/utils/coordinates.d.ts.map +1 -0
  254. package/lib/typescript/utils/hitTesting.d.ts +18 -0
  255. package/lib/typescript/utils/hitTesting.d.ts.map +1 -0
  256. package/lib/typescript/utils/serialization.d.ts +17 -0
  257. package/lib/typescript/utils/serialization.d.ts.map +1 -0
  258. package/lib/typescript/utils/shapeDetection.d.ts +22 -0
  259. package/lib/typescript/utils/shapeDetection.d.ts.map +1 -0
  260. package/lib/typescript/utils/smoothing.d.ts +16 -0
  261. package/lib/typescript/utils/smoothing.d.ts.map +1 -0
  262. package/package.json +108 -0
  263. package/src/DrawingEditor.tsx +1071 -0
  264. package/src/assets/toolbar-icons/arrow-disabled.png +0 -0
  265. package/src/assets/toolbar-icons/arrow-enabled.png +0 -0
  266. package/src/assets/toolbar-icons/arrow.png +0 -0
  267. package/src/assets/toolbar-icons/circle-disabled.png +0 -0
  268. package/src/assets/toolbar-icons/circle-enabled.png +0 -0
  269. package/src/assets/toolbar-icons/circle.png +0 -0
  270. package/src/assets/toolbar-icons/freehand-disabled.png +0 -0
  271. package/src/assets/toolbar-icons/freehand-enabled.png +0 -0
  272. package/src/assets/toolbar-icons/freehand.png +0 -0
  273. package/src/assets/toolbar-icons/line-disabled.png +0 -0
  274. package/src/assets/toolbar-icons/line-enabled.png +0 -0
  275. package/src/assets/toolbar-icons/line.png +0 -0
  276. package/src/assets/toolbar-icons/measure-disabled.png +0 -0
  277. package/src/assets/toolbar-icons/measure-enabled.png +0 -0
  278. package/src/assets/toolbar-icons/measure.png +0 -0
  279. package/src/assets/toolbar-icons/move-disabled.png +0 -0
  280. package/src/assets/toolbar-icons/move-enabled.png +0 -0
  281. package/src/assets/toolbar-icons/move.png +0 -0
  282. package/src/assets/toolbar-icons/polygon-disabled.png +0 -0
  283. package/src/assets/toolbar-icons/polygon-enabled.png +0 -0
  284. package/src/assets/toolbar-icons/polygon.png +0 -0
  285. package/src/assets/toolbar-icons/rectangle-disabled.png +0 -0
  286. package/src/assets/toolbar-icons/rectangle-enabled.png +0 -0
  287. package/src/assets/toolbar-icons/rectangle.png +0 -0
  288. package/src/assets/toolbar-icons/text-disabled.png +0 -0
  289. package/src/assets/toolbar-icons/text-enabled.png +0 -0
  290. package/src/assets/toolbar-icons/text.png +0 -0
  291. package/src/components/ColorPalette.tsx +497 -0
  292. package/src/components/LineWidthSlider.tsx +87 -0
  293. package/src/components/MeasurementEditModal.tsx +163 -0
  294. package/src/components/MiniMap.tsx +275 -0
  295. package/src/components/TextAnnotation.tsx +198 -0
  296. package/src/components/TextEditModal.tsx +139 -0
  297. package/src/components/Toolbar.tsx +254 -0
  298. package/src/components/ZoomBadge.tsx +166 -0
  299. package/src/hooks/useFreehandGesture.ts +249 -0
  300. package/src/hooks/usePolygonGesture.ts +162 -0
  301. package/src/hooks/useSelectionGesture.ts +293 -0
  302. package/src/hooks/useShapeGesture.ts +256 -0
  303. package/src/hooks/useViewportGesture.ts +337 -0
  304. package/src/index.tsx +51 -0
  305. package/src/renderers/ArrowRenderer.tsx +123 -0
  306. package/src/renderers/CircleRenderer.tsx +60 -0
  307. package/src/renderers/FreehandRenderer.tsx +33 -0
  308. package/src/renderers/InProgressRenderer.tsx +217 -0
  309. package/src/renderers/LineRenderer.tsx +34 -0
  310. package/src/renderers/MeasurementRenderer.tsx +179 -0
  311. package/src/renderers/ObjectRenderer.tsx +42 -0
  312. package/src/renderers/PolygonRenderer.tsx +66 -0
  313. package/src/renderers/RectRenderer.tsx +60 -0
  314. package/src/renderers/SelectedObjectRenderer.tsx +738 -0
  315. package/src/renderers/SelectionOverlay.tsx +170 -0
  316. package/src/store/useDrawingStore.ts +357 -0
  317. package/src/types.ts +186 -0
  318. package/src/utils/colors.ts +98 -0
  319. package/src/utils/coordinates.ts +75 -0
  320. package/src/utils/hitTesting.ts +242 -0
  321. package/src/utils/serialization.ts +45 -0
  322. package/src/utils/shapeDetection.ts +160 -0
  323. package/src/utils/smoothing.ts +84 -0
@@ -0,0 +1,497 @@
1
+ import React, { useCallback, useMemo, useState } from "react";
2
+ import {
3
+ View,
4
+ TouchableOpacity,
5
+ StyleSheet,
6
+ Modal,
7
+ Pressable,
8
+ ScrollView,
9
+ Platform,
10
+ useWindowDimensions,
11
+ } from "react-native";
12
+ import { PRESET_COLORS, EXTENDED_COLOR_GROUPS } from "../utils/colors";
13
+
14
+ interface ColorPaletteProps {
15
+ strokeColor: string;
16
+ onSelectColor: (color: string) => void;
17
+ direction?: "horizontal" | "vertical";
18
+ }
19
+
20
+ export const ColorPalette = React.memo(function ColorPalette({
21
+ strokeColor,
22
+ onSelectColor,
23
+ direction = "vertical",
24
+ }: ColorPaletteProps) {
25
+ const [modalVisible, setModalVisible] = useState(false);
26
+
27
+ const isExtendedColor =
28
+ !PRESET_COLORS.includes(strokeColor as (typeof PRESET_COLORS)[number]);
29
+
30
+ const handleOpenModal = useCallback(() => {
31
+ setModalVisible(true);
32
+ }, []);
33
+
34
+ const handleSelectExtended = useCallback(
35
+ (color: string) => {
36
+ onSelectColor(color);
37
+ setModalVisible(false);
38
+ },
39
+ [onSelectColor]
40
+ );
41
+
42
+ const handleCloseModal = useCallback(() => {
43
+ setModalVisible(false);
44
+ }, []);
45
+
46
+ return (
47
+ <>
48
+ <View
49
+ style={[
50
+ styles.container,
51
+ direction === "horizontal"
52
+ ? styles.containerHorizontal
53
+ : styles.containerVertical,
54
+ ]}
55
+ >
56
+ {/* Multi-color button */}
57
+ <MultiColorButton
58
+ isSelected={isExtendedColor}
59
+ selectedColor={isExtendedColor ? strokeColor : undefined}
60
+ onPress={handleOpenModal}
61
+ />
62
+
63
+ {PRESET_COLORS.map((color, index) => (
64
+ <ColorButton
65
+ key={index}
66
+ color={color}
67
+ isSelected={strokeColor === color}
68
+ onPress={onSelectColor}
69
+ />
70
+ ))}
71
+ </View>
72
+
73
+ <ColorPickerModal
74
+ visible={modalVisible}
75
+ selectedColor={strokeColor}
76
+ onSelectColor={handleSelectExtended}
77
+ onClose={handleCloseModal}
78
+ />
79
+ </>
80
+ );
81
+ });
82
+
83
+ /* ─── Multi-color rainbow button ──────────────────────────────────────────── */
84
+
85
+ interface MultiColorButtonProps {
86
+ isSelected: boolean;
87
+ selectedColor?: string;
88
+ onPress: () => void;
89
+ }
90
+
91
+ const RAINBOW_SEGMENTS = [
92
+ "#FF4F3D",
93
+ "#FFa345",
94
+ "#FACE15",
95
+ "#42E07D",
96
+ "#5E70FF",
97
+ "#A855F7",
98
+ ];
99
+
100
+ const MultiColorButton = React.memo(function MultiColorButton({
101
+ isSelected,
102
+ selectedColor,
103
+ onPress,
104
+ }: MultiColorButtonProps) {
105
+ return (
106
+ <TouchableOpacity
107
+ onPress={onPress}
108
+ activeOpacity={0.7}
109
+ style={[styles.buttonOuter, isSelected && styles.buttonOuterSelected]}
110
+ >
111
+ {isSelected && selectedColor ? (
112
+ <View style={[styles.colorCircle, { backgroundColor: selectedColor }]}>
113
+ <View style={styles.rainbowDot} />
114
+ </View>
115
+ ) : (
116
+ <View style={[styles.colorCircle, styles.rainbowCircle]}>
117
+ {RAINBOW_SEGMENTS.map((c, i) => (
118
+ <View
119
+ key={i}
120
+ style={[
121
+ styles.rainbowSegment,
122
+ {
123
+ backgroundColor: c,
124
+ transform: [{ rotate: `${i * 60}deg` }],
125
+ },
126
+ ]}
127
+ />
128
+ ))}
129
+ <View style={styles.rainbowCenter} />
130
+ </View>
131
+ )}
132
+ </TouchableOpacity>
133
+ );
134
+ });
135
+
136
+ /* ─── Single color button ─────────────────────────────────────────────────── */
137
+
138
+ interface ColorButtonProps {
139
+ color: string;
140
+ isSelected: boolean;
141
+ onPress: (color: string) => void;
142
+ }
143
+
144
+ const ColorButton = React.memo(function ColorButton({
145
+ color,
146
+ isSelected,
147
+ onPress,
148
+ }: ColorButtonProps) {
149
+ const handlePress = useCallback(() => {
150
+ onPress(color);
151
+ }, [onPress, color]);
152
+
153
+ return (
154
+ <TouchableOpacity
155
+ onPress={handlePress}
156
+ activeOpacity={0.7}
157
+ style={[styles.buttonOuter, isSelected && styles.buttonOuterSelected]}
158
+ >
159
+ <View style={[styles.colorCircle, { backgroundColor: color }]} />
160
+ </TouchableOpacity>
161
+ );
162
+ });
163
+
164
+ /* ─── Color picker modal ──────────────────────────────────────────────────── */
165
+
166
+ interface ColorPickerModalProps {
167
+ visible: boolean;
168
+ selectedColor: string;
169
+ onSelectColor: (color: string) => void;
170
+ onClose: () => void;
171
+ }
172
+
173
+ const ColorPickerModal = React.memo(function ColorPickerModal({
174
+ visible,
175
+ selectedColor,
176
+ onSelectColor,
177
+ onClose,
178
+ }: ColorPickerModalProps) {
179
+ const { width, height } = useWindowDimensions();
180
+ const shortestSide = Math.min(width, height);
181
+ const isTablet = shortestSide >= 768;
182
+ const horizontalPadding = isTablet ? 32 : 20;
183
+ const cardWidth = Math.min(
184
+ width - horizontalPadding * 2,
185
+ isTablet ? 760 : 380
186
+ );
187
+ const columns = Math.max(
188
+ 4,
189
+ Math.min(isTablet ? 8 : 6, Math.floor(cardWidth / (isTablet ? 84 : 68)))
190
+ );
191
+ const swatchOuterSize = isTablet ? 76 : 60;
192
+ const swatchInnerSize = isTablet ? 58 : 46;
193
+ const cardRadius = isTablet ? 28 : 24;
194
+ const phoneBottomPadding = Platform.OS === "ios" ? 12 : 16;
195
+ const paletteRows = useMemo(
196
+ () =>
197
+ EXTENDED_COLOR_GROUPS.flatMap((group) =>
198
+ Array.from({ length: Math.ceil(group.length / columns) }, (_item, index) =>
199
+ group.slice(index * columns, index * columns + columns)
200
+ )
201
+ ),
202
+ [columns]
203
+ );
204
+
205
+ const renderSwatch = useCallback(
206
+ (item: string) => (
207
+ <TouchableOpacity
208
+ key={item}
209
+ onPress={() => onSelectColor(item)}
210
+ activeOpacity={0.7}
211
+ accessibilityRole="button"
212
+ accessibilityLabel={`Select color ${item}`}
213
+ style={[
214
+ styles.gridItem,
215
+ {
216
+ width: swatchOuterSize,
217
+ height: swatchOuterSize,
218
+ borderRadius: swatchOuterSize / 2,
219
+ },
220
+ selectedColor === item && styles.gridItemSelected,
221
+ ]}
222
+ >
223
+ <View
224
+ style={[
225
+ styles.gridCircle,
226
+ {
227
+ width: swatchInnerSize,
228
+ height: swatchInnerSize,
229
+ borderRadius: swatchInnerSize / 2,
230
+ backgroundColor: item,
231
+ },
232
+ ]}
233
+ >
234
+ {selectedColor === item && (
235
+ <View
236
+ style={[
237
+ styles.checkmark,
238
+ {
239
+ width: swatchInnerSize * 0.34,
240
+ height: swatchInnerSize * 0.34,
241
+ },
242
+ ]}
243
+ >
244
+ <View
245
+ style={[
246
+ styles.checkmarkShort,
247
+ {
248
+ width: Math.max(2, Math.round(swatchInnerSize * 0.07)),
249
+ height: Math.max(6, Math.round(swatchInnerSize * 0.18)),
250
+ left: Math.round(swatchInnerSize * 0.04),
251
+ top: Math.round(swatchInnerSize * 0.16),
252
+ },
253
+ ]}
254
+ />
255
+ <View
256
+ style={[
257
+ styles.checkmarkLong,
258
+ {
259
+ width: Math.max(2, Math.round(swatchInnerSize * 0.07)),
260
+ height: Math.max(10, Math.round(swatchInnerSize * 0.28)),
261
+ left: Math.round(swatchInnerSize * 0.16),
262
+ top: Math.round(swatchInnerSize * 0.02),
263
+ },
264
+ ]}
265
+ />
266
+ </View>
267
+ )}
268
+ </View>
269
+ </TouchableOpacity>
270
+ ),
271
+ [onSelectColor, selectedColor, swatchInnerSize, swatchOuterSize]
272
+ );
273
+
274
+ return (
275
+ <Modal
276
+ visible={visible}
277
+ transparent
278
+ animationType="fade"
279
+ onRequestClose={onClose}
280
+ >
281
+ <Pressable style={styles.modalOverlay} onPress={onClose}>
282
+ {isTablet ? (
283
+ <Pressable
284
+ style={[
285
+ styles.modalCard,
286
+ styles.modalCardTablet,
287
+ {
288
+ width: cardWidth,
289
+ maxHeight: height * 0.84,
290
+ borderRadius: cardRadius,
291
+ padding: 28,
292
+ },
293
+ ]}
294
+ onPress={() => {}}
295
+ >
296
+ <View style={styles.modalHandle} />
297
+ <ScrollView
298
+ contentContainerStyle={styles.gridContainer}
299
+ showsVerticalScrollIndicator={false}
300
+ >
301
+ {paletteRows.map((row, index) => (
302
+ <View key={`row-${index}`} style={styles.gridRow}>
303
+ {row.map(renderSwatch)}
304
+ </View>
305
+ ))}
306
+ </ScrollView>
307
+ </Pressable>
308
+ ) : (
309
+ <View
310
+ pointerEvents="box-none"
311
+ style={[
312
+ styles.modalPhoneSafeArea,
313
+ {
314
+ paddingHorizontal: horizontalPadding,
315
+ paddingBottom: phoneBottomPadding,
316
+ },
317
+ ]}
318
+ >
319
+ <Pressable
320
+ style={[
321
+ styles.modalCard,
322
+ styles.modalCardPhone,
323
+ {
324
+ width: "100%",
325
+ maxWidth: cardWidth,
326
+ maxHeight: height * 0.74,
327
+ borderRadius: cardRadius,
328
+ padding: 18,
329
+ },
330
+ ]}
331
+ onPress={() => {}}
332
+ >
333
+ <View style={styles.modalHandle} />
334
+ <ScrollView
335
+ contentContainerStyle={styles.gridContainer}
336
+ showsVerticalScrollIndicator={false}
337
+ >
338
+ {paletteRows.map((row, index) => (
339
+ <View key={`row-${index}`} style={styles.gridRow}>
340
+ {row.map(renderSwatch)}
341
+ </View>
342
+ ))}
343
+ </ScrollView>
344
+ </Pressable>
345
+ </View>
346
+ )}
347
+ </Pressable>
348
+ </Modal>
349
+ );
350
+ });
351
+
352
+ /* ─── Styles ──────────────────────────────────────────────────────────────── */
353
+
354
+ const styles = StyleSheet.create({
355
+ container: {
356
+ padding: 8,
357
+ gap: 6,
358
+ },
359
+ containerVertical: {
360
+ flexDirection: "column",
361
+ alignItems: "center",
362
+ },
363
+ containerHorizontal: {
364
+ flexDirection: "row",
365
+ alignItems: "center",
366
+ },
367
+ buttonOuter: {
368
+ width: 32,
369
+ height: 32,
370
+ borderRadius: 16,
371
+ justifyContent: "center",
372
+ alignItems: "center",
373
+ borderWidth: 2,
374
+ borderColor: "transparent",
375
+ },
376
+ buttonOuterSelected: {
377
+ borderColor: "#FFFFFF",
378
+ },
379
+ colorCircle: {
380
+ width: 24,
381
+ height: 24,
382
+ borderRadius: 12,
383
+ borderWidth: 1,
384
+ borderColor: "rgba(255, 255, 255, 0.3)",
385
+ overflow: "hidden",
386
+ },
387
+
388
+ /* Rainbow button */
389
+ rainbowCircle: {
390
+ justifyContent: "center",
391
+ alignItems: "center",
392
+ },
393
+ rainbowSegment: {
394
+ position: "absolute",
395
+ width: 12,
396
+ height: 12,
397
+ top: 0,
398
+ left: 6,
399
+ transformOrigin: "0% 100%",
400
+ },
401
+ rainbowCenter: {
402
+ width: 8,
403
+ height: 8,
404
+ borderRadius: 4,
405
+ backgroundColor: "#222",
406
+ zIndex: 1,
407
+ },
408
+ rainbowDot: {
409
+ position: "absolute",
410
+ bottom: 1,
411
+ right: 1,
412
+ width: 8,
413
+ height: 8,
414
+ borderRadius: 4,
415
+ borderWidth: 1,
416
+ borderColor: "rgba(0,0,0,0.3)",
417
+ backgroundColor: "transparent",
418
+ overflow: "hidden",
419
+ },
420
+
421
+ /* Modal */
422
+ modalOverlay: {
423
+ flex: 1,
424
+ backgroundColor: "rgba(0, 0, 0, 0.6)",
425
+ justifyContent: "center",
426
+ alignItems: "center",
427
+ },
428
+ modalCard: {
429
+ backgroundColor: "#1C1C1E",
430
+ shadowColor: "#000",
431
+ shadowOffset: { width: 0, height: 4 },
432
+ shadowOpacity: 0.5,
433
+ shadowRadius: 12,
434
+ elevation: 12,
435
+ },
436
+ modalCardTablet: {
437
+ alignSelf: "center",
438
+ },
439
+ modalPhoneSafeArea: {
440
+ flex: 1,
441
+ width: "100%",
442
+ justifyContent: "flex-end",
443
+ alignItems: "center",
444
+ paddingTop: 24,
445
+ },
446
+ modalCardPhone: {
447
+ alignSelf: "center",
448
+ },
449
+ modalHandle: {
450
+ alignSelf: "center",
451
+ width: 44,
452
+ height: 5,
453
+ borderRadius: 999,
454
+ marginBottom: 16,
455
+ backgroundColor: "rgba(255, 255, 255, 0.14)",
456
+ },
457
+ gridContainer: {
458
+ gap: 10,
459
+ },
460
+ gridRow: {
461
+ flexDirection: "row",
462
+ justifyContent: "center",
463
+ alignItems: "center",
464
+ gap: 10,
465
+ },
466
+ gridItem: {
467
+ justifyContent: "center",
468
+ alignItems: "center",
469
+ borderWidth: 3,
470
+ borderColor: "transparent",
471
+ },
472
+ gridItemSelected: {
473
+ borderColor: "#FFFFFF",
474
+ },
475
+ gridCircle: {
476
+ borderWidth: 1,
477
+ borderColor: "rgba(255, 255, 255, 0.2)",
478
+ justifyContent: "center",
479
+ alignItems: "center",
480
+ },
481
+ checkmark: {
482
+ justifyContent: "center",
483
+ alignItems: "center",
484
+ },
485
+ checkmarkShort: {
486
+ position: "absolute",
487
+ backgroundColor: "#FFF",
488
+ borderRadius: 1,
489
+ transform: [{ rotate: "-45deg" }],
490
+ },
491
+ checkmarkLong: {
492
+ position: "absolute",
493
+ backgroundColor: "#FFF",
494
+ borderRadius: 1,
495
+ transform: [{ rotate: "35deg" }],
496
+ },
497
+ });
@@ -0,0 +1,87 @@
1
+ import React, { useCallback } from "react";
2
+ import { View, StyleSheet } from "react-native";
3
+ import Slider from "@react-native-community/slider";
4
+
5
+ interface LineWidthSliderProps {
6
+ value: number;
7
+ onValueChange: (value: number) => void;
8
+ minimumValue?: number;
9
+ maximumValue?: number;
10
+ orientation?: "horizontal" | "vertical";
11
+ }
12
+
13
+ export const LineWidthSlider = React.memo(function LineWidthSlider({
14
+ value,
15
+ onValueChange,
16
+ minimumValue = 1,
17
+ maximumValue = 20,
18
+ orientation = "vertical",
19
+ }: LineWidthSliderProps) {
20
+ const handleChange = useCallback(
21
+ (v: number) => {
22
+ onValueChange(Math.round(v));
23
+ },
24
+ [onValueChange],
25
+ );
26
+
27
+ return (
28
+ <View
29
+ style={[
30
+ styles.container,
31
+ orientation === "vertical" && styles.containerVertical,
32
+ ]}
33
+ >
34
+ <View
35
+ style={
36
+ orientation === "vertical"
37
+ ? styles.sliderWrapperVertical
38
+ : styles.sliderWrapperHorizontal
39
+ }
40
+ >
41
+ <Slider
42
+ style={
43
+ orientation === "vertical"
44
+ ? styles.sliderVertical
45
+ : styles.sliderHorizontal
46
+ }
47
+ value={value}
48
+ onValueChange={handleChange}
49
+ minimumValue={minimumValue}
50
+ maximumValue={maximumValue}
51
+ minimumTrackTintColor="#FFFFFF"
52
+ maximumTrackTintColor="rgba(255, 255, 255, 0.3)"
53
+ thumbTintColor="#FFFFFF"
54
+ />
55
+ </View>
56
+ </View>
57
+ );
58
+ });
59
+
60
+ const styles = StyleSheet.create({
61
+ container: {
62
+ padding: 8,
63
+ },
64
+ containerVertical: {
65
+ justifyContent: "center",
66
+ alignItems: "center",
67
+ },
68
+ sliderWrapperVertical: {
69
+ width: 40,
70
+ height: 200,
71
+ transform: [{ rotate: "-90deg" }],
72
+ justifyContent: "center",
73
+ },
74
+ sliderWrapperHorizontal: {
75
+ width: 150,
76
+ height: 40,
77
+ justifyContent: "center",
78
+ },
79
+ sliderVertical: {
80
+ width: 200,
81
+ height: 40,
82
+ },
83
+ sliderHorizontal: {
84
+ width: 150,
85
+ height: 40,
86
+ },
87
+ });