This file teaches AI coding agents (Claude Code, Cursor, Copilot, Codex, Gemini CLI, and any MCP-compatible client) how to use Roxy UI when integrating RoxyAPI into a project.
This file ships inside both @roxyapi/ui and @roxyapi/ui-react on npm. After install, read it at node_modules/@roxyapi/ui/AGENTS.md.
Live preview: https://roxyapi.github.io/ui/. Source of truth for component types: the combined OpenAPI spec at https://roxyapi.com/api/v2/openapi.json, regenerated into packages/ui/src/types/types.gen.ts. Per-product specs live at https://roxyapi.com/api/v2/{slug}/openapi.json.
Roxy UI is the official web component library for the RoxyAPI catalog. Components and helpers cover Western astrology, Vedic astrology, numerology, tarot, Human Design, forecast, biorhythm, I Ching, crystals, dreams, angel numbers, with the location helper for geocoding. New endpoints regenerate component types automatically.
Map the natural-language request to a component first; fall back to the table below if the request names a specific endpoint.
| If the user says... | Render |
|---|---|
"daily horoscope for {sign}", "weekly horoscope", "monthly horoscope" |
<roxy-horoscope-card> |
| "birth chart", "natal chart", "Western chart", "show me my planets" | <roxy-natal-chart> |
| "match two birth charts", "compare us in Western astrology", "synastry" | <roxy-synastry-chart> |
| "kundli", "Vedic chart", "rashi chart", "South/North Indian chart" | <roxy-vedic-kundli> |
| "D9", "navamsa", "varga chart", "divisional chart", "D10 dasamsa", "D60 shashtiamsa" | <roxy-divisional-chart> (request body needs division: integer, supported 2,3,4,7,9,10,12,16,20,24,27,30,40,45,60) |
| "kundli matching", "Guna Milan", "match for marriage", "36-point compatibility" | <roxy-guna-milan> |
| "are we compatible", "compatibility score", "love score" (cross-domain) | <roxy-compatibility-card> |
| "panchang for today", "tithi", "nakshatra", "muhurta", "auspicious times" | <roxy-panchang-table> |
| "dasha", "mahadasha", "current planetary period", "Vimshottari" | <roxy-dasha-timeline> |
| "manglik", "kalsarpa", "sadhesati", "any doshas in my chart" | <roxy-dosha-card> |
| "KP planets", "sub-lord", "Krishnamurti" | <roxy-kp-planets-table> |
| "life path number", "expression number", "personal year", "numerology chart" | <roxy-numerology-card> |
| "draw a tarot card", "card of the day", "card meaning" | <roxy-tarot-card> |
| "tarot reading", "three-card spread", "Celtic Cross", "yes or no tarot" | <roxy-tarot-spread> |
| "Human Design chart", "bodygraph", "my type and authority", "defined centers", "channels and gates" | <roxy-bodygraph> |
| "forecast", "what is coming up", "upcoming transits and events", "timeline of my year" | <roxy-forecast-timeline> |
| "biorhythm", "physical/emotional/intellectual cycle", "critical days" | <roxy-biorhythm-chart> |
| "I Ching", "hexagram", "cast the coins", "Book of Changes" | <roxy-hexagram> |
| "moon phase", "moon calendar", "next full moon", "current moon" | <roxy-moon-phase> |
| "what does my dream mean", "dream symbol", "dream dictionary", "I dreamt of {symbol}" | <roxy-dream-card> |
| "angel number {n}", "meaning of 111 / 222 / 1111", "I keep seeing this number" | <roxy-angel-number-card> |
| "what does {any number} mean", "analyze this number", "is 1234 an angel number" | <roxy-angel-number-lookup> |
| "crystals for {chakra}", "healing stones", "birthstone for {month}", "crystals for {sign}" | <roxy-crystal-grid> |
| "search a city", "geocode", "lat/long for a place" | <roxy-location-search> |
| "build a form for endpoint X" | <roxy-endpoint-form> |
Fallback rule. If the response shape does not match any component above, render with <roxy-data>. It accepts any RoxyAPI response and produces a structured layout from the JSON.
Use the table below for the formal endpoint to component mapping.
| Element | Domain | Endpoint(s) | What it renders |
|---|---|---|---|
<roxy-natal-chart> |
Western | POST /astrology/natal-chart | Natal chart wheel with planet glyphs and aspect lines |
<roxy-synastry-chart> |
Western | POST /astrology/synastry | Dual-wheel synastry with inter-aspects table |
<roxy-western-planets-table> |
Western | POST /astrology/natal-chart | Sign, degree, house, motion columns plus ASC, MC, PoF, Vertex |
<roxy-transits-table> |
Western | POST /astrology/transits | Transit planet positions plus optional aspects to a natal chart |
<roxy-aspects-table> |
Western | POST /astrology/aspects, /astrology/transit-aspects, /astrology/aspect-patterns | Aspect rows coloured by nature with orb and strength, plus detected chart patterns |
<roxy-moon-phase> |
Western | GET /astrology/moon-phase/{current,upcoming,calendar/...} | Moon phase card and calendar |
<roxy-horoscope-card> |
Western | GET /astrology/horoscope/{sign}/{daily,weekly,monthly} | Daily, weekly, or monthly horoscope card |
<roxy-compatibility-card> |
Cross | POST /astrology/compatibility-score, /numerology/compatibility, /biorhythm/compatibility | Score card with category breakdown |
<roxy-vedic-kundli> |
Vedic | POST /vedic-astrology/birth-chart | South, North, or East Indian kundli with degree detail and optional Chandra Lagna view |
<roxy-divisional-chart> |
Vedic | POST /vedic-astrology/divisional-chart | Generic divisional varga wheel from D2 Hora to D60 Shashtiamsa |
<roxy-kp-chart> |
Vedic (KP) | POST /vedic-astrology/kp/chart | Ascendant, cusps, and planets with KP stellar hierarchy |
<roxy-vedic-planets-table> |
Vedic | POST /vedic-astrology/birth-chart | Degree, nakshatra, pada, lord, bhava, avastha columns |
<roxy-kp-planets-table> |
Vedic (KP) | POST /vedic-astrology/kp/planets | Sub-lord and sub-sub-lord columns |
<roxy-kp-ruling-planets> |
Vedic (KP) | POST /vedic-astrology/kp/ruling-planets | Day lord, Moon/Lagna hierarchies, ruling planets, significators |
<roxy-ashtakavarga-grid> |
Vedic | POST /vedic-astrology/ashtakavarga | Sarva, Bhinna, and Shodhya Pinda views in a tabbed heatmap |
<roxy-shadbala-table> |
Vedic | POST /vedic-astrology/shadbala | Six-fold planetary strength bar plus rupas and adequacy badge |
<roxy-dasha-timeline> |
Vedic | POST /vedic-astrology/dasha/{current,major,sub/...} | Vimshottari mahadasha + antardasha + pratyantardasha |
<roxy-guna-milan> |
Vedic | POST /vedic-astrology/compatibility | 36-point Ashtakoota with eight sub-scores |
<roxy-panchang-table> |
Vedic | POST /vedic-astrology/panchang/{basic,detailed} | 15+ muhurtas in detailed mode |
<roxy-vedic-aspects> |
Vedic | POST /vedic-astrology/aspects | Graha drishti rows with aspect type, strength, and orb, plus mutual aspects |
<roxy-hora-table> |
Vedic | POST /vedic-astrology/panchang/hora | Day and night planetary hours with ruling planet and window |
<roxy-choghadiya-grid> |
Vedic | POST /vedic-astrology/panchang/choghadiya | Day and night Choghadiya muhurta tiles colored by effect |
<roxy-yoga-list> |
Vedic | GET /vedic-astrology/yoga, /yoga/{id} | Filterable yoga cards from the 300 plus yoga catalog |
<roxy-nakshatra-card> |
Vedic | GET /vedic-astrology/nakshatras/{id} | Lord, deity, symbol, characteristics, remedies |
<roxy-dosha-card> |
Vedic | POST /vedic-astrology/dosha/{manglik,kalsarpa,sadhesati} | Presence, severity, remedies, scoped effects |
<roxy-numerology-card> |
Numerology | POST /numerology/{life-path,expression,soul-urge,personality,birth-day,maturity,daily,personal-day,personal-month,personal-year,chart} | Life path, expression, soul urge, personality, personal year, full chart |
<roxy-tarot-card> |
Tarot | GET /tarot/cards/{id}, POST /tarot/daily | Single card with upright and reversed flip |
<roxy-tarot-catalog> |
Tarot | GET /tarot/cards | Deck gallery tiles with card art, name, and arcana and suit |
<roxy-tarot-spread> |
Tarot | POST /tarot/spreads/{three-card,celtic-cross,love}, /tarot/yes-no, /tarot/draw | Spreads with positions and reading |
<roxy-bodygraph> |
Human Design | POST /human-design/bodygraph | Nine-center chart with defined and open centers, active channels, gates, and a type and authority summary |
<roxy-hd-connection> |
Human Design | POST /human-design/connection | Electromagnetic, compromise, and dominance channels between two charts |
<roxy-hd-penta> |
Human Design | POST /human-design/penta | Group penta channels split into upper and lower triangles |
<roxy-hd-variables> |
Human Design | POST /human-design/variables | The four transformation arrows with direction and PHS labels |
<roxy-forecast-timeline> |
Forecast | POST /forecast/timeline | Date-grouped events across Western, Vedic, and biorhythm domains, weighted by significance |
<roxy-forecast-digest> |
Forecast | POST /forecast/digest | Per-window event counts, domain breakdown, and the highest-significance events |
<roxy-biorhythm-chart> |
Biorhythm | POST /biorhythm/{daily,forecast,critical-days} | Daily bars, forecast cycle lines, critical days |
<roxy-hexagram> |
I Ching | GET /iching/hexagrams/{number}, /iching/cast, POST /iching/daily, /iching/daily/cast | Hexagram with trigrams, judgment, image, changing lines |
<roxy-crystal-card> |
Crystals | GET /crystals/{id} | Photo, meaning sections, chakra, zodiac, element, hardness, keywords, and pairings |
<roxy-crystal-grid> |
Crystals | GET /crystals, /crystals/chakra/{chakra}, /crystals/element/{element}, /crystals/zodiac/{sign}, /crystals/birthstone/{month}, /crystals/search | Crystal gallery tiles with photo, name, and colour swatches |
<roxy-dream-card> |
Dreams | GET /dreams/symbols/{id} | Symbol name, interpretation body, and letter chip |
<roxy-dream-search> |
Dreams | GET /dreams/symbols | Matched dream symbols as selectable tiles with a letter chip |
<roxy-angel-number-card> |
Angel Numbers | GET /angel-numbers/numbers/{number} | Number meaning with spiritual, love, career, money, twin flame, biblical, and shadow sections |
<roxy-angel-number-lookup> |
Angel Numbers | GET /angel-numbers/lookup | Pattern analysis plus known meaning and digit-root fallback |
<roxy-reference-card> |
Reference | GET /astrology/{signs,planet-meanings}/{id}, /vedic-astrology/rashis/{id}, /iching/trigrams/{id}, /human-design/{gates,centers}/{id}, /numerology/{meanings,compound-number}/{number} | Symbol, name, description, keyword chips, and an attribute grid for any glossary lookup |
<roxy-endpoint-form> |
Helper | Any endpoint, from the spec | Schema-driven form, emits roxy-submit |
<roxy-location-search> |
Helper | GET /location/search | Debounced city search input, emits roxy-location-select |
<roxy-data> |
Helper | Any response shape | Generic fallback renderer for unknown shapes |
These are the bugs that come up over and over. Read this section before writing the first line of integration code.
The @roxyapi/sdk returns { data, error, request, response }. Always destructure data before passing to a component. Passing the full envelope renders [object Object]. This is the single most common integration bug.
// Wrong: passes the envelope
const response = await roxy.astrology.generateNatalChart({ body });
element.data = response; // → renders [object Object]
// Right: unwrap data
const { data } = await roxy.astrology.generateNatalChart({ body });
element.data = data;Every snippet below follows this rule.
Every chart endpoint (Western, Vedic, KP, synastry, transits, dasha, dosha, panchang) needs latitude, longitude, and timezone. Never ask the user to type coordinates. Call /location/search first, then feed the result into the chart endpoint.
// Right
const { data: cities } = await roxy.location.searchCities({ query: { q: 'Mumbai' } });
const { latitude, longitude, timezone } = cities.cities[0];
const { data: chart } = await roxy.astrology.generateNatalChart({
body: { date, time, latitude, longitude, timezone },
});Every chart endpoint accepts timezone as either a decimal-hour offset (5.5 for IST, -5 for EST) or an IANA name ('Asia/Kolkata', 'America/New_York'). The decimal form is what /location/search returns; the IANA form is correct over DST boundaries. Pick one and stay consistent in a single integration. Mixing them does not break the API but makes the bug surface area larger.
Secret keys (sk_*) grant full account access and are server side only. Call createRoxy(process.env.ROXY_API_KEY!) on your server (Node, Bun, Hono, Next.js route handlers, Workers, Edge functions), then send the response, not the key, to the component. Never ship a secret key in a client bundle.
// Secret key: server side only
const roxy = createRoxy(process.env.ROXY_API_KEY!);For direct client-side calls, use a publishable key (pk_live_* / pk_test_*) instead. Publishable keys are browser-safe: mint one at roxyapi.com/account, register the origins you embed on, and the API gateway returns 403 for any other origin. See the client-side pattern below.
The React components in @roxyapi/ui-react mount Custom Elements, which need the DOM. In the App Router, files that import them must declare 'use client' at the top. Server Components can fetch with the SDK; the client component renders.
// app/chart-view.tsx
'use client';
import { RoxyNatalChart } from '@roxyapi/ui-react';
export default function ChartView({ data }) {
return <RoxyNatalChart data={data} />;
}React 19 routes hyphenated DOM events through camelCase props correctly. React 17 and 18 do not. On 17/18, attach the listener with a ref:
const ref = useRef<HTMLElement>(null);
useEffect(() => {
const el = ref.current;
if (!el) return;
const handler = (e: Event) => setData((e as CustomEvent).detail);
el.addEventListener('roxy-location-select', handler);
return () => el.removeEventListener('roxy-location-select', handler);
}, []);
return <roxy-location-search ref={ref} />;The React 19 path is <RoxyLocationSearch onRoxyLocationSelect={handler} />.
Several components select a view, mode, or chart layout in addition to data. The React components type these as literal-union props alongside data, so editors autocomplete the allowed values and the build flags a typo. Set them as camelCase props.
<RoxyVedicKundli data={chart} chartStyle="south" />
<RoxyDoshaCard data={kalsarpa} type="kalsarpa" />
<RoxyHoroscopeCard data={weekly} period="weekly" />
<RoxyPanchangTable data={panchang} detail="detailed" />The full set: RoxyNatalChart houseSystem, RoxyHoroscopeCard period, RoxyMoonPhase mode, RoxyCompatibilityCard mode, RoxyVedicKundli and RoxyDivisionalChart chartStyle, RoxyPanchangTable detail, RoxyDashaTimeline period, RoxyDoshaCard type, RoxyNumerologyCard type, RoxyTarotSpread spread, RoxyBiorhythmChart mode, RoxyHexagram mode. Outside React, set the same value as a kebab-case attribute or a JS property on the element (for example chart-style="south" or el.chartStyle = 'south').
Do not declare interface XyzData { ... } for a RoxyAPI response. Import the spec-derived type from @roxyapi/sdk (or let the SDK return type flow through inference). Local interfaces drift the moment the spec changes; the component will keep compiling while rendering nothing.
// Wrong
interface NatalChart { planets: ...; houses: ...; }
// Right
import type { NatalChartResponse } from '@roxyapi/sdk';Fetch on your server with the secret key, then inline the response into the component as a child <script type="application/json" class="roxy-data">. The component reads it on load. No key in the browser.
<script
src="https://cdn.jsdelivr.net/npm/@roxyapi/ui@latest/dist/cdn/roxy-ui.js"
crossorigin="anonymous"
></script>
<roxy-natal-chart>
<script type="application/json" class="roxy-data">
{ "planets": [ ... ], "houses": [ ... ], "aspects": [ ... ] }
</script>
</roxy-natal-chart>Setting the JavaScript data property always wins over the inlined JSON, so the same element also drives dynamic pages.
<RoxyLocationSearch> runs in the browser. On select, call your own route, which holds the secret key, and set the returned data on the chart. The key never reaches the client.
'use client';
import {
RoxyNatalChart,
RoxyLocationSearch,
type RoxyNatalChartProps,
} from '@roxyapi/ui-react';
import { useState } from 'react';
export function BirthChartView() {
const [chart, setChart] = useState<RoxyNatalChartProps['data']>(undefined);
const onLocationSelect = async (e: CustomEvent<{ latitude?: number; longitude?: number; timezone?: number | string }>) => {
const { latitude, longitude, timezone } = e.detail;
if (latitude == null || longitude == null) return;
// Your route calls roxy.astrology.generateNatalChart with the secret key.
const res = await fetch('/api/natal-chart', {
method: 'POST',
body: JSON.stringify({ date: '1990-01-15', time: '14:30:00', latitude, longitude, timezone }),
});
setChart(await res.json());
};
return (
<div>
<RoxyLocationSearch onRoxyLocationSelect={onLocationSelect} />
{chart && <RoxyNatalChart data={chart} />}
</div>
);
}For a static chart with no picker, fetch in a Server Component and pass data to a client component (Pattern 6).
<roxy-endpoint-form> reads the OpenAPI spec and renders the inputs for any endpoint. On roxy-submit, POST the validated values to your own route, which calls the SDK with the secret key, then set the returned data on the target component.
<roxy-endpoint-form
data-endpoint="vedic-astrology/birth-chart"
method="POST"
submit-label="Generate kundli"
></roxy-endpoint-form>
<roxy-vedic-kundli chart-style="south"></roxy-vedic-kundli>
<script type="module">
const form = document.querySelector('roxy-endpoint-form');
form.addEventListener('roxy-submit', async (e) => {
// Your route calls roxy.vedicAstrology.generateBirthChart with the secret key.
const res = await fetch('/api/kundli', { method: 'POST', body: JSON.stringify(e.detail.values) });
document.querySelector('roxy-vedic-kundli').data = await res.json();
});
</script>When you do not want a backend at all, mint a publishable key (pk_live_* / pk_test_*) at roxyapi.com/account, register the origins you embed on, and let the component fetch itself. Every rendering component is self-fetching: give it a data-endpoint and a publishable-key and it renders its own input form, calls RoxyAPI on submit, and displays the result. No script, no separate location wiring, no envelope handling.
<roxy-natal-chart
data-endpoint="astrology/natal-chart"
publishable-key="pk_live_..."
></roxy-natal-chart>That single element shows a schema-driven form (city search included for endpoints that need coordinates), fetches on submit, shows a loading then error-or-result state, and re-shows the form so the user can try another query. method defaults to POST; set method="GET" for GET endpoints. Set data-endpoint to the spec path without the leading slash (dreams/symbols/{id}, astrology/horoscope/{sign}/daily).
Key handling is the contract. The component enforces it, not you:
- The publishable key is safe in client code: it is origin-restricted (any other origin gets 403) and cannot read your account.
- A secret key never works here. If
publishable-keyis not apk_key the component refuses to fetch, sends nothing, and emits aroxy-validation-errorevent. A secret key cannot leak through self-fetch even by mistake. - For production with a backend, prefer controlled mode (Patterns 1, 6, 7): the server fetches with the
sk_key and injects the response, so no key of any kind reaches the browser.
In React, the same props are typed: <RoxyNatalChart endpoint="astrology/natal-chart" publishableKey={process.env.NEXT_PUBLIC_ROXY_PK} />.
A remote MCP server at roxyapi.com/mcp/{domain} exposes each RoxyAPI endpoint as an MCP tool. The JSON returned by the tool call has the same shape as the SDK response. Pass it straight into the matching component.
// Pseudocode for any MCP-aware agent
const result = await mcp.call('roxyapi.astrology.generate_natal_chart', {
date: '1990-01-15', time: '14:30:00', latitude: 19.07, longitude: 72.88, timezone: 5.5,
});
document.querySelector('roxy-natal-chart').data = result;No field renames. No glue code. Use the decision tree above to pick the component for any tool.
Server fetches with the secret key, client renders with the React component. The API key never crosses the network.
// app/page.tsx (Server Component)
import { createRoxy } from '@roxyapi/sdk';
import BirthChartView from './birth-chart-view';
const roxy = createRoxy(process.env.ROXY_API_KEY!);
export default async function Page() {
const { data } = await roxy.vedicAstrology.generateBirthChart({
body: { date: '1990-01-15', time: '14:30:00', latitude: 19.07, longitude: 72.88, timezone: 5.5 },
});
return <BirthChartView data={data} />;
}// app/birth-chart-view.tsx (Client Component)
'use client';
import { RoxyVedicKundli } from '@roxyapi/ui-react';
export default function BirthChartView({ data }: { data: unknown }) {
return <RoxyVedicKundli data={data} />;
}When the page is rendered on the server or served from cache, there may be no JavaScript to set the data property per element. Render the response into a child <script type="application/json" class="roxy-data"> instead. The component reads the embedded JSON on load. No per-element script, no API key in the browser.
Always serialize with the shipped helper. Never hand-roll the escape and never use a bare JSON.stringify. @roxyapi/ui exports roxyDataScript(data) (returns the full <script class="roxy-data">…</script> element) and serializeRoxyData(data) (returns just the escaped JSON string). They escape <, >, and & so a string field containing </script> cannot break out of the block. A raw JSON.stringify of a response with interpretation prose can contain </script> and corrupt the page or open an injection hole.
import { roxyDataScript } from '@roxyapi/ui';
const { data } = await roxy.astrology.generateNatalChart({ body });
const html = `<roxy-natal-chart>${roxyDataScript(data)}</roxy-natal-chart>`;The emitted markup:
<roxy-natal-chart>
<script type="application/json" class="roxy-data">{ "planets": [ ], "houses": [ ], "aspects": [ ] }</script>
</roxy-natal-chart>Rules for this pattern:
- The JSON must be the unwrapped RoxyAPI response, the same shape you would assign to
element.data. Do not embed the SDK envelope ({ data, error, request, response }); embeddata. - The script must be a direct child of the component and carry both
type="application/json"andclass="roxy-data".roxyDataScriptemits both. - The JavaScript property always wins. If you assign
element.datain script, the markup is ignored. One component covers both server-rendered and dynamic pages with no branching. - You can nest a server-rendered HTML fallback inside the same element for no-JavaScript and crawler views. The component reads only the marked script and leaves the fallback in place.
- In a language that cannot call the TS helper (PHP, Python, Go), mirror its rule exactly: escape
<,>, and&to their\u003c,\u003e,\u0026JSON escapes. The WordPress example does this in PHP.
This is how the WordPress plugin renders: PHP fetches the response server-side, caches it, and writes the script into the page. The same shape works in any framework that emits HTML.
Components react to three signals in priority order. No events to dispatch. No JS bridge to write. The CDN bundle (dist/cdn/roxy-ui.js) auto-loads the design tokens, so a single script tag yields full theming and dark mode with nothing else to add. The npm and React paths inherit the same tokens through the components; only set up tokens.css yourself if you import per component without the full bundle.
| Signal | Where | Effect |
|---|---|---|
prefers-color-scheme: dark |
OS | Default. Follows user system setting. |
data-theme="light" or data-theme="dark" |
<html> / <body> / any ancestor / the component itself |
Wins over OS. Per-element override scope works. |
.dark class |
The component itself or any ancestor (typically <html>) |
Same effect as data-theme="dark". Use when the host stack already ships a .dark toggle (Tailwind, shadcn). |
To toggle at runtime:
document.documentElement.dataset.theme = 'dark'; // or 'light'That single line re-themes every Roxy UI component on the page. Persist user choice in localStorage from your own code; the library does not own preferences.
Per-element scope is supported:
<roxy-natal-chart data-theme="dark" .data=${chart}></roxy-natal-chart>Every visible aspect of the chart is driven by --roxy-* CSS custom properties on :host. Override any token on :root, on :host, or per element. Do not write Tailwind utility classes inside the components; the Shadow DOM boundary stops them at the door.
- Always call
/location/searchfirst before any chart endpoint that takes latitude, longitude, or timezone. Use<roxy-location-search>for the input UI. - Pass the response object directly. Components are stateless; they do not fetch internally except for
<roxy-location-search>,<roxy-endpoint-form>, and the widgets auto-mount script. - Use the typed SDK from
@roxyapi/sdkso prop shapes match the spec automatically. - Theming is CSS custom properties on
:rootor per element. Switch light and dark viadata-themeon any ancestor (see the table above). Do not write Tailwind classes inside the components; the shadow DOM ignores them. - Honor reduced motion. The library already respects
prefers-reduced-motion: reduceand the--roxy-motion-durationvariable. - A11y violations are CI failures. Do not paste over
roleoraria-*attributes; the components emit them correctly already. - Component types come from the OpenAPI spec via
@hey-api/openapi-ts. Do not redefine response shapes locally; if a field is missing, fix the spec, regenerate, propagate.
When listing domains in user-visible copy, use the canonical order: Western astrology, Vedic astrology, numerology, tarot, human design, forecast, biorhythm, I Ching, crystals, dreams, angel numbers. Location is utility, not a selling domain.
- Do not bundle
@roxyapi/uiand@roxyapi/ui-reacttogether; they ship independently. - Use
@roxyapi/ui-reactfor React projects. Use@roxyapi/uidirectly elsewhere. - Do not write your own kundli component. The lifted layout in
<roxy-vedic-kundli>is the canonical RoxyAPI render path. - Do not call astrology endpoints with hardcoded coordinates. Always geocode first via
<roxy-location-search>orroxy.location.searchCities(). - Do not declare a local
interface XyzDatato describe a RoxyAPI response. Import the type from the spec-derived bundle:import type { XyzResponse } from '@roxyapi/sdk'. Local interfaces drift the moment the spec changes. - Do not write Tailwind utility classes inside a component. The Shadow DOM boundary stops them at the door. Theme through
--roxy-*CSS custom properties on:rootor per element instead. - Two ways to feed a component, no third. Controlled (default, recommended for production): pass
dataas a prop or hydrate from a childroxy-dataJSON island; your server holds the secret key. Self-fetch (no backend): setdata-endpoint+ apk_publishable-keyand the component renders its own form and fetches in the browser (a secret key is refused client-side). Do not wrap a component in your own fetch loop or call a chart/table/card's internals. - Do not redefine theme tokens or invent your own naming. Override the existing
--roxy-*custom properties; the full list is inTHEMING.md.
- Component source:
packages/ui/src/components/ - Sample data for every component:
apps/docs/sample-data.js - Token reference:
packages/ui/THEMING.md - Live preview:
bun run previewthen openhttp://localhost:3001 - Endpoint reference: https://roxyapi.com/api-reference
- Machine-readable component catalog (every component, its domain, what it renders, and the endpoint(s) it consumes): https://cdn.jsdelivr.net/npm/@roxyapi/ui@latest/components-catalog.json. Fetch it to discover or map components programmatically instead of scraping this table.