@page-speed/maps 0.1.3 → 0.1.5

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 (54) hide show
  1. package/README.md +120 -0
  2. package/dist/components/geo-map.cjs +1228 -0
  3. package/dist/components/geo-map.cjs.map +1 -0
  4. package/dist/components/geo-map.d.cts +138 -0
  5. package/dist/components/geo-map.d.ts +138 -0
  6. package/dist/components/geo-map.js +1207 -0
  7. package/dist/components/geo-map.js.map +1 -0
  8. package/dist/components/index.cjs +1350 -0
  9. package/dist/components/index.cjs.map +1 -0
  10. package/dist/components/index.d.cts +5 -0
  11. package/dist/components/index.d.ts +5 -0
  12. package/dist/components/index.js +1326 -0
  13. package/dist/components/index.js.map +1 -0
  14. package/dist/components/map-marker.cjs +137 -0
  15. package/dist/components/map-marker.cjs.map +1 -0
  16. package/dist/components/map-marker.d.cts +76 -0
  17. package/dist/components/map-marker.d.ts +76 -0
  18. package/dist/components/map-marker.js +130 -0
  19. package/dist/components/map-marker.js.map +1 -0
  20. package/dist/core/MapLibre.cjs +46 -20
  21. package/dist/core/MapLibre.cjs.map +1 -1
  22. package/dist/core/MapLibre.js +46 -20
  23. package/dist/core/MapLibre.js.map +1 -1
  24. package/dist/core/index.cjs +46 -20
  25. package/dist/core/index.cjs.map +1 -1
  26. package/dist/core/index.js +46 -20
  27. package/dist/core/index.js.map +1 -1
  28. package/dist/index.cjs +964 -39
  29. package/dist/index.cjs.map +1 -1
  30. package/dist/index.d.cts +2 -0
  31. package/dist/index.d.ts +2 -0
  32. package/dist/index.js +945 -39
  33. package/dist/index.js.map +1 -1
  34. package/dist/types/index.d.cts +5 -5
  35. package/dist/types/index.d.ts +5 -5
  36. package/dist/utils/cn.cjs +13 -0
  37. package/dist/utils/cn.cjs.map +1 -0
  38. package/dist/utils/cn.d.cts +16 -0
  39. package/dist/utils/cn.d.ts +16 -0
  40. package/dist/utils/cn.js +11 -0
  41. package/dist/utils/cn.js.map +1 -0
  42. package/dist/utils/index.cjs +63 -0
  43. package/dist/utils/index.cjs.map +1 -1
  44. package/dist/utils/index.d.cts +4 -0
  45. package/dist/utils/index.d.ts +4 -0
  46. package/dist/utils/index.js +42 -1
  47. package/dist/utils/index.js.map +1 -1
  48. package/dist/utils/simple-pressable.cjs +63 -0
  49. package/dist/utils/simple-pressable.cjs.map +1 -0
  50. package/dist/utils/simple-pressable.d.cts +20 -0
  51. package/dist/utils/simple-pressable.d.ts +20 -0
  52. package/dist/utils/simple-pressable.js +41 -0
  53. package/dist/utils/simple-pressable.js.map +1 -0
  54. package/package.json +29 -2
@@ -1,2 +1,6 @@
1
1
  export { DEFAULT_STADIA_STYLE_URL, MAPLIBRE_DEFAULT_STYLE_URL, MapLibreBuiltInStyle, appendStadiaApiKey, getMapLibreStyleUrl } from './getMapLibreStyleUrl.cjs';
2
2
  export { generateGoogleDirectionsLink, generateGoogleMapLink } from './googleMapLinks.cjs';
3
+ export { cn } from './cn.cjs';
4
+ export { SimplePressable, SimplePressableProps } from './simple-pressable.cjs';
5
+ import 'clsx';
6
+ import 'react';
@@ -1,2 +1,6 @@
1
1
  export { DEFAULT_STADIA_STYLE_URL, MAPLIBRE_DEFAULT_STYLE_URL, MapLibreBuiltInStyle, appendStadiaApiKey, getMapLibreStyleUrl } from './getMapLibreStyleUrl.js';
2
2
  export { generateGoogleDirectionsLink, generateGoogleMapLink } from './googleMapLinks.js';
3
+ export { cn } from './cn.js';
4
+ export { SimplePressable, SimplePressableProps } from './simple-pressable.js';
5
+ import 'clsx';
6
+ import 'react';
@@ -1,3 +1,8 @@
1
+ import { clsx } from 'clsx';
2
+ import { twMerge } from 'tailwind-merge';
3
+ import * as React from 'react';
4
+ import { jsx } from 'react/jsx-runtime';
5
+
1
6
  // src/utils/getMapLibreStyleUrl.ts
2
7
  var MAPLIBRE_DEFAULT_STYLE_URL = "https://basemaps.cartocdn.com/gl/positron-gl-style/style.json";
3
8
  var DEFAULT_STADIA_STYLE_URL = "https://tiles.stadiamaps.com/styles/osm_bright.json";
@@ -73,7 +78,43 @@ function generateGoogleMapLink(latitude, longitude, zoom = 15) {
73
78
  function generateGoogleDirectionsLink(latitude, longitude) {
74
79
  return `https://www.google.com/maps/dir/?api=1&destination=${latitude},${longitude}`;
75
80
  }
81
+ function cn(...inputs) {
82
+ return twMerge(clsx(inputs));
83
+ }
84
+ var SimplePressable = React.forwardRef(({ children, className, href, onClick, ...props }, ref) => {
85
+ if (href) {
86
+ const isExternal = href.startsWith("http://") || href.startsWith("https://");
87
+ return /* @__PURE__ */ jsx(
88
+ "a",
89
+ {
90
+ ref,
91
+ href,
92
+ className,
93
+ target: isExternal ? "_blank" : props.target,
94
+ rel: isExternal ? "noopener noreferrer" : props.rel,
95
+ onClick,
96
+ ...props,
97
+ children
98
+ }
99
+ );
100
+ }
101
+ if (onClick) {
102
+ return /* @__PURE__ */ jsx(
103
+ "button",
104
+ {
105
+ ref,
106
+ type: "button",
107
+ className,
108
+ onClick,
109
+ ...props,
110
+ children
111
+ }
112
+ );
113
+ }
114
+ return /* @__PURE__ */ jsx("span", { className, children });
115
+ });
116
+ SimplePressable.displayName = "SimplePressable";
76
117
 
77
- export { DEFAULT_STADIA_STYLE_URL, MAPLIBRE_DEFAULT_STYLE_URL, appendStadiaApiKey, generateGoogleDirectionsLink, generateGoogleMapLink, getMapLibreStyleUrl };
118
+ export { DEFAULT_STADIA_STYLE_URL, MAPLIBRE_DEFAULT_STYLE_URL, SimplePressable, appendStadiaApiKey, cn, generateGoogleDirectionsLink, generateGoogleMapLink, getMapLibreStyleUrl };
78
119
  //# sourceMappingURL=index.js.map
79
120
  //# sourceMappingURL=index.js.map
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/utils/getMapLibreStyleUrl.ts","../../src/utils/googleMapLinks.ts"],"names":[],"mappings":";AAAA,IAAM,0BAAA,GACJ;AACF,IAAM,wBAAA,GACJ;AAEF,IAAM,SAAA,GAAoC;AAAA,EACxC,OAAA,EAAS,wBAAA;AAAA,EACT,gBAAA,EAAkB,yDAAA;AAAA,EAClB,qBAAA,EAAuB,8DAAA;AAAA,EACvB,kBAAA,EAAoB,0BAAA;AAAA,EACpB,YAAA,EAAc,qDAAA;AAAA,EACd,iBAAA,EAAmB,mDAAA;AAAA,EACnB,cAAA,EAAgB,uDAAA;AAAA,EAChB,gBAAA,EAAkB,yDAAA;AAAA,EAClB,mBAAA,EAAqB;AACvB,CAAA;AAEA,IAAM,cAAA,GAAiB,eAAA;AAIvB,SAAS,gBAAgB,GAAA,EAAsB;AAC7C,EAAA,IAAI;AACF,IAAA,MAAM,MAAA,GAAS,IAAI,GAAA,CAAI,GAAG,CAAA;AAC1B,IAAA,OAAO,OAAO,QAAA,KAAa,sBAAA;AAAA,EAC7B,CAAA,CAAA,MAAQ;AACN,IAAA,OAAO,KAAA;AAAA,EACT;AACF;AAEA,SAAS,mBAAmB,YAAA,EAA4B;AACtD,EAAA,IAAI,CAAC,YAAA,CAAa,IAAA,EAAK,EAAG;AACxB,IAAA,MAAM,IAAI,KAAA;AAAA,MACR;AAAA,KACF;AAAA,EACF;AACF;AAEO,SAAS,kBAAA,CACd,UACA,YAAA,EACQ;AACR,EAAA,IAAI,CAAC,eAAA,CAAgB,QAAQ,CAAA,EAAG;AAC9B,IAAA,OAAO,QAAA;AAAA,EACT;AAEA,EAAA,kBAAA,CAAmB,YAAY,CAAA;AAE/B,EAAA,MAAM,MAAA,GAAS,IAAI,GAAA,CAAI,QAAQ,CAAA;AAC/B,EAAA,IAAI,CAAC,MAAA,CAAO,YAAA,CAAa,GAAA,CAAI,SAAS,CAAA,EAAG;AACvC,IAAA,MAAA,CAAO,YAAA,CAAa,GAAA,CAAI,SAAA,EAAW,YAAY,CAAA;AAAA,EACjD;AAEA,EAAA,OAAO,OAAO,QAAA,EAAS;AACzB;AAEO,SAAS,mBAAA,CACd,OACA,YAAA,EACQ;AACR,EAAA,MAAM,gBAAA,GAAmB,aAAa,IAAA,EAAK;AAE3C,EAAA,IAAI,CAAC,KAAA,IAAS,OAAO,KAAA,KAAU,QAAA,EAAU;AACvC,IAAA,IAAI,CAAC,gBAAA,EAAkB;AACrB,MAAA,OAAO,0BAAA;AAAA,IACT;AACA,IAAA,OAAO,kBAAA,CAAmB,0BAA0B,gBAAgB,CAAA;AAAA,EACtE;AAEA,EAAA,IAAI,SAAA,CAAU,KAAK,CAAA,EAAG;AACpB,IAAA,MAAM,cAAA,GAAiB,UAAU,KAAK,CAAA;AACtC,IAAA,IAAI,eAAA,CAAgB,cAAc,CAAA,IAAK,CAAC,gBAAA,EAAkB;AACxD,MAAA,OAAO,0BAAA;AAAA,IACT;AACA,IAAA,OAAO,kBAAA,CAAmB,gBAAgB,gBAAgB,CAAA;AAAA,EAC5D;AAEA,EAAA,IAAI,cAAA,CAAe,IAAA,CAAK,KAAK,CAAA,EAAG;AAC9B,IAAA,IAAI,eAAA,CAAgB,KAAK,CAAA,IAAK,CAAC,gBAAA,EAAkB;AAC/C,MAAA,OAAO,0BAAA;AAAA,IACT;AACA,IAAA,OAAO,kBAAA,CAAmB,OAAO,gBAAgB,CAAA;AAAA,EACnD;AAEA,EAAA,IAAI,CAAC,gBAAA,EAAkB;AACrB,IAAA,OAAO,0BAAA;AAAA,EACT;AAEA,EAAA,OAAO,kBAAA,CAAmB,0BAA0B,gBAAgB,CAAA;AACtE;;;ACzFO,SAAS,qBAAA,CACd,QAAA,EACA,SAAA,EACA,IAAA,GAAO,EAAA,EACC;AACR,EAAA,OAAO,CAAA,6BAAA,EAAgC,QAAQ,CAAA,CAAA,EAAI,SAAS,IAAI,IAAI,CAAA,CAAA,CAAA;AACtE;AAEO,SAAS,4BAAA,CACd,UACA,SAAA,EACQ;AACR,EAAA,OAAO,CAAA,mDAAA,EAAsD,QAAQ,CAAA,CAAA,EAAI,SAAS,CAAA,CAAA;AACpF","file":"index.js","sourcesContent":["const MAPLIBRE_DEFAULT_STYLE_URL =\n \"https://basemaps.cartocdn.com/gl/positron-gl-style/style.json\";\nconst DEFAULT_STADIA_STYLE_URL =\n \"https://tiles.stadiamaps.com/styles/osm_bright.json\";\n\nconst STYLE_MAP: Record<string, string> = {\n default: DEFAULT_STADIA_STYLE_URL,\n \"alidade-smooth\": \"https://tiles.stadiamaps.com/styles/alidade_smooth.json\",\n \"alidade-smooth-dark\": \"https://tiles.stadiamaps.com/styles/alidade_smooth_dark.json\",\n \"maplibre-default\": MAPLIBRE_DEFAULT_STYLE_URL,\n \"osm-bright\": \"https://tiles.stadiamaps.com/styles/osm_bright.json\",\n \"stadia-outdoors\": \"https://tiles.stadiamaps.com/styles/outdoors.json\",\n \"stamen-toner\": \"https://tiles.stadiamaps.com/styles/stamen_toner.json\",\n \"stamen-terrain\": \"https://tiles.stadiamaps.com/styles/stamen_terrain.json\",\n \"stamen-watercolor\": \"https://tiles.stadiamaps.com/styles/stamen_watercolor.json\"\n};\n\nconst HTTP_URL_REGEX = /^https?:\\/\\//i;\n\nexport type MapLibreBuiltInStyle = keyof typeof STYLE_MAP;\n\nfunction isStadiaMapsUrl(url: string): boolean {\n try {\n const parsed = new URL(url);\n return parsed.hostname === \"tiles.stadiamaps.com\";\n } catch {\n return false;\n }\n}\n\nfunction assertStadiaApiKey(stadiaApiKey: string): void {\n if (!stadiaApiKey.trim()) {\n throw new Error(\n \"A non-empty stadiaApiKey is required for Stadia Maps style URLs.\"\n );\n }\n}\n\nexport function appendStadiaApiKey(\n styleUrl: string,\n stadiaApiKey: string\n): string {\n if (!isStadiaMapsUrl(styleUrl)) {\n return styleUrl;\n }\n\n assertStadiaApiKey(stadiaApiKey);\n\n const parsed = new URL(styleUrl);\n if (!parsed.searchParams.has(\"api_key\")) {\n parsed.searchParams.set(\"api_key\", stadiaApiKey);\n }\n\n return parsed.toString();\n}\n\nexport function getMapLibreStyleUrl(\n value: string | undefined,\n stadiaApiKey: string\n): string {\n const normalizedApiKey = stadiaApiKey.trim();\n\n if (!value || typeof value !== \"string\") {\n if (!normalizedApiKey) {\n return MAPLIBRE_DEFAULT_STYLE_URL;\n }\n return appendStadiaApiKey(DEFAULT_STADIA_STYLE_URL, normalizedApiKey);\n }\n\n if (STYLE_MAP[value]) {\n const mappedStyleUrl = STYLE_MAP[value];\n if (isStadiaMapsUrl(mappedStyleUrl) && !normalizedApiKey) {\n return MAPLIBRE_DEFAULT_STYLE_URL;\n }\n return appendStadiaApiKey(mappedStyleUrl, normalizedApiKey);\n }\n\n if (HTTP_URL_REGEX.test(value)) {\n if (isStadiaMapsUrl(value) && !normalizedApiKey) {\n return MAPLIBRE_DEFAULT_STYLE_URL;\n }\n return appendStadiaApiKey(value, normalizedApiKey);\n }\n\n if (!normalizedApiKey) {\n return MAPLIBRE_DEFAULT_STYLE_URL;\n }\n\n return appendStadiaApiKey(DEFAULT_STADIA_STYLE_URL, normalizedApiKey);\n}\n\nexport { DEFAULT_STADIA_STYLE_URL, MAPLIBRE_DEFAULT_STYLE_URL };\n","export function generateGoogleMapLink(\n latitude: number,\n longitude: number,\n zoom = 15\n): string {\n return `https://www.google.com/maps/@${latitude},${longitude},${zoom}z`;\n}\n\nexport function generateGoogleDirectionsLink(\n latitude: number,\n longitude: number\n): string {\n return `https://www.google.com/maps/dir/?api=1&destination=${latitude},${longitude}`;\n}\n"]}
1
+ {"version":3,"sources":["../../src/utils/getMapLibreStyleUrl.ts","../../src/utils/googleMapLinks.ts","../../src/utils/cn.ts","../../src/utils/simple-pressable.tsx"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["const MAPLIBRE_DEFAULT_STYLE_URL =\n \"https://basemaps.cartocdn.com/gl/positron-gl-style/style.json\";\nconst DEFAULT_STADIA_STYLE_URL =\n \"https://tiles.stadiamaps.com/styles/osm_bright.json\";\n\nconst STYLE_MAP: Record<string, string> = {\n default: DEFAULT_STADIA_STYLE_URL,\n \"alidade-smooth\": \"https://tiles.stadiamaps.com/styles/alidade_smooth.json\",\n \"alidade-smooth-dark\": \"https://tiles.stadiamaps.com/styles/alidade_smooth_dark.json\",\n \"maplibre-default\": MAPLIBRE_DEFAULT_STYLE_URL,\n \"osm-bright\": \"https://tiles.stadiamaps.com/styles/osm_bright.json\",\n \"stadia-outdoors\": \"https://tiles.stadiamaps.com/styles/outdoors.json\",\n \"stamen-toner\": \"https://tiles.stadiamaps.com/styles/stamen_toner.json\",\n \"stamen-terrain\": \"https://tiles.stadiamaps.com/styles/stamen_terrain.json\",\n \"stamen-watercolor\": \"https://tiles.stadiamaps.com/styles/stamen_watercolor.json\"\n};\n\nconst HTTP_URL_REGEX = /^https?:\\/\\//i;\n\nexport type MapLibreBuiltInStyle = keyof typeof STYLE_MAP;\n\nfunction isStadiaMapsUrl(url: string): boolean {\n try {\n const parsed = new URL(url);\n return parsed.hostname === \"tiles.stadiamaps.com\";\n } catch {\n return false;\n }\n}\n\nfunction assertStadiaApiKey(stadiaApiKey: string): void {\n if (!stadiaApiKey.trim()) {\n throw new Error(\n \"A non-empty stadiaApiKey is required for Stadia Maps style URLs.\"\n );\n }\n}\n\nexport function appendStadiaApiKey(\n styleUrl: string,\n stadiaApiKey: string\n): string {\n if (!isStadiaMapsUrl(styleUrl)) {\n return styleUrl;\n }\n\n assertStadiaApiKey(stadiaApiKey);\n\n const parsed = new URL(styleUrl);\n if (!parsed.searchParams.has(\"api_key\")) {\n parsed.searchParams.set(\"api_key\", stadiaApiKey);\n }\n\n return parsed.toString();\n}\n\nexport function getMapLibreStyleUrl(\n value: string | undefined,\n stadiaApiKey: string\n): string {\n const normalizedApiKey = stadiaApiKey.trim();\n\n if (!value || typeof value !== \"string\") {\n if (!normalizedApiKey) {\n return MAPLIBRE_DEFAULT_STYLE_URL;\n }\n return appendStadiaApiKey(DEFAULT_STADIA_STYLE_URL, normalizedApiKey);\n }\n\n if (STYLE_MAP[value]) {\n const mappedStyleUrl = STYLE_MAP[value];\n if (isStadiaMapsUrl(mappedStyleUrl) && !normalizedApiKey) {\n return MAPLIBRE_DEFAULT_STYLE_URL;\n }\n return appendStadiaApiKey(mappedStyleUrl, normalizedApiKey);\n }\n\n if (HTTP_URL_REGEX.test(value)) {\n if (isStadiaMapsUrl(value) && !normalizedApiKey) {\n return MAPLIBRE_DEFAULT_STYLE_URL;\n }\n return appendStadiaApiKey(value, normalizedApiKey);\n }\n\n if (!normalizedApiKey) {\n return MAPLIBRE_DEFAULT_STYLE_URL;\n }\n\n return appendStadiaApiKey(DEFAULT_STADIA_STYLE_URL, normalizedApiKey);\n}\n\nexport { DEFAULT_STADIA_STYLE_URL, MAPLIBRE_DEFAULT_STYLE_URL };\n","export function generateGoogleMapLink(\n latitude: number,\n longitude: number,\n zoom = 15\n): string {\n return `https://www.google.com/maps/@${latitude},${longitude},${zoom}z`;\n}\n\nexport function generateGoogleDirectionsLink(\n latitude: number,\n longitude: number\n): string {\n return `https://www.google.com/maps/dir/?api=1&destination=${latitude},${longitude}`;\n}\n","import { clsx, type ClassValue } from \"clsx\";\nimport { twMerge } from \"tailwind-merge\";\n\n/**\n * Utility function to merge Tailwind CSS classes with proper conflict resolution.\n * Uses clsx for conditional class names and tailwind-merge for deduplication.\n *\n * @param inputs - Class names to merge\n * @returns Merged class string\n *\n * @example\n * cn(\"px-2 py-1\", \"px-4\") // => \"py-1 px-4\"\n * cn(\"text-red-500\", condition && \"text-blue-500\") // => \"text-blue-500\" (if condition is true)\n */\nexport function cn(...inputs: ClassValue[]) {\n return twMerge(clsx(inputs));\n}\n","/**\n * Simple Pressable component for internal use by map components.\n * This is a lightweight version focused on basic link/button functionality.\n * For full-featured Pressable, use @opensite/ui.\n */\n\nimport * as React from \"react\";\n\nexport interface SimplePressableProps {\n children: React.ReactNode;\n className?: string;\n href?: string;\n onClick?: React.MouseEventHandler<HTMLElement>;\n \"aria-label\"?: string;\n target?: string;\n rel?: string;\n}\n\nexport const SimplePressable = React.forwardRef<\n HTMLAnchorElement | HTMLButtonElement,\n SimplePressableProps\n>(({ children, className, href, onClick, ...props }, ref) => {\n if (href) {\n const isExternal = href.startsWith(\"http://\") || href.startsWith(\"https://\");\n return (\n <a\n ref={ref as React.Ref<HTMLAnchorElement>}\n href={href}\n className={className}\n target={isExternal ? \"_blank\" : props.target}\n rel={isExternal ? \"noopener noreferrer\" : props.rel}\n onClick={onClick}\n {...props}\n >\n {children}\n </a>\n );\n }\n\n if (onClick) {\n return (\n <button\n ref={ref as React.Ref<HTMLButtonElement>}\n type=\"button\"\n className={className}\n onClick={onClick}\n {...props}\n >\n {children}\n </button>\n );\n }\n\n return <span className={className}>{children}</span>;\n});\n\nSimplePressable.displayName = \"SimplePressable\";\n"]}
@@ -0,0 +1,63 @@
1
+ 'use strict';
2
+
3
+ var React = require('react');
4
+ var jsxRuntime = require('react/jsx-runtime');
5
+
6
+ function _interopNamespace(e) {
7
+ if (e && e.__esModule) return e;
8
+ var n = Object.create(null);
9
+ if (e) {
10
+ Object.keys(e).forEach(function (k) {
11
+ if (k !== 'default') {
12
+ var d = Object.getOwnPropertyDescriptor(e, k);
13
+ Object.defineProperty(n, k, d.get ? d : {
14
+ enumerable: true,
15
+ get: function () { return e[k]; }
16
+ });
17
+ }
18
+ });
19
+ }
20
+ n.default = e;
21
+ return Object.freeze(n);
22
+ }
23
+
24
+ var React__namespace = /*#__PURE__*/_interopNamespace(React);
25
+
26
+ // src/utils/simple-pressable.tsx
27
+ var SimplePressable = React__namespace.forwardRef(({ children, className, href, onClick, ...props }, ref) => {
28
+ if (href) {
29
+ const isExternal = href.startsWith("http://") || href.startsWith("https://");
30
+ return /* @__PURE__ */ jsxRuntime.jsx(
31
+ "a",
32
+ {
33
+ ref,
34
+ href,
35
+ className,
36
+ target: isExternal ? "_blank" : props.target,
37
+ rel: isExternal ? "noopener noreferrer" : props.rel,
38
+ onClick,
39
+ ...props,
40
+ children
41
+ }
42
+ );
43
+ }
44
+ if (onClick) {
45
+ return /* @__PURE__ */ jsxRuntime.jsx(
46
+ "button",
47
+ {
48
+ ref,
49
+ type: "button",
50
+ className,
51
+ onClick,
52
+ ...props,
53
+ children
54
+ }
55
+ );
56
+ }
57
+ return /* @__PURE__ */ jsxRuntime.jsx("span", { className, children });
58
+ });
59
+ SimplePressable.displayName = "SimplePressable";
60
+
61
+ exports.SimplePressable = SimplePressable;
62
+ //# sourceMappingURL=simple-pressable.cjs.map
63
+ //# sourceMappingURL=simple-pressable.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/utils/simple-pressable.tsx"],"names":["React","jsx"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;AAkBO,IAAM,eAAA,GAAwBA,gBAAA,CAAA,UAAA,CAGnC,CAAC,EAAE,QAAA,EAAU,SAAA,EAAW,IAAA,EAAM,OAAA,EAAS,GAAG,KAAA,EAAM,EAAG,GAAA,KAAQ;AAC3D,EAAA,IAAI,IAAA,EAAM;AACR,IAAA,MAAM,aAAa,IAAA,CAAK,UAAA,CAAW,SAAS,CAAA,IAAK,IAAA,CAAK,WAAW,UAAU,CAAA;AAC3E,IAAA,uBACEC,cAAA;AAAA,MAAC,GAAA;AAAA,MAAA;AAAA,QACC,GAAA;AAAA,QACA,IAAA;AAAA,QACA,SAAA;AAAA,QACA,MAAA,EAAQ,UAAA,GAAa,QAAA,GAAW,KAAA,CAAM,MAAA;AAAA,QACtC,GAAA,EAAK,UAAA,GAAa,qBAAA,GAAwB,KAAA,CAAM,GAAA;AAAA,QAChD,OAAA;AAAA,QACC,GAAG,KAAA;AAAA,QAEH;AAAA;AAAA,KACH;AAAA,EAEJ;AAEA,EAAA,IAAI,OAAA,EAAS;AACX,IAAA,uBACEA,cAAA;AAAA,MAAC,QAAA;AAAA,MAAA;AAAA,QACC,GAAA;AAAA,QACA,IAAA,EAAK,QAAA;AAAA,QACL,SAAA;AAAA,QACA,OAAA;AAAA,QACC,GAAG,KAAA;AAAA,QAEH;AAAA;AAAA,KACH;AAAA,EAEJ;AAEA,EAAA,uBAAOA,cAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAuB,QAAA,EAAS,CAAA;AAC/C,CAAC;AAED,eAAA,CAAgB,WAAA,GAAc,iBAAA","file":"simple-pressable.cjs","sourcesContent":["/**\n * Simple Pressable component for internal use by map components.\n * This is a lightweight version focused on basic link/button functionality.\n * For full-featured Pressable, use @opensite/ui.\n */\n\nimport * as React from \"react\";\n\nexport interface SimplePressableProps {\n children: React.ReactNode;\n className?: string;\n href?: string;\n onClick?: React.MouseEventHandler<HTMLElement>;\n \"aria-label\"?: string;\n target?: string;\n rel?: string;\n}\n\nexport const SimplePressable = React.forwardRef<\n HTMLAnchorElement | HTMLButtonElement,\n SimplePressableProps\n>(({ children, className, href, onClick, ...props }, ref) => {\n if (href) {\n const isExternal = href.startsWith(\"http://\") || href.startsWith(\"https://\");\n return (\n <a\n ref={ref as React.Ref<HTMLAnchorElement>}\n href={href}\n className={className}\n target={isExternal ? \"_blank\" : props.target}\n rel={isExternal ? \"noopener noreferrer\" : props.rel}\n onClick={onClick}\n {...props}\n >\n {children}\n </a>\n );\n }\n\n if (onClick) {\n return (\n <button\n ref={ref as React.Ref<HTMLButtonElement>}\n type=\"button\"\n className={className}\n onClick={onClick}\n {...props}\n >\n {children}\n </button>\n );\n }\n\n return <span className={className}>{children}</span>;\n});\n\nSimplePressable.displayName = \"SimplePressable\";\n"]}
@@ -0,0 +1,20 @@
1
+ import * as React from 'react';
2
+
3
+ /**
4
+ * Simple Pressable component for internal use by map components.
5
+ * This is a lightweight version focused on basic link/button functionality.
6
+ * For full-featured Pressable, use @opensite/ui.
7
+ */
8
+
9
+ interface SimplePressableProps {
10
+ children: React.ReactNode;
11
+ className?: string;
12
+ href?: string;
13
+ onClick?: React.MouseEventHandler<HTMLElement>;
14
+ "aria-label"?: string;
15
+ target?: string;
16
+ rel?: string;
17
+ }
18
+ declare const SimplePressable: React.ForwardRefExoticComponent<SimplePressableProps & React.RefAttributes<HTMLAnchorElement | HTMLButtonElement>>;
19
+
20
+ export { SimplePressable, type SimplePressableProps };
@@ -0,0 +1,20 @@
1
+ import * as React from 'react';
2
+
3
+ /**
4
+ * Simple Pressable component for internal use by map components.
5
+ * This is a lightweight version focused on basic link/button functionality.
6
+ * For full-featured Pressable, use @opensite/ui.
7
+ */
8
+
9
+ interface SimplePressableProps {
10
+ children: React.ReactNode;
11
+ className?: string;
12
+ href?: string;
13
+ onClick?: React.MouseEventHandler<HTMLElement>;
14
+ "aria-label"?: string;
15
+ target?: string;
16
+ rel?: string;
17
+ }
18
+ declare const SimplePressable: React.ForwardRefExoticComponent<SimplePressableProps & React.RefAttributes<HTMLAnchorElement | HTMLButtonElement>>;
19
+
20
+ export { SimplePressable, type SimplePressableProps };
@@ -0,0 +1,41 @@
1
+ import * as React from 'react';
2
+ import { jsx } from 'react/jsx-runtime';
3
+
4
+ // src/utils/simple-pressable.tsx
5
+ var SimplePressable = React.forwardRef(({ children, className, href, onClick, ...props }, ref) => {
6
+ if (href) {
7
+ const isExternal = href.startsWith("http://") || href.startsWith("https://");
8
+ return /* @__PURE__ */ jsx(
9
+ "a",
10
+ {
11
+ ref,
12
+ href,
13
+ className,
14
+ target: isExternal ? "_blank" : props.target,
15
+ rel: isExternal ? "noopener noreferrer" : props.rel,
16
+ onClick,
17
+ ...props,
18
+ children
19
+ }
20
+ );
21
+ }
22
+ if (onClick) {
23
+ return /* @__PURE__ */ jsx(
24
+ "button",
25
+ {
26
+ ref,
27
+ type: "button",
28
+ className,
29
+ onClick,
30
+ ...props,
31
+ children
32
+ }
33
+ );
34
+ }
35
+ return /* @__PURE__ */ jsx("span", { className, children });
36
+ });
37
+ SimplePressable.displayName = "SimplePressable";
38
+
39
+ export { SimplePressable };
40
+ //# sourceMappingURL=simple-pressable.js.map
41
+ //# sourceMappingURL=simple-pressable.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../src/utils/simple-pressable.tsx"],"names":[],"mappings":";;;;AAkBO,IAAM,eAAA,GAAwB,KAAA,CAAA,UAAA,CAGnC,CAAC,EAAE,QAAA,EAAU,SAAA,EAAW,IAAA,EAAM,OAAA,EAAS,GAAG,KAAA,EAAM,EAAG,GAAA,KAAQ;AAC3D,EAAA,IAAI,IAAA,EAAM;AACR,IAAA,MAAM,aAAa,IAAA,CAAK,UAAA,CAAW,SAAS,CAAA,IAAK,IAAA,CAAK,WAAW,UAAU,CAAA;AAC3E,IAAA,uBACE,GAAA;AAAA,MAAC,GAAA;AAAA,MAAA;AAAA,QACC,GAAA;AAAA,QACA,IAAA;AAAA,QACA,SAAA;AAAA,QACA,MAAA,EAAQ,UAAA,GAAa,QAAA,GAAW,KAAA,CAAM,MAAA;AAAA,QACtC,GAAA,EAAK,UAAA,GAAa,qBAAA,GAAwB,KAAA,CAAM,GAAA;AAAA,QAChD,OAAA;AAAA,QACC,GAAG,KAAA;AAAA,QAEH;AAAA;AAAA,KACH;AAAA,EAEJ;AAEA,EAAA,IAAI,OAAA,EAAS;AACX,IAAA,uBACE,GAAA;AAAA,MAAC,QAAA;AAAA,MAAA;AAAA,QACC,GAAA;AAAA,QACA,IAAA,EAAK,QAAA;AAAA,QACL,SAAA;AAAA,QACA,OAAA;AAAA,QACC,GAAG,KAAA;AAAA,QAEH;AAAA;AAAA,KACH;AAAA,EAEJ;AAEA,EAAA,uBAAO,GAAA,CAAC,MAAA,EAAA,EAAK,SAAA,EAAuB,QAAA,EAAS,CAAA;AAC/C,CAAC;AAED,eAAA,CAAgB,WAAA,GAAc,iBAAA","file":"simple-pressable.js","sourcesContent":["/**\n * Simple Pressable component for internal use by map components.\n * This is a lightweight version focused on basic link/button functionality.\n * For full-featured Pressable, use @opensite/ui.\n */\n\nimport * as React from \"react\";\n\nexport interface SimplePressableProps {\n children: React.ReactNode;\n className?: string;\n href?: string;\n onClick?: React.MouseEventHandler<HTMLElement>;\n \"aria-label\"?: string;\n target?: string;\n rel?: string;\n}\n\nexport const SimplePressable = React.forwardRef<\n HTMLAnchorElement | HTMLButtonElement,\n SimplePressableProps\n>(({ children, className, href, onClick, ...props }, ref) => {\n if (href) {\n const isExternal = href.startsWith(\"http://\") || href.startsWith(\"https://\");\n return (\n <a\n ref={ref as React.Ref<HTMLAnchorElement>}\n href={href}\n className={className}\n target={isExternal ? \"_blank\" : props.target}\n rel={isExternal ? \"noopener noreferrer\" : props.rel}\n onClick={onClick}\n {...props}\n >\n {children}\n </a>\n );\n }\n\n if (onClick) {\n return (\n <button\n ref={ref as React.Ref<HTMLButtonElement>}\n type=\"button\"\n className={className}\n onClick={onClick}\n {...props}\n >\n {children}\n </button>\n );\n }\n\n return <span className={className}>{children}</span>;\n});\n\nSimplePressable.displayName = \"SimplePressable\";\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@page-speed/maps",
3
- "version": "0.1.3",
3
+ "version": "0.1.5",
4
4
  "description": "Performance-optimized MapLibre React components and style utilities for DashTrack.",
5
5
  "keywords": [
6
6
  "react",
@@ -85,6 +85,31 @@
85
85
  "types": "./dist/hooks/useDefaultZoom.d.ts",
86
86
  "import": "./dist/hooks/useDefaultZoom.js",
87
87
  "require": "./dist/hooks/useDefaultZoom.cjs"
88
+ },
89
+ "./components": {
90
+ "types": "./dist/components/index.d.ts",
91
+ "import": "./dist/components/index.js",
92
+ "require": "./dist/components/index.cjs"
93
+ },
94
+ "./components/geo-map": {
95
+ "types": "./dist/components/geo-map.d.ts",
96
+ "import": "./dist/components/geo-map.js",
97
+ "require": "./dist/components/geo-map.cjs"
98
+ },
99
+ "./components/map-marker": {
100
+ "types": "./dist/components/map-marker.d.ts",
101
+ "import": "./dist/components/map-marker.js",
102
+ "require": "./dist/components/map-marker.cjs"
103
+ },
104
+ "./utils/cn": {
105
+ "types": "./dist/utils/cn.d.ts",
106
+ "import": "./dist/utils/cn.js",
107
+ "require": "./dist/utils/cn.cjs"
108
+ },
109
+ "./utils/simple-pressable": {
110
+ "types": "./dist/utils/simple-pressable.d.ts",
111
+ "import": "./dist/utils/simple-pressable.js",
112
+ "require": "./dist/utils/simple-pressable.cjs"
88
113
  }
89
114
  },
90
115
  "scripts": {
@@ -103,8 +128,10 @@
103
128
  "react-dom": ">=18.0.0"
104
129
  },
105
130
  "dependencies": {
131
+ "clsx": "^2.1.1",
106
132
  "maplibre-gl": "^5.18.0",
107
- "react-map-gl": "^8.1.0"
133
+ "react-map-gl": "^8.1.0",
134
+ "tailwind-merge": "^2.5.5"
108
135
  },
109
136
  "devDependencies": {
110
137
  "@testing-library/dom": "^10.4.1",