@sonordev/site-kit 2.2.1 → 2.2.2

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.
@@ -7,15 +7,32 @@ var jsxRuntime = require('react/jsx-runtime');
7
7
  // src/reputation/TestimonialSection.tsx
8
8
 
9
9
  // src/reputation/api.ts
10
+ var reviewsCache = null;
11
+ var CACHE_TTL = 6e4;
10
12
  function getApiConfig() {
11
13
  const apiUrl = typeof window !== "undefined" ? window.__SITE_KIT_API_URL__ || "https://api.sonor.io" : process.env.SONOR_API_URL || "https://api.sonor.io";
12
14
  const apiKey = typeof window !== "undefined" ? window.__SITE_KIT_API_KEY__ || "" : process.env.SONOR_API_KEY || "";
13
15
  return { apiUrl, apiKey };
14
16
  }
17
+ async function fetchWithRetry(url, headers, retries = 2) {
18
+ for (let attempt = 0; attempt <= retries; attempt++) {
19
+ const response = await fetch(url, { headers });
20
+ if (response.ok) return response;
21
+ if (response.status === 429 && attempt < retries) {
22
+ const delay = Math.pow(2, attempt + 1) * 1e3;
23
+ await new Promise((r) => setTimeout(r, delay));
24
+ continue;
25
+ }
26
+ return response;
27
+ }
28
+ return new Response(null, { status: 500 });
29
+ }
15
30
  async function fetchReviews(options = {}) {
31
+ if (reviewsCache && Date.now() - reviewsCache.timestamp < CACHE_TTL) {
32
+ return reviewsCache.data;
33
+ }
16
34
  const { apiUrl, apiKey } = getApiConfig();
17
35
  if (!apiKey) {
18
- console.warn("[Reputation] No API key configured");
19
36
  return [];
20
37
  }
21
38
  try {
@@ -24,15 +41,20 @@ async function fetchReviews(options = {}) {
24
41
  if (options.limit) params.append("limit", options.limit.toString());
25
42
  if (options.featured) params.append("featured", "true");
26
43
  const qs = params.toString() ? `?${params.toString()}` : "";
27
- const response = await fetch(`${apiUrl}/api/public/reviews${qs}`, {
28
- headers: { "x-api-key": apiKey }
29
- });
44
+ const response = await fetchWithRetry(
45
+ `${apiUrl}/api/public/reviews${qs}`,
46
+ { "x-api-key": apiKey }
47
+ );
30
48
  if (!response.ok) {
31
- console.error("[Reputation] Error fetching reviews:", response.statusText);
49
+ if (response.status !== 429) {
50
+ console.error("[Reputation] Error fetching reviews:", response.statusText);
51
+ }
32
52
  return [];
33
53
  }
34
54
  const data = await response.json();
35
- return data.reviews || [];
55
+ const reviews = data.reviews || [];
56
+ reviewsCache = { data: reviews, timestamp: Date.now() };
57
+ return reviews;
36
58
  } catch (error) {
37
59
  console.error("[Reputation] Error fetching reviews:", error);
38
60
  return [];
@@ -41,15 +63,14 @@ async function fetchReviews(options = {}) {
41
63
  async function fetchReviewStats() {
42
64
  const { apiUrl, apiKey } = getApiConfig();
43
65
  if (!apiKey) {
44
- console.warn("[Reputation] No API key configured");
45
66
  return null;
46
67
  }
47
68
  try {
48
- const response = await fetch(`${apiUrl}/api/public/reviews/stats`, {
49
- headers: { "x-api-key": apiKey }
50
- });
69
+ const response = await fetchWithRetry(
70
+ `${apiUrl}/api/public/reviews/stats`,
71
+ { "x-api-key": apiKey }
72
+ );
51
73
  if (!response.ok) {
52
- console.error("[Reputation] Error fetching stats:", response.statusText);
53
74
  return null;
54
75
  }
55
76
  return await response.json();
@@ -282,5 +303,5 @@ function TestimonialSection({
282
303
  exports.TestimonialSection = TestimonialSection;
283
304
  exports.fetchReviewStats = fetchReviewStats;
284
305
  exports.fetchReviews = fetchReviews;
285
- //# sourceMappingURL=chunk-KLIT3WTM.js.map
286
- //# sourceMappingURL=chunk-KLIT3WTM.js.map
306
+ //# sourceMappingURL=chunk-EATGSR46.js.map
307
+ //# sourceMappingURL=chunk-EATGSR46.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/reputation/api.ts","../src/reputation/TestimonialSection.tsx"],"names":["useState","useEffect","useCallback","jsx","jsxs"],"mappings":"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","file":"chunk-EATGSR46.js","sourcesContent":["/**\n * @sonordev/site-kit/reputation - API Functions\n *\n * Fetch reviews and stats from Portal API public endpoint.\n * Uses API key authentication (x-api-key header) - project is resolved server-side.\n * Includes retry with backoff for rate limiting (429).\n */\n\nimport type { Review, ReviewStats } from './types'\n\ninterface FetchReviewsOptions {\n service?: string\n limit?: number\n featured?: boolean\n}\n\n/** In-memory cache to avoid redundant fetches during dev hot reload */\nlet reviewsCache: { data: Review[]; timestamp: number } | null = null\nconst CACHE_TTL = 60_000 // 1 minute\n\nfunction getApiConfig() {\n const apiUrl =\n typeof window !== 'undefined'\n ? (window as any).__SITE_KIT_API_URL__ || 'https://api.sonor.io'\n : process.env.SONOR_API_URL || 'https://api.sonor.io'\n const apiKey =\n typeof window !== 'undefined'\n ? (window as any).__SITE_KIT_API_KEY__ || ''\n : process.env.SONOR_API_KEY || ''\n return { apiUrl, apiKey }\n}\n\nasync function fetchWithRetry(\n url: string,\n headers: Record<string, string>,\n retries = 2,\n): Promise<Response> {\n for (let attempt = 0; attempt <= retries; attempt++) {\n const response = await fetch(url, { headers })\n\n if (response.ok) return response\n\n // Rate limited: wait and retry\n if (response.status === 429 && attempt < retries) {\n const delay = Math.pow(2, attempt + 1) * 1000 // 2s, 4s\n await new Promise((r) => setTimeout(r, delay))\n continue\n }\n\n return response // Return the error response\n }\n\n // Should not reach here, but satisfy TS\n return new Response(null, { status: 500 })\n}\n\nexport async function fetchReviews(\n options: FetchReviewsOptions = {},\n): Promise<Review[]> {\n // Check cache first (prevents hammering API during dev hot reload)\n if (reviewsCache && Date.now() - reviewsCache.timestamp < CACHE_TTL) {\n return reviewsCache.data\n }\n\n const { apiUrl, apiKey } = getApiConfig()\n\n if (!apiKey) {\n // Silent in production, only warn in debug\n return []\n }\n\n try {\n const params = new URLSearchParams()\n if (options.service) params.append('service', options.service)\n if (options.limit) params.append('limit', options.limit.toString())\n if (options.featured) params.append('featured', 'true')\n\n const qs = params.toString() ? `?${params.toString()}` : ''\n const response = await fetchWithRetry(\n `${apiUrl}/api/public/reviews${qs}`,\n { 'x-api-key': apiKey },\n )\n\n if (!response.ok) {\n // Only log non-429 errors (429 was already retried)\n if (response.status !== 429) {\n console.error('[Reputation] Error fetching reviews:', response.statusText)\n }\n return []\n }\n\n const data = await response.json()\n const reviews = data.reviews || []\n\n // Cache the result\n reviewsCache = { data: reviews, timestamp: Date.now() }\n\n return reviews\n } catch (error) {\n console.error('[Reputation] Error fetching reviews:', error)\n return []\n }\n}\n\nexport async function fetchReviewStats(): Promise<ReviewStats | null> {\n const { apiUrl, apiKey } = getApiConfig()\n\n if (!apiKey) {\n return null\n }\n\n try {\n const response = await fetchWithRetry(\n `${apiUrl}/api/public/reviews/stats`,\n { 'x-api-key': apiKey },\n )\n\n if (!response.ok) {\n return null\n }\n\n return await response.json()\n } catch (error) {\n console.error('[Reputation] Error fetching stats:', error)\n return null\n }\n}\n","/**\n * @sonordev/site-kit/reputation - Testimonial Section\n * \n * Displays client reviews in a rotating carousel\n * Fetches published reviews from Portal API public endpoint\n * \n * Minimal styling - let site CSS control appearance\n */\n\n'use client'\n\nimport React, { useEffect, useState, useCallback } from 'react'\nimport { fetchReviews } from './api'\nimport type { Review, TestimonialSectionProps } from './types'\n\nexport function TestimonialSection({\n title,\n subtitle,\n autoplay = true,\n autoplayInterval = 5000,\n showRating = true,\n maxReviews,\n featuredOnly = false,\n service,\n className = '',\n}: TestimonialSectionProps) {\n const [reviews, setReviews] = useState<Review[]>([])\n const [currentIndex, setCurrentIndex] = useState(0)\n const [isLoading, setIsLoading] = useState(true)\n const [direction, setDirection] = useState<'next' | 'prev'>('next')\n\n // Load reviews\n useEffect(() => {\n async function loadReviews() {\n setIsLoading(true)\n const data = await fetchReviews({\n service,\n limit: maxReviews,\n featured: featuredOnly,\n })\n setReviews(data)\n setIsLoading(false)\n }\n loadReviews()\n }, [service, maxReviews, featuredOnly])\n\n // Auto-rotate\n useEffect(() => {\n if (!autoplay || reviews.length <= 1) return\n\n const interval = setInterval(() => {\n setDirection('next')\n setCurrentIndex((prev) => (prev + 1) % reviews.length)\n }, autoplayInterval)\n\n return () => clearInterval(interval)\n }, [autoplay, autoplayInterval, reviews.length])\n\n const goToSlide = useCallback((index: number) => {\n setDirection(index > currentIndex ? 'next' : 'prev')\n setCurrentIndex(index)\n }, [currentIndex])\n\n const nextSlide = useCallback(() => {\n setDirection('next')\n setCurrentIndex((prev) => (prev + 1) % reviews.length)\n }, [reviews.length])\n\n const prevSlide = useCallback(() => {\n setDirection('prev')\n setCurrentIndex((prev) => (prev - 1 + reviews.length) % reviews.length)\n }, [reviews.length])\n\n if (isLoading) {\n return null // Let site handle loading state\n }\n\n if (!reviews.length) {\n return null\n }\n\n const currentReview = reviews[currentIndex]\n\n return (\n <section className={`testimonial-section ${className}`} data-site-kit-testimonials>\n <style>{`\n .testimonial-section {\n position: relative;\n padding: 4rem 1rem;\n }\n \n .testimonial-content {\n position: relative;\n overflow: hidden;\n }\n \n .testimonial-slide {\n animation: ${direction === 'next' ? 'slideInRight' : 'slideInLeft'} 0.6s cubic-bezier(0.4, 0, 0.2, 1);\n }\n \n @keyframes slideInRight {\n from {\n opacity: 0;\n transform: translateX(50px);\n }\n to {\n opacity: 1;\n transform: translateX(0);\n }\n }\n \n @keyframes slideInLeft {\n from {\n opacity: 0;\n transform: translateX(-50px);\n }\n to {\n opacity: 1;\n transform: translateX(0);\n }\n }\n \n .testimonial-quote-icon {\n font-size: 4rem;\n line-height: 1;\n text-align: center;\n margin-bottom: 1.5rem;\n opacity: 0.15;\n }\n \n .testimonial-stars {\n display: flex;\n justify-content: center;\n gap: 0.25rem;\n margin-bottom: 1.5rem;\n }\n \n .testimonial-quote {\n text-align: center;\n font-size: 1.25rem;\n line-height: 1.8;\n margin-bottom: 2rem;\n }\n \n .testimonial-author {\n display: flex;\n flex-direction: column;\n align-items: center;\n gap: 1rem;\n }\n \n .testimonial-avatar {\n width: 4rem;\n height: 4rem;\n border-radius: 50%;\n display: flex;\n align-items: center;\n justify-content: center;\n font-size: 1.5rem;\n }\n \n .testimonial-name {\n font-weight: 600;\n font-size: 1.125rem;\n }\n \n .testimonial-role {\n font-size: 0.875rem;\n opacity: 0.7;\n }\n \n .testimonial-nav {\n display: flex;\n align-items: center;\n justify-content: center;\n gap: 1.5rem;\n margin-top: 2rem;\n }\n \n .testimonial-nav-button {\n border: none;\n background: none;\n cursor: pointer;\n padding: 0.5rem;\n transition: opacity 0.2s;\n font-size: 1.5rem;\n }\n \n .testimonial-nav-button:hover {\n opacity: 0.7;\n }\n \n .testimonial-nav-button:disabled {\n opacity: 0.3;\n cursor: not-allowed;\n }\n `}</style>\n\n <div className=\"container\">\n {title && <h2 className=\"testimonial-title\">{title}</h2>}\n {subtitle && <p className=\"testimonial-subtitle\">{subtitle}</p>}\n\n <div className=\"testimonial-content\">\n <div className=\"testimonial-slide\" key={currentIndex}>\n {/* Quote Icon - site can style this */}\n <div className=\"testimonial-quote-icon\">&ldquo;</div>\n\n {/* Stars */}\n {showRating && (\n <div className=\"testimonial-stars\">\n {[...Array(5)].map((_, i) => (\n <span\n key={i}\n className={`star ${i < currentReview.rating ? 'star-filled' : 'star-empty'}`}\n aria-hidden=\"true\"\n >\n ★\n </span>\n ))}\n </div>\n )}\n\n {/* Quote Text */}\n <blockquote className=\"testimonial-quote\">\n &ldquo;{currentReview.quote}&rdquo;\n </blockquote>\n\n {/* Author */}\n <div className=\"testimonial-author\">\n <div className=\"testimonial-avatar\" aria-hidden=\"true\">\n {currentReview.image ? (\n <img src={currentReview.image} alt=\"\" />\n ) : (\n <span>👤</span>\n )}\n </div>\n <div className=\"testimonial-author-info\">\n <div className=\"testimonial-name\">{currentReview.name}</div>\n {currentReview.role && (\n <div className=\"testimonial-role\">{currentReview.role}</div>\n )}\n </div>\n </div>\n </div>\n </div>\n\n {/* Navigation */}\n {reviews.length > 1 && (\n <div className=\"testimonial-nav\">\n <button\n onClick={prevSlide}\n className=\"testimonial-nav-button testimonial-nav-prev\"\n aria-label=\"Previous review\"\n disabled={reviews.length <= 1}\n >\n ‹\n </button>\n\n <button\n onClick={nextSlide}\n className=\"testimonial-nav-button testimonial-nav-next\"\n aria-label=\"Next review\"\n disabled={reviews.length <= 1}\n >\n ›\n </button>\n </div>\n )}\n </div>\n </section>\n )\n}\n"]}
@@ -5,15 +5,32 @@ import { jsxs, jsx } from 'react/jsx-runtime';
5
5
  // src/reputation/TestimonialSection.tsx
6
6
 
7
7
  // src/reputation/api.ts
8
+ var reviewsCache = null;
9
+ var CACHE_TTL = 6e4;
8
10
  function getApiConfig() {
9
11
  const apiUrl = typeof window !== "undefined" ? window.__SITE_KIT_API_URL__ || "https://api.sonor.io" : process.env.SONOR_API_URL || "https://api.sonor.io";
10
12
  const apiKey = typeof window !== "undefined" ? window.__SITE_KIT_API_KEY__ || "" : process.env.SONOR_API_KEY || "";
11
13
  return { apiUrl, apiKey };
12
14
  }
15
+ async function fetchWithRetry(url, headers, retries = 2) {
16
+ for (let attempt = 0; attempt <= retries; attempt++) {
17
+ const response = await fetch(url, { headers });
18
+ if (response.ok) return response;
19
+ if (response.status === 429 && attempt < retries) {
20
+ const delay = Math.pow(2, attempt + 1) * 1e3;
21
+ await new Promise((r) => setTimeout(r, delay));
22
+ continue;
23
+ }
24
+ return response;
25
+ }
26
+ return new Response(null, { status: 500 });
27
+ }
13
28
  async function fetchReviews(options = {}) {
29
+ if (reviewsCache && Date.now() - reviewsCache.timestamp < CACHE_TTL) {
30
+ return reviewsCache.data;
31
+ }
14
32
  const { apiUrl, apiKey } = getApiConfig();
15
33
  if (!apiKey) {
16
- console.warn("[Reputation] No API key configured");
17
34
  return [];
18
35
  }
19
36
  try {
@@ -22,15 +39,20 @@ async function fetchReviews(options = {}) {
22
39
  if (options.limit) params.append("limit", options.limit.toString());
23
40
  if (options.featured) params.append("featured", "true");
24
41
  const qs = params.toString() ? `?${params.toString()}` : "";
25
- const response = await fetch(`${apiUrl}/api/public/reviews${qs}`, {
26
- headers: { "x-api-key": apiKey }
27
- });
42
+ const response = await fetchWithRetry(
43
+ `${apiUrl}/api/public/reviews${qs}`,
44
+ { "x-api-key": apiKey }
45
+ );
28
46
  if (!response.ok) {
29
- console.error("[Reputation] Error fetching reviews:", response.statusText);
47
+ if (response.status !== 429) {
48
+ console.error("[Reputation] Error fetching reviews:", response.statusText);
49
+ }
30
50
  return [];
31
51
  }
32
52
  const data = await response.json();
33
- return data.reviews || [];
53
+ const reviews = data.reviews || [];
54
+ reviewsCache = { data: reviews, timestamp: Date.now() };
55
+ return reviews;
34
56
  } catch (error) {
35
57
  console.error("[Reputation] Error fetching reviews:", error);
36
58
  return [];
@@ -39,15 +61,14 @@ async function fetchReviews(options = {}) {
39
61
  async function fetchReviewStats() {
40
62
  const { apiUrl, apiKey } = getApiConfig();
41
63
  if (!apiKey) {
42
- console.warn("[Reputation] No API key configured");
43
64
  return null;
44
65
  }
45
66
  try {
46
- const response = await fetch(`${apiUrl}/api/public/reviews/stats`, {
47
- headers: { "x-api-key": apiKey }
48
- });
67
+ const response = await fetchWithRetry(
68
+ `${apiUrl}/api/public/reviews/stats`,
69
+ { "x-api-key": apiKey }
70
+ );
49
71
  if (!response.ok) {
50
- console.error("[Reputation] Error fetching stats:", response.statusText);
51
72
  return null;
52
73
  }
53
74
  return await response.json();
@@ -278,5 +299,5 @@ function TestimonialSection({
278
299
  }
279
300
 
280
301
  export { TestimonialSection, fetchReviewStats, fetchReviews };
281
- //# sourceMappingURL=chunk-BJ5IHVFZ.mjs.map
282
- //# sourceMappingURL=chunk-BJ5IHVFZ.mjs.map
302
+ //# sourceMappingURL=chunk-UUDE46TZ.mjs.map
303
+ //# sourceMappingURL=chunk-UUDE46TZ.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/reputation/api.ts","../src/reputation/TestimonialSection.tsx"],"names":[],"mappings":"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","file":"chunk-UUDE46TZ.mjs","sourcesContent":["/**\n * @sonordev/site-kit/reputation - API Functions\n *\n * Fetch reviews and stats from Portal API public endpoint.\n * Uses API key authentication (x-api-key header) - project is resolved server-side.\n * Includes retry with backoff for rate limiting (429).\n */\n\nimport type { Review, ReviewStats } from './types'\n\ninterface FetchReviewsOptions {\n service?: string\n limit?: number\n featured?: boolean\n}\n\n/** In-memory cache to avoid redundant fetches during dev hot reload */\nlet reviewsCache: { data: Review[]; timestamp: number } | null = null\nconst CACHE_TTL = 60_000 // 1 minute\n\nfunction getApiConfig() {\n const apiUrl =\n typeof window !== 'undefined'\n ? (window as any).__SITE_KIT_API_URL__ || 'https://api.sonor.io'\n : process.env.SONOR_API_URL || 'https://api.sonor.io'\n const apiKey =\n typeof window !== 'undefined'\n ? (window as any).__SITE_KIT_API_KEY__ || ''\n : process.env.SONOR_API_KEY || ''\n return { apiUrl, apiKey }\n}\n\nasync function fetchWithRetry(\n url: string,\n headers: Record<string, string>,\n retries = 2,\n): Promise<Response> {\n for (let attempt = 0; attempt <= retries; attempt++) {\n const response = await fetch(url, { headers })\n\n if (response.ok) return response\n\n // Rate limited: wait and retry\n if (response.status === 429 && attempt < retries) {\n const delay = Math.pow(2, attempt + 1) * 1000 // 2s, 4s\n await new Promise((r) => setTimeout(r, delay))\n continue\n }\n\n return response // Return the error response\n }\n\n // Should not reach here, but satisfy TS\n return new Response(null, { status: 500 })\n}\n\nexport async function fetchReviews(\n options: FetchReviewsOptions = {},\n): Promise<Review[]> {\n // Check cache first (prevents hammering API during dev hot reload)\n if (reviewsCache && Date.now() - reviewsCache.timestamp < CACHE_TTL) {\n return reviewsCache.data\n }\n\n const { apiUrl, apiKey } = getApiConfig()\n\n if (!apiKey) {\n // Silent in production, only warn in debug\n return []\n }\n\n try {\n const params = new URLSearchParams()\n if (options.service) params.append('service', options.service)\n if (options.limit) params.append('limit', options.limit.toString())\n if (options.featured) params.append('featured', 'true')\n\n const qs = params.toString() ? `?${params.toString()}` : ''\n const response = await fetchWithRetry(\n `${apiUrl}/api/public/reviews${qs}`,\n { 'x-api-key': apiKey },\n )\n\n if (!response.ok) {\n // Only log non-429 errors (429 was already retried)\n if (response.status !== 429) {\n console.error('[Reputation] Error fetching reviews:', response.statusText)\n }\n return []\n }\n\n const data = await response.json()\n const reviews = data.reviews || []\n\n // Cache the result\n reviewsCache = { data: reviews, timestamp: Date.now() }\n\n return reviews\n } catch (error) {\n console.error('[Reputation] Error fetching reviews:', error)\n return []\n }\n}\n\nexport async function fetchReviewStats(): Promise<ReviewStats | null> {\n const { apiUrl, apiKey } = getApiConfig()\n\n if (!apiKey) {\n return null\n }\n\n try {\n const response = await fetchWithRetry(\n `${apiUrl}/api/public/reviews/stats`,\n { 'x-api-key': apiKey },\n )\n\n if (!response.ok) {\n return null\n }\n\n return await response.json()\n } catch (error) {\n console.error('[Reputation] Error fetching stats:', error)\n return null\n }\n}\n","/**\n * @sonordev/site-kit/reputation - Testimonial Section\n * \n * Displays client reviews in a rotating carousel\n * Fetches published reviews from Portal API public endpoint\n * \n * Minimal styling - let site CSS control appearance\n */\n\n'use client'\n\nimport React, { useEffect, useState, useCallback } from 'react'\nimport { fetchReviews } from './api'\nimport type { Review, TestimonialSectionProps } from './types'\n\nexport function TestimonialSection({\n title,\n subtitle,\n autoplay = true,\n autoplayInterval = 5000,\n showRating = true,\n maxReviews,\n featuredOnly = false,\n service,\n className = '',\n}: TestimonialSectionProps) {\n const [reviews, setReviews] = useState<Review[]>([])\n const [currentIndex, setCurrentIndex] = useState(0)\n const [isLoading, setIsLoading] = useState(true)\n const [direction, setDirection] = useState<'next' | 'prev'>('next')\n\n // Load reviews\n useEffect(() => {\n async function loadReviews() {\n setIsLoading(true)\n const data = await fetchReviews({\n service,\n limit: maxReviews,\n featured: featuredOnly,\n })\n setReviews(data)\n setIsLoading(false)\n }\n loadReviews()\n }, [service, maxReviews, featuredOnly])\n\n // Auto-rotate\n useEffect(() => {\n if (!autoplay || reviews.length <= 1) return\n\n const interval = setInterval(() => {\n setDirection('next')\n setCurrentIndex((prev) => (prev + 1) % reviews.length)\n }, autoplayInterval)\n\n return () => clearInterval(interval)\n }, [autoplay, autoplayInterval, reviews.length])\n\n const goToSlide = useCallback((index: number) => {\n setDirection(index > currentIndex ? 'next' : 'prev')\n setCurrentIndex(index)\n }, [currentIndex])\n\n const nextSlide = useCallback(() => {\n setDirection('next')\n setCurrentIndex((prev) => (prev + 1) % reviews.length)\n }, [reviews.length])\n\n const prevSlide = useCallback(() => {\n setDirection('prev')\n setCurrentIndex((prev) => (prev - 1 + reviews.length) % reviews.length)\n }, [reviews.length])\n\n if (isLoading) {\n return null // Let site handle loading state\n }\n\n if (!reviews.length) {\n return null\n }\n\n const currentReview = reviews[currentIndex]\n\n return (\n <section className={`testimonial-section ${className}`} data-site-kit-testimonials>\n <style>{`\n .testimonial-section {\n position: relative;\n padding: 4rem 1rem;\n }\n \n .testimonial-content {\n position: relative;\n overflow: hidden;\n }\n \n .testimonial-slide {\n animation: ${direction === 'next' ? 'slideInRight' : 'slideInLeft'} 0.6s cubic-bezier(0.4, 0, 0.2, 1);\n }\n \n @keyframes slideInRight {\n from {\n opacity: 0;\n transform: translateX(50px);\n }\n to {\n opacity: 1;\n transform: translateX(0);\n }\n }\n \n @keyframes slideInLeft {\n from {\n opacity: 0;\n transform: translateX(-50px);\n }\n to {\n opacity: 1;\n transform: translateX(0);\n }\n }\n \n .testimonial-quote-icon {\n font-size: 4rem;\n line-height: 1;\n text-align: center;\n margin-bottom: 1.5rem;\n opacity: 0.15;\n }\n \n .testimonial-stars {\n display: flex;\n justify-content: center;\n gap: 0.25rem;\n margin-bottom: 1.5rem;\n }\n \n .testimonial-quote {\n text-align: center;\n font-size: 1.25rem;\n line-height: 1.8;\n margin-bottom: 2rem;\n }\n \n .testimonial-author {\n display: flex;\n flex-direction: column;\n align-items: center;\n gap: 1rem;\n }\n \n .testimonial-avatar {\n width: 4rem;\n height: 4rem;\n border-radius: 50%;\n display: flex;\n align-items: center;\n justify-content: center;\n font-size: 1.5rem;\n }\n \n .testimonial-name {\n font-weight: 600;\n font-size: 1.125rem;\n }\n \n .testimonial-role {\n font-size: 0.875rem;\n opacity: 0.7;\n }\n \n .testimonial-nav {\n display: flex;\n align-items: center;\n justify-content: center;\n gap: 1.5rem;\n margin-top: 2rem;\n }\n \n .testimonial-nav-button {\n border: none;\n background: none;\n cursor: pointer;\n padding: 0.5rem;\n transition: opacity 0.2s;\n font-size: 1.5rem;\n }\n \n .testimonial-nav-button:hover {\n opacity: 0.7;\n }\n \n .testimonial-nav-button:disabled {\n opacity: 0.3;\n cursor: not-allowed;\n }\n `}</style>\n\n <div className=\"container\">\n {title && <h2 className=\"testimonial-title\">{title}</h2>}\n {subtitle && <p className=\"testimonial-subtitle\">{subtitle}</p>}\n\n <div className=\"testimonial-content\">\n <div className=\"testimonial-slide\" key={currentIndex}>\n {/* Quote Icon - site can style this */}\n <div className=\"testimonial-quote-icon\">&ldquo;</div>\n\n {/* Stars */}\n {showRating && (\n <div className=\"testimonial-stars\">\n {[...Array(5)].map((_, i) => (\n <span\n key={i}\n className={`star ${i < currentReview.rating ? 'star-filled' : 'star-empty'}`}\n aria-hidden=\"true\"\n >\n ★\n </span>\n ))}\n </div>\n )}\n\n {/* Quote Text */}\n <blockquote className=\"testimonial-quote\">\n &ldquo;{currentReview.quote}&rdquo;\n </blockquote>\n\n {/* Author */}\n <div className=\"testimonial-author\">\n <div className=\"testimonial-avatar\" aria-hidden=\"true\">\n {currentReview.image ? (\n <img src={currentReview.image} alt=\"\" />\n ) : (\n <span>👤</span>\n )}\n </div>\n <div className=\"testimonial-author-info\">\n <div className=\"testimonial-name\">{currentReview.name}</div>\n {currentReview.role && (\n <div className=\"testimonial-role\">{currentReview.role}</div>\n )}\n </div>\n </div>\n </div>\n </div>\n\n {/* Navigation */}\n {reviews.length > 1 && (\n <div className=\"testimonial-nav\">\n <button\n onClick={prevSlide}\n className=\"testimonial-nav-button testimonial-nav-prev\"\n aria-label=\"Previous review\"\n disabled={reviews.length <= 1}\n >\n ‹\n </button>\n\n <button\n onClick={nextSlide}\n className=\"testimonial-nav-button testimonial-nav-next\"\n aria-label=\"Next review\"\n disabled={reviews.length <= 1}\n >\n ›\n </button>\n </div>\n )}\n </div>\n </section>\n )\n}\n"]}
package/dist/index.js CHANGED
@@ -3,7 +3,7 @@
3
3
  var chunkW6K7MQCC_js = require('./chunk-W6K7MQCC.js');
4
4
  var chunkIFVMZFNT_js = require('./chunk-IFVMZFNT.js');
5
5
  require('./chunk-GCPPXUAI.js');
6
- var chunkKLIT3WTM_js = require('./chunk-KLIT3WTM.js');
6
+ var chunkEATGSR46_js = require('./chunk-EATGSR46.js');
7
7
  require('./chunk-JP5WGL75.js');
8
8
  var chunkTA2RGEE3_js = require('./chunk-TA2RGEE3.js');
9
9
  var chunkWETCVFIK_js = require('./chunk-WETCVFIK.js');
@@ -202,15 +202,15 @@ Object.defineProperty(exports, "uploadImage", {
202
202
  });
203
203
  Object.defineProperty(exports, "TestimonialSection", {
204
204
  enumerable: true,
205
- get: function () { return chunkKLIT3WTM_js.TestimonialSection; }
205
+ get: function () { return chunkEATGSR46_js.TestimonialSection; }
206
206
  });
207
207
  Object.defineProperty(exports, "fetchReviewStats", {
208
208
  enumerable: true,
209
- get: function () { return chunkKLIT3WTM_js.fetchReviewStats; }
209
+ get: function () { return chunkEATGSR46_js.fetchReviewStats; }
210
210
  });
211
211
  Object.defineProperty(exports, "fetchReviews", {
212
212
  enumerable: true,
213
- get: function () { return chunkKLIT3WTM_js.fetchReviews; }
213
+ get: function () { return chunkEATGSR46_js.fetchReviews; }
214
214
  });
215
215
  Object.defineProperty(exports, "CalendarView", {
216
216
  enumerable: true,
package/dist/index.mjs CHANGED
@@ -1,7 +1,7 @@
1
1
  export { clearRedirectCache, fetchRedirectRules, generateNextRedirects, handleManagedRedirects } from './chunk-7LAOWIWE.mjs';
2
2
  export { ManagedImage, assignImageToSlot, clearImageSlot, fetchManagedImage, fetchManagedImages, listImageFiles, uploadImage } from './chunk-3YQCPLGP.mjs';
3
3
  import './chunk-CU6OGBKK.mjs';
4
- export { TestimonialSection, fetchReviewStats, fetchReviews } from './chunk-BJ5IHVFZ.mjs';
4
+ export { TestimonialSection, fetchReviewStats, fetchReviews } from './chunk-UUDE46TZ.mjs';
5
5
  import './chunk-B22WSN4L.mjs';
6
6
  export { CalendarView, CheckoutForm, EventCalendar, EventEmbed, EventModal, EventTile, OfferingCard, OfferingList, ProductEmbed, RegistrationForm, UpcomingEvents, createCheckoutSession, fetchNextEvent, fetchOffering, fetchOfferings, fetchProducts, fetchServices, fetchUpcomingEvents, formatDate, formatDateTime, formatPrice, getOfferingUrl, registerForEvent, useEventModal } from './chunk-WET5VJRP.mjs';
7
7
  export { BookingWidget, createBooking, createSlotHold, detectTimezone, fetchAvailability, fetchAvailableDates, fetchBookingTypeDetails, fetchBookingTypes, formatDate as formatBookingDate, formatTime as formatBookingTime, formatDuration, releaseSlotHold } from './chunk-2OJCYWSQ.mjs';
@@ -45,6 +45,7 @@ declare function TestimonialSection({ title, subtitle, autoplay, autoplayInterva
45
45
  *
46
46
  * Fetch reviews and stats from Portal API public endpoint.
47
47
  * Uses API key authentication (x-api-key header) - project is resolved server-side.
48
+ * Includes retry with backoff for rate limiting (429).
48
49
  */
49
50
 
50
51
  interface FetchReviewsOptions {
@@ -45,6 +45,7 @@ declare function TestimonialSection({ title, subtitle, autoplay, autoplayInterva
45
45
  *
46
46
  * Fetch reviews and stats from Portal API public endpoint.
47
47
  * Uses API key authentication (x-api-key header) - project is resolved server-side.
48
+ * Includes retry with backoff for rate limiting (429).
48
49
  */
49
50
 
50
51
  interface FetchReviewsOptions {
@@ -1,21 +1,21 @@
1
1
  'use strict';
2
2
 
3
- var chunkKLIT3WTM_js = require('../chunk-KLIT3WTM.js');
3
+ var chunkEATGSR46_js = require('../chunk-EATGSR46.js');
4
4
  require('../chunk-ZSMWDLMK.js');
5
5
 
6
6
 
7
7
 
8
8
  Object.defineProperty(exports, "TestimonialSection", {
9
9
  enumerable: true,
10
- get: function () { return chunkKLIT3WTM_js.TestimonialSection; }
10
+ get: function () { return chunkEATGSR46_js.TestimonialSection; }
11
11
  });
12
12
  Object.defineProperty(exports, "fetchReviewStats", {
13
13
  enumerable: true,
14
- get: function () { return chunkKLIT3WTM_js.fetchReviewStats; }
14
+ get: function () { return chunkEATGSR46_js.fetchReviewStats; }
15
15
  });
16
16
  Object.defineProperty(exports, "fetchReviews", {
17
17
  enumerable: true,
18
- get: function () { return chunkKLIT3WTM_js.fetchReviews; }
18
+ get: function () { return chunkEATGSR46_js.fetchReviews; }
19
19
  });
20
20
  //# sourceMappingURL=index.js.map
21
21
  //# sourceMappingURL=index.js.map
@@ -1,4 +1,4 @@
1
- export { TestimonialSection, fetchReviewStats, fetchReviews } from '../chunk-BJ5IHVFZ.mjs';
1
+ export { TestimonialSection, fetchReviewStats, fetchReviews } from '../chunk-UUDE46TZ.mjs';
2
2
  import '../chunk-4XPGGLVP.mjs';
3
3
  //# sourceMappingURL=index.mjs.map
4
4
  //# sourceMappingURL=index.mjs.map
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@sonordev/site-kit",
3
- "version": "2.2.1",
3
+ "version": "2.2.2",
4
4
  "description": "Complete client-side integration kit for Sonor - SEO, Analytics, Engage, Forms, Blog",
5
5
  "license": "MIT",
6
6
  "main": "./dist/index.js",
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/reputation/api.ts","../src/reputation/TestimonialSection.tsx"],"names":[],"mappings":";;;;;;AAeA,SAAS,YAAA,GAAe;AACtB,EAAA,MAAM,MAAA,GACJ,OAAO,MAAA,KAAW,WAAA,GACb,OAAe,oBAAA,IAAwB,sBAAA,GACxC,OAAA,CAAQ,GAAA,CAAI,aAAA,IAAiB,sBAAA;AACnC,EAAA,MAAM,MAAA,GACJ,OAAO,MAAA,KAAW,WAAA,GACb,OAAe,oBAAA,IAAwB,EAAA,GACxC,OAAA,CAAQ,GAAA,CAAI,aAAA,IAAiB,EAAA;AACnC,EAAA,OAAO,EAAE,QAAQ,MAAA,EAAO;AAC1B;AAEA,eAAsB,YAAA,CACpB,OAAA,GAA+B,EAAC,EACb;AACnB,EAAA,MAAM,EAAE,MAAA,EAAQ,MAAA,EAAO,GAAI,YAAA,EAAa;AAExC,EAAA,IAAI,CAAC,MAAA,EAAQ;AACX,IAAA,OAAA,CAAQ,KAAK,oCAAoC,CAAA;AACjD,IAAA,OAAO,EAAC;AAAA,EACV;AAEA,EAAA,IAAI;AACF,IAAA,MAAM,MAAA,GAAS,IAAI,eAAA,EAAgB;AACnC,IAAA,IAAI,QAAQ,OAAA,EAAS,MAAA,CAAO,MAAA,CAAO,SAAA,EAAW,QAAQ,OAAO,CAAA;AAC7D,IAAA,IAAI,OAAA,CAAQ,OAAO,MAAA,CAAO,MAAA,CAAO,SAAS,OAAA,CAAQ,KAAA,CAAM,UAAU,CAAA;AAClE,IAAA,IAAI,OAAA,CAAQ,QAAA,EAAU,MAAA,CAAO,MAAA,CAAO,YAAY,MAAM,CAAA;AAEtD,IAAA,MAAM,EAAA,GAAK,OAAO,QAAA,EAAS,GAAI,IAAI,MAAA,CAAO,QAAA,EAAU,CAAA,CAAA,GAAK,EAAA;AACzD,IAAA,MAAM,WAAW,MAAM,KAAA,CAAM,GAAG,MAAM,CAAA,mBAAA,EAAsB,EAAE,CAAA,CAAA,EAAI;AAAA,MAChE,OAAA,EAAS,EAAE,WAAA,EAAa,MAAA;AAAO,KAChC,CAAA;AAED,IAAA,IAAI,CAAC,SAAS,EAAA,EAAI;AAChB,MAAA,OAAA,CAAQ,KAAA,CAAM,sCAAA,EAAwC,QAAA,CAAS,UAAU,CAAA;AACzE,MAAA,OAAO,EAAC;AAAA,IACV;AAEA,IAAA,MAAM,IAAA,GAAO,MAAM,QAAA,CAAS,IAAA,EAAK;AACjC,IAAA,OAAO,IAAA,CAAK,WAAW,EAAC;AAAA,EAC1B,SAAS,KAAA,EAAO;AACd,IAAA,OAAA,CAAQ,KAAA,CAAM,wCAAwC,KAAK,CAAA;AAC3D,IAAA,OAAO,EAAC;AAAA,EACV;AACF;AAEA,eAAsB,gBAAA,GAAgD;AACpE,EAAA,MAAM,EAAE,MAAA,EAAQ,MAAA,EAAO,GAAI,YAAA,EAAa;AAExC,EAAA,IAAI,CAAC,MAAA,EAAQ;AACX,IAAA,OAAA,CAAQ,KAAK,oCAAoC,CAAA;AACjD,IAAA,OAAO,IAAA;AAAA,EACT;AAEA,EAAA,IAAI;AACF,IAAA,MAAM,QAAA,GAAW,MAAM,KAAA,CAAM,CAAA,EAAG,MAAM,CAAA,yBAAA,CAAA,EAA6B;AAAA,MACjE,OAAA,EAAS,EAAE,WAAA,EAAa,MAAA;AAAO,KAChC,CAAA;AAED,IAAA,IAAI,CAAC,SAAS,EAAA,EAAI;AAChB,MAAA,OAAA,CAAQ,KAAA,CAAM,oCAAA,EAAsC,QAAA,CAAS,UAAU,CAAA;AACvE,MAAA,OAAO,IAAA;AAAA,IACT;AAEA,IAAA,OAAO,MAAM,SAAS,IAAA,EAAK;AAAA,EAC7B,SAAS,KAAA,EAAO;AACd,IAAA,OAAA,CAAQ,KAAA,CAAM,sCAAsC,KAAK,CAAA;AACzD,IAAA,OAAO,IAAA;AAAA,EACT;AACF;ACrEO,SAAS,kBAAA,CAAmB;AAAA,EACjC,KAAA;AAAA,EACA,QAAA;AAAA,EACA,QAAA,GAAW,IAAA;AAAA,EACX,gBAAA,GAAmB,GAAA;AAAA,EACnB,UAAA,GAAa,IAAA;AAAA,EACb,UAAA;AAAA,EACA,YAAA,GAAe,KAAA;AAAA,EACf,OAAA;AAAA,EACA,SAAA,GAAY;AACd,CAAA,EAA4B;AAC1B,EAAA,MAAM,CAAC,OAAA,EAAS,UAAU,CAAA,GAAI,QAAA,CAAmB,EAAE,CAAA;AACnD,EAAA,MAAM,CAAC,YAAA,EAAc,eAAe,CAAA,GAAI,SAAS,CAAC,CAAA;AAClD,EAAA,MAAM,CAAC,SAAA,EAAW,YAAY,CAAA,GAAI,SAAS,IAAI,CAAA;AAC/C,EAAA,MAAM,CAAC,SAAA,EAAW,YAAY,CAAA,GAAI,SAA0B,MAAM,CAAA;AAGlE,EAAA,SAAA,CAAU,MAAM;AACd,IAAA,eAAe,WAAA,GAAc;AAC3B,MAAA,YAAA,CAAa,IAAI,CAAA;AACjB,MAAA,MAAM,IAAA,GAAO,MAAM,YAAA,CAAa;AAAA,QAC9B,OAAA;AAAA,QACA,KAAA,EAAO,UAAA;AAAA,QACP,QAAA,EAAU;AAAA,OACX,CAAA;AACD,MAAA,UAAA,CAAW,IAAI,CAAA;AACf,MAAA,YAAA,CAAa,KAAK,CAAA;AAAA,IACpB;AACA,IAAA,WAAA,EAAY;AAAA,EACd,CAAA,EAAG,CAAC,OAAA,EAAS,UAAA,EAAY,YAAY,CAAC,CAAA;AAGtC,EAAA,SAAA,CAAU,MAAM;AACd,IAAA,IAAI,CAAC,QAAA,IAAY,OAAA,CAAQ,MAAA,IAAU,CAAA,EAAG;AAEtC,IAAA,MAAM,QAAA,GAAW,YAAY,MAAM;AACjC,MAAA,YAAA,CAAa,MAAM,CAAA;AACnB,MAAA,eAAA,CAAgB,CAAC,IAAA,KAAA,CAAU,IAAA,GAAO,CAAA,IAAK,QAAQ,MAAM,CAAA;AAAA,IACvD,GAAG,gBAAgB,CAAA;AAEnB,IAAA,OAAO,MAAM,cAAc,QAAQ,CAAA;AAAA,EACrC,GAAG,CAAC,QAAA,EAAU,gBAAA,EAAkB,OAAA,CAAQ,MAAM,CAAC,CAAA;AAE/C,EAAkB,WAAA,CAAY,CAAC,KAAA,KAAkB;AAC/C,IAAA,YAAA,CAAa,KAAA,GAAQ,YAAA,GAAe,MAAA,GAAS,MAAM,CAAA;AACnD,IAAA,eAAA,CAAgB,KAAK,CAAA;AAAA,EACvB,CAAA,EAAG,CAAC,YAAY,CAAC;AAEjB,EAAA,MAAM,SAAA,GAAY,YAAY,MAAM;AAClC,IAAA,YAAA,CAAa,MAAM,CAAA;AACnB,IAAA,eAAA,CAAgB,CAAC,IAAA,KAAA,CAAU,IAAA,GAAO,CAAA,IAAK,QAAQ,MAAM,CAAA;AAAA,EACvD,CAAA,EAAG,CAAC,OAAA,CAAQ,MAAM,CAAC,CAAA;AAEnB,EAAA,MAAM,SAAA,GAAY,YAAY,MAAM;AAClC,IAAA,YAAA,CAAa,MAAM,CAAA;AACnB,IAAA,eAAA,CAAgB,CAAC,IAAA,KAAA,CAAU,IAAA,GAAO,IAAI,OAAA,CAAQ,MAAA,IAAU,QAAQ,MAAM,CAAA;AAAA,EACxE,CAAA,EAAG,CAAC,OAAA,CAAQ,MAAM,CAAC,CAAA;AAEnB,EAAA,IAAI,SAAA,EAAW;AACb,IAAA,OAAO,IAAA;AAAA,EACT;AAEA,EAAA,IAAI,CAAC,QAAQ,MAAA,EAAQ;AACnB,IAAA,OAAO,IAAA;AAAA,EACT;AAEA,EAAA,MAAM,aAAA,GAAgB,QAAQ,YAAY,CAAA;AAE1C,EAAA,4BACG,SAAA,EAAA,EAAQ,SAAA,EAAW,uBAAuB,SAAS,CAAA,CAAA,EAAI,8BAA0B,IAAA,EAChF,QAAA,EAAA;AAAA,oBAAA,GAAA,CAAC,OAAA,EAAA,EAAO,QAAA,EAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,qBAAA,EAYS,SAAA,KAAc,MAAA,GAAS,cAAA,GAAiB,aAAa,CAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,MAAA,CAAA,EAmGpE,CAAA;AAAA,oBAEF,IAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,WAAA,EACZ,QAAA,EAAA;AAAA,MAAA,KAAA,oBAAS,GAAA,CAAC,IAAA,EAAA,EAAG,SAAA,EAAU,mBAAA,EAAqB,QAAA,EAAA,KAAA,EAAM,CAAA;AAAA,MAClD,QAAA,oBAAY,GAAA,CAAC,GAAA,EAAA,EAAE,SAAA,EAAU,wBAAwB,QAAA,EAAA,QAAA,EAAS,CAAA;AAAA,0BAE1D,KAAA,EAAA,EAAI,SAAA,EAAU,uBACb,QAAA,kBAAA,IAAA,CAAC,KAAA,EAAA,EAAI,WAAU,mBAAA,EAEb,QAAA,EAAA;AAAA,wBAAA,GAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,wBAAA,EAAyB,QAAA,EAAA,QAAA,EAAO,CAAA;AAAA,QAG9C,UAAA,oBACC,GAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,qBACZ,QAAA,EAAA,CAAC,GAAG,KAAA,CAAM,CAAC,CAAC,CAAA,CAAE,GAAA,CAAI,CAAC,GAAG,CAAA,qBACrB,GAAA;AAAA,UAAC,MAAA;AAAA,UAAA;AAAA,YAEC,WAAW,CAAA,KAAA,EAAQ,CAAA,GAAI,aAAA,CAAc,MAAA,GAAS,gBAAgB,YAAY,CAAA,CAAA;AAAA,YAC1E,aAAA,EAAY,MAAA;AAAA,YACb,QAAA,EAAA;AAAA,WAAA;AAAA,UAHM;AAAA,SAMR,CAAA,EACH,CAAA;AAAA,wBAIF,IAAA,CAAC,YAAA,EAAA,EAAW,SAAA,EAAU,mBAAA,EAAoB,QAAA,EAAA;AAAA,UAAA,QAAA;AAAA,UAChC,aAAA,CAAc,KAAA;AAAA,UAAM;AAAA,SAAA,EAC9B,CAAA;AAAA,wBAGA,IAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,oBAAA,EACb,QAAA,EAAA;AAAA,0BAAA,GAAA,CAAC,SAAI,SAAA,EAAU,oBAAA,EAAqB,eAAY,MAAA,EAC7C,QAAA,EAAA,aAAA,CAAc,wBACb,GAAA,CAAC,KAAA,EAAA,EAAI,GAAA,EAAK,aAAA,CAAc,OAAO,GAAA,EAAI,EAAA,EAAG,oBAEtC,GAAA,CAAC,MAAA,EAAA,EAAK,uBAAE,CAAA,EAEZ,CAAA;AAAA,0BACA,IAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,yBAAA,EACb,QAAA,EAAA;AAAA,4BAAA,GAAA,CAAC,KAAA,EAAA,EAAI,SAAA,EAAU,kBAAA,EAAoB,QAAA,EAAA,aAAA,CAAc,IAAA,EAAK,CAAA;AAAA,YACrD,cAAc,IAAA,oBACb,GAAA,CAAC,SAAI,SAAA,EAAU,kBAAA,EAAoB,wBAAc,IAAA,EAAK;AAAA,WAAA,EAE1D;AAAA,SAAA,EACF;AAAA,OAAA,EAAA,EAvCsC,YAwCxC,CAAA,EACF,CAAA;AAAA,MAGC,QAAQ,MAAA,GAAS,CAAA,oBAChB,IAAA,CAAC,KAAA,EAAA,EAAI,WAAU,iBAAA,EACb,QAAA,EAAA;AAAA,wBAAA,GAAA;AAAA,UAAC,QAAA;AAAA,UAAA;AAAA,YACC,OAAA,EAAS,SAAA;AAAA,YACT,SAAA,EAAU,6CAAA;AAAA,YACV,YAAA,EAAW,iBAAA;AAAA,YACX,QAAA,EAAU,QAAQ,MAAA,IAAU,CAAA;AAAA,YAC7B,QAAA,EAAA;AAAA;AAAA,SAED;AAAA,wBAEA,GAAA;AAAA,UAAC,QAAA;AAAA,UAAA;AAAA,YACC,OAAA,EAAS,SAAA;AAAA,YACT,SAAA,EAAU,6CAAA;AAAA,YACV,YAAA,EAAW,aAAA;AAAA,YACX,QAAA,EAAU,QAAQ,MAAA,IAAU,CAAA;AAAA,YAC7B,QAAA,EAAA;AAAA;AAAA;AAED,OAAA,EACF;AAAA,KAAA,EAEJ;AAAA,GAAA,EACF,CAAA;AAEJ","file":"chunk-BJ5IHVFZ.mjs","sourcesContent":["/**\n * @sonordev/site-kit/reputation - API Functions\n *\n * Fetch reviews and stats from Portal API public endpoint.\n * Uses API key authentication (x-api-key header) - project is resolved server-side.\n */\n\nimport type { Review, ReviewStats } from './types'\n\ninterface FetchReviewsOptions {\n service?: string\n limit?: number\n featured?: boolean\n}\n\nfunction getApiConfig() {\n const apiUrl =\n typeof window !== 'undefined'\n ? (window as any).__SITE_KIT_API_URL__ || 'https://api.sonor.io'\n : process.env.SONOR_API_URL || 'https://api.sonor.io'\n const apiKey =\n typeof window !== 'undefined'\n ? (window as any).__SITE_KIT_API_KEY__ || ''\n : process.env.SONOR_API_KEY || ''\n return { apiUrl, apiKey }\n}\n\nexport async function fetchReviews(\n options: FetchReviewsOptions = {},\n): Promise<Review[]> {\n const { apiUrl, apiKey } = getApiConfig()\n\n if (!apiKey) {\n console.warn('[Reputation] No API key configured')\n return []\n }\n\n try {\n const params = new URLSearchParams()\n if (options.service) params.append('service', options.service)\n if (options.limit) params.append('limit', options.limit.toString())\n if (options.featured) params.append('featured', 'true')\n\n const qs = params.toString() ? `?${params.toString()}` : ''\n const response = await fetch(`${apiUrl}/api/public/reviews${qs}`, {\n headers: { 'x-api-key': apiKey },\n })\n\n if (!response.ok) {\n console.error('[Reputation] Error fetching reviews:', response.statusText)\n return []\n }\n\n const data = await response.json()\n return data.reviews || []\n } catch (error) {\n console.error('[Reputation] Error fetching reviews:', error)\n return []\n }\n}\n\nexport async function fetchReviewStats(): Promise<ReviewStats | null> {\n const { apiUrl, apiKey } = getApiConfig()\n\n if (!apiKey) {\n console.warn('[Reputation] No API key configured')\n return null\n }\n\n try {\n const response = await fetch(`${apiUrl}/api/public/reviews/stats`, {\n headers: { 'x-api-key': apiKey },\n })\n\n if (!response.ok) {\n console.error('[Reputation] Error fetching stats:', response.statusText)\n return null\n }\n\n return await response.json()\n } catch (error) {\n console.error('[Reputation] Error fetching stats:', error)\n return null\n }\n}\n","/**\n * @sonordev/site-kit/reputation - Testimonial Section\n * \n * Displays client reviews in a rotating carousel\n * Fetches published reviews from Portal API public endpoint\n * \n * Minimal styling - let site CSS control appearance\n */\n\n'use client'\n\nimport React, { useEffect, useState, useCallback } from 'react'\nimport { fetchReviews } from './api'\nimport type { Review, TestimonialSectionProps } from './types'\n\nexport function TestimonialSection({\n title,\n subtitle,\n autoplay = true,\n autoplayInterval = 5000,\n showRating = true,\n maxReviews,\n featuredOnly = false,\n service,\n className = '',\n}: TestimonialSectionProps) {\n const [reviews, setReviews] = useState<Review[]>([])\n const [currentIndex, setCurrentIndex] = useState(0)\n const [isLoading, setIsLoading] = useState(true)\n const [direction, setDirection] = useState<'next' | 'prev'>('next')\n\n // Load reviews\n useEffect(() => {\n async function loadReviews() {\n setIsLoading(true)\n const data = await fetchReviews({\n service,\n limit: maxReviews,\n featured: featuredOnly,\n })\n setReviews(data)\n setIsLoading(false)\n }\n loadReviews()\n }, [service, maxReviews, featuredOnly])\n\n // Auto-rotate\n useEffect(() => {\n if (!autoplay || reviews.length <= 1) return\n\n const interval = setInterval(() => {\n setDirection('next')\n setCurrentIndex((prev) => (prev + 1) % reviews.length)\n }, autoplayInterval)\n\n return () => clearInterval(interval)\n }, [autoplay, autoplayInterval, reviews.length])\n\n const goToSlide = useCallback((index: number) => {\n setDirection(index > currentIndex ? 'next' : 'prev')\n setCurrentIndex(index)\n }, [currentIndex])\n\n const nextSlide = useCallback(() => {\n setDirection('next')\n setCurrentIndex((prev) => (prev + 1) % reviews.length)\n }, [reviews.length])\n\n const prevSlide = useCallback(() => {\n setDirection('prev')\n setCurrentIndex((prev) => (prev - 1 + reviews.length) % reviews.length)\n }, [reviews.length])\n\n if (isLoading) {\n return null // Let site handle loading state\n }\n\n if (!reviews.length) {\n return null\n }\n\n const currentReview = reviews[currentIndex]\n\n return (\n <section className={`testimonial-section ${className}`} data-site-kit-testimonials>\n <style>{`\n .testimonial-section {\n position: relative;\n padding: 4rem 1rem;\n }\n \n .testimonial-content {\n position: relative;\n overflow: hidden;\n }\n \n .testimonial-slide {\n animation: ${direction === 'next' ? 'slideInRight' : 'slideInLeft'} 0.6s cubic-bezier(0.4, 0, 0.2, 1);\n }\n \n @keyframes slideInRight {\n from {\n opacity: 0;\n transform: translateX(50px);\n }\n to {\n opacity: 1;\n transform: translateX(0);\n }\n }\n \n @keyframes slideInLeft {\n from {\n opacity: 0;\n transform: translateX(-50px);\n }\n to {\n opacity: 1;\n transform: translateX(0);\n }\n }\n \n .testimonial-quote-icon {\n font-size: 4rem;\n line-height: 1;\n text-align: center;\n margin-bottom: 1.5rem;\n opacity: 0.15;\n }\n \n .testimonial-stars {\n display: flex;\n justify-content: center;\n gap: 0.25rem;\n margin-bottom: 1.5rem;\n }\n \n .testimonial-quote {\n text-align: center;\n font-size: 1.25rem;\n line-height: 1.8;\n margin-bottom: 2rem;\n }\n \n .testimonial-author {\n display: flex;\n flex-direction: column;\n align-items: center;\n gap: 1rem;\n }\n \n .testimonial-avatar {\n width: 4rem;\n height: 4rem;\n border-radius: 50%;\n display: flex;\n align-items: center;\n justify-content: center;\n font-size: 1.5rem;\n }\n \n .testimonial-name {\n font-weight: 600;\n font-size: 1.125rem;\n }\n \n .testimonial-role {\n font-size: 0.875rem;\n opacity: 0.7;\n }\n \n .testimonial-nav {\n display: flex;\n align-items: center;\n justify-content: center;\n gap: 1.5rem;\n margin-top: 2rem;\n }\n \n .testimonial-nav-button {\n border: none;\n background: none;\n cursor: pointer;\n padding: 0.5rem;\n transition: opacity 0.2s;\n font-size: 1.5rem;\n }\n \n .testimonial-nav-button:hover {\n opacity: 0.7;\n }\n \n .testimonial-nav-button:disabled {\n opacity: 0.3;\n cursor: not-allowed;\n }\n `}</style>\n\n <div className=\"container\">\n {title && <h2 className=\"testimonial-title\">{title}</h2>}\n {subtitle && <p className=\"testimonial-subtitle\">{subtitle}</p>}\n\n <div className=\"testimonial-content\">\n <div className=\"testimonial-slide\" key={currentIndex}>\n {/* Quote Icon - site can style this */}\n <div className=\"testimonial-quote-icon\">&ldquo;</div>\n\n {/* Stars */}\n {showRating && (\n <div className=\"testimonial-stars\">\n {[...Array(5)].map((_, i) => (\n <span\n key={i}\n className={`star ${i < currentReview.rating ? 'star-filled' : 'star-empty'}`}\n aria-hidden=\"true\"\n >\n ★\n </span>\n ))}\n </div>\n )}\n\n {/* Quote Text */}\n <blockquote className=\"testimonial-quote\">\n &ldquo;{currentReview.quote}&rdquo;\n </blockquote>\n\n {/* Author */}\n <div className=\"testimonial-author\">\n <div className=\"testimonial-avatar\" aria-hidden=\"true\">\n {currentReview.image ? (\n <img src={currentReview.image} alt=\"\" />\n ) : (\n <span>👤</span>\n )}\n </div>\n <div className=\"testimonial-author-info\">\n <div className=\"testimonial-name\">{currentReview.name}</div>\n {currentReview.role && (\n <div className=\"testimonial-role\">{currentReview.role}</div>\n )}\n </div>\n </div>\n </div>\n </div>\n\n {/* Navigation */}\n {reviews.length > 1 && (\n <div className=\"testimonial-nav\">\n <button\n onClick={prevSlide}\n className=\"testimonial-nav-button testimonial-nav-prev\"\n aria-label=\"Previous review\"\n disabled={reviews.length <= 1}\n >\n ‹\n </button>\n\n <button\n onClick={nextSlide}\n className=\"testimonial-nav-button testimonial-nav-next\"\n aria-label=\"Next review\"\n disabled={reviews.length <= 1}\n >\n ›\n </button>\n </div>\n )}\n </div>\n </section>\n )\n}\n"]}
@@ -1 +0,0 @@
1
- {"version":3,"sources":["../src/reputation/api.ts","../src/reputation/TestimonialSection.tsx"],"names":["useState","useEffect","useCallback","jsx","jsxs"],"mappings":"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","file":"chunk-KLIT3WTM.js","sourcesContent":["/**\n * @sonordev/site-kit/reputation - API Functions\n *\n * Fetch reviews and stats from Portal API public endpoint.\n * Uses API key authentication (x-api-key header) - project is resolved server-side.\n */\n\nimport type { Review, ReviewStats } from './types'\n\ninterface FetchReviewsOptions {\n service?: string\n limit?: number\n featured?: boolean\n}\n\nfunction getApiConfig() {\n const apiUrl =\n typeof window !== 'undefined'\n ? (window as any).__SITE_KIT_API_URL__ || 'https://api.sonor.io'\n : process.env.SONOR_API_URL || 'https://api.sonor.io'\n const apiKey =\n typeof window !== 'undefined'\n ? (window as any).__SITE_KIT_API_KEY__ || ''\n : process.env.SONOR_API_KEY || ''\n return { apiUrl, apiKey }\n}\n\nexport async function fetchReviews(\n options: FetchReviewsOptions = {},\n): Promise<Review[]> {\n const { apiUrl, apiKey } = getApiConfig()\n\n if (!apiKey) {\n console.warn('[Reputation] No API key configured')\n return []\n }\n\n try {\n const params = new URLSearchParams()\n if (options.service) params.append('service', options.service)\n if (options.limit) params.append('limit', options.limit.toString())\n if (options.featured) params.append('featured', 'true')\n\n const qs = params.toString() ? `?${params.toString()}` : ''\n const response = await fetch(`${apiUrl}/api/public/reviews${qs}`, {\n headers: { 'x-api-key': apiKey },\n })\n\n if (!response.ok) {\n console.error('[Reputation] Error fetching reviews:', response.statusText)\n return []\n }\n\n const data = await response.json()\n return data.reviews || []\n } catch (error) {\n console.error('[Reputation] Error fetching reviews:', error)\n return []\n }\n}\n\nexport async function fetchReviewStats(): Promise<ReviewStats | null> {\n const { apiUrl, apiKey } = getApiConfig()\n\n if (!apiKey) {\n console.warn('[Reputation] No API key configured')\n return null\n }\n\n try {\n const response = await fetch(`${apiUrl}/api/public/reviews/stats`, {\n headers: { 'x-api-key': apiKey },\n })\n\n if (!response.ok) {\n console.error('[Reputation] Error fetching stats:', response.statusText)\n return null\n }\n\n return await response.json()\n } catch (error) {\n console.error('[Reputation] Error fetching stats:', error)\n return null\n }\n}\n","/**\n * @sonordev/site-kit/reputation - Testimonial Section\n * \n * Displays client reviews in a rotating carousel\n * Fetches published reviews from Portal API public endpoint\n * \n * Minimal styling - let site CSS control appearance\n */\n\n'use client'\n\nimport React, { useEffect, useState, useCallback } from 'react'\nimport { fetchReviews } from './api'\nimport type { Review, TestimonialSectionProps } from './types'\n\nexport function TestimonialSection({\n title,\n subtitle,\n autoplay = true,\n autoplayInterval = 5000,\n showRating = true,\n maxReviews,\n featuredOnly = false,\n service,\n className = '',\n}: TestimonialSectionProps) {\n const [reviews, setReviews] = useState<Review[]>([])\n const [currentIndex, setCurrentIndex] = useState(0)\n const [isLoading, setIsLoading] = useState(true)\n const [direction, setDirection] = useState<'next' | 'prev'>('next')\n\n // Load reviews\n useEffect(() => {\n async function loadReviews() {\n setIsLoading(true)\n const data = await fetchReviews({\n service,\n limit: maxReviews,\n featured: featuredOnly,\n })\n setReviews(data)\n setIsLoading(false)\n }\n loadReviews()\n }, [service, maxReviews, featuredOnly])\n\n // Auto-rotate\n useEffect(() => {\n if (!autoplay || reviews.length <= 1) return\n\n const interval = setInterval(() => {\n setDirection('next')\n setCurrentIndex((prev) => (prev + 1) % reviews.length)\n }, autoplayInterval)\n\n return () => clearInterval(interval)\n }, [autoplay, autoplayInterval, reviews.length])\n\n const goToSlide = useCallback((index: number) => {\n setDirection(index > currentIndex ? 'next' : 'prev')\n setCurrentIndex(index)\n }, [currentIndex])\n\n const nextSlide = useCallback(() => {\n setDirection('next')\n setCurrentIndex((prev) => (prev + 1) % reviews.length)\n }, [reviews.length])\n\n const prevSlide = useCallback(() => {\n setDirection('prev')\n setCurrentIndex((prev) => (prev - 1 + reviews.length) % reviews.length)\n }, [reviews.length])\n\n if (isLoading) {\n return null // Let site handle loading state\n }\n\n if (!reviews.length) {\n return null\n }\n\n const currentReview = reviews[currentIndex]\n\n return (\n <section className={`testimonial-section ${className}`} data-site-kit-testimonials>\n <style>{`\n .testimonial-section {\n position: relative;\n padding: 4rem 1rem;\n }\n \n .testimonial-content {\n position: relative;\n overflow: hidden;\n }\n \n .testimonial-slide {\n animation: ${direction === 'next' ? 'slideInRight' : 'slideInLeft'} 0.6s cubic-bezier(0.4, 0, 0.2, 1);\n }\n \n @keyframes slideInRight {\n from {\n opacity: 0;\n transform: translateX(50px);\n }\n to {\n opacity: 1;\n transform: translateX(0);\n }\n }\n \n @keyframes slideInLeft {\n from {\n opacity: 0;\n transform: translateX(-50px);\n }\n to {\n opacity: 1;\n transform: translateX(0);\n }\n }\n \n .testimonial-quote-icon {\n font-size: 4rem;\n line-height: 1;\n text-align: center;\n margin-bottom: 1.5rem;\n opacity: 0.15;\n }\n \n .testimonial-stars {\n display: flex;\n justify-content: center;\n gap: 0.25rem;\n margin-bottom: 1.5rem;\n }\n \n .testimonial-quote {\n text-align: center;\n font-size: 1.25rem;\n line-height: 1.8;\n margin-bottom: 2rem;\n }\n \n .testimonial-author {\n display: flex;\n flex-direction: column;\n align-items: center;\n gap: 1rem;\n }\n \n .testimonial-avatar {\n width: 4rem;\n height: 4rem;\n border-radius: 50%;\n display: flex;\n align-items: center;\n justify-content: center;\n font-size: 1.5rem;\n }\n \n .testimonial-name {\n font-weight: 600;\n font-size: 1.125rem;\n }\n \n .testimonial-role {\n font-size: 0.875rem;\n opacity: 0.7;\n }\n \n .testimonial-nav {\n display: flex;\n align-items: center;\n justify-content: center;\n gap: 1.5rem;\n margin-top: 2rem;\n }\n \n .testimonial-nav-button {\n border: none;\n background: none;\n cursor: pointer;\n padding: 0.5rem;\n transition: opacity 0.2s;\n font-size: 1.5rem;\n }\n \n .testimonial-nav-button:hover {\n opacity: 0.7;\n }\n \n .testimonial-nav-button:disabled {\n opacity: 0.3;\n cursor: not-allowed;\n }\n `}</style>\n\n <div className=\"container\">\n {title && <h2 className=\"testimonial-title\">{title}</h2>}\n {subtitle && <p className=\"testimonial-subtitle\">{subtitle}</p>}\n\n <div className=\"testimonial-content\">\n <div className=\"testimonial-slide\" key={currentIndex}>\n {/* Quote Icon - site can style this */}\n <div className=\"testimonial-quote-icon\">&ldquo;</div>\n\n {/* Stars */}\n {showRating && (\n <div className=\"testimonial-stars\">\n {[...Array(5)].map((_, i) => (\n <span\n key={i}\n className={`star ${i < currentReview.rating ? 'star-filled' : 'star-empty'}`}\n aria-hidden=\"true\"\n >\n ★\n </span>\n ))}\n </div>\n )}\n\n {/* Quote Text */}\n <blockquote className=\"testimonial-quote\">\n &ldquo;{currentReview.quote}&rdquo;\n </blockquote>\n\n {/* Author */}\n <div className=\"testimonial-author\">\n <div className=\"testimonial-avatar\" aria-hidden=\"true\">\n {currentReview.image ? (\n <img src={currentReview.image} alt=\"\" />\n ) : (\n <span>👤</span>\n )}\n </div>\n <div className=\"testimonial-author-info\">\n <div className=\"testimonial-name\">{currentReview.name}</div>\n {currentReview.role && (\n <div className=\"testimonial-role\">{currentReview.role}</div>\n )}\n </div>\n </div>\n </div>\n </div>\n\n {/* Navigation */}\n {reviews.length > 1 && (\n <div className=\"testimonial-nav\">\n <button\n onClick={prevSlide}\n className=\"testimonial-nav-button testimonial-nav-prev\"\n aria-label=\"Previous review\"\n disabled={reviews.length <= 1}\n >\n ‹\n </button>\n\n <button\n onClick={nextSlide}\n className=\"testimonial-nav-button testimonial-nav-next\"\n aria-label=\"Next review\"\n disabled={reviews.length <= 1}\n >\n ›\n </button>\n </div>\n )}\n </div>\n </section>\n )\n}\n"]}