feat(storeContext.tsx): add hasApiKey state and setHasApiKey function to storeContext to track if API key is present
fix(StorePage/index.tsx): fix useEffect dependencies and add missing handleChangeTab function fix(API/index.ts): rename getNumberOfComponents function to getCountComponents to improve semantics
This commit is contained in:
parent
0c90c12c94
commit
b3bb568526
4 changed files with 107 additions and 149 deletions
|
|
@ -8,6 +8,8 @@ const initialValue = {
|
|||
setSavedFlows: () => {},
|
||||
hasStore: true,
|
||||
setHasStore: () => {},
|
||||
hasApiKey: true,
|
||||
setHasApiKey: () => {},
|
||||
};
|
||||
|
||||
export const StoreContext = createContext<storeContextType>(initialValue);
|
||||
|
|
@ -16,14 +18,23 @@ export function StoreProvider({ children }) {
|
|||
const [savedFlows, setSavedFlows] = useState<Set<string>>(new Set());
|
||||
|
||||
const [hasStore, setHasStore] = useState(true);
|
||||
const [hasApiKey, setHasApiKey] = useState(true);
|
||||
|
||||
checkHasStore().then((res) => {
|
||||
setHasStore(res["enabled"]);
|
||||
setHasApiKey(res["has_api_key"]);
|
||||
});
|
||||
|
||||
return (
|
||||
<StoreContext.Provider
|
||||
value={{ savedFlows, setSavedFlows, hasStore, setHasStore }}
|
||||
value={{
|
||||
savedFlows,
|
||||
setSavedFlows,
|
||||
hasStore,
|
||||
setHasStore,
|
||||
hasApiKey,
|
||||
setHasApiKey,
|
||||
}}
|
||||
>
|
||||
{children}
|
||||
</StoreContext.Provider>
|
||||
|
|
|
|||
|
|
@ -713,9 +713,20 @@ export async function checkHasStore() {
|
|||
}
|
||||
}
|
||||
|
||||
export async function getNumberOfComponents() {
|
||||
export async function getCountComponents(is_component?: boolean | null) {
|
||||
try {
|
||||
const res = await api.get(`${BASE_URL_API}store/components/count`);
|
||||
let url = `${BASE_URL_API}store/components/count`;
|
||||
const queryParams: any = [];
|
||||
if (is_component !== undefined) {
|
||||
queryParams.push(`is_component=${is_component}`);
|
||||
}
|
||||
|
||||
if (queryParams.length > 0) {
|
||||
url += `?${queryParams.join("&")}`;
|
||||
}
|
||||
|
||||
const res = await api.get(url);
|
||||
|
||||
if (res.status === 200) {
|
||||
return res.data;
|
||||
}
|
||||
|
|
|
|||
|
|
@ -20,7 +20,7 @@ import { alertContext } from "../../contexts/alertContext";
|
|||
import { StoreContext } from "../../contexts/storeContext";
|
||||
import { TabsContext } from "../../contexts/tabsContext";
|
||||
import {
|
||||
getNumberOfComponents,
|
||||
getCountComponents,
|
||||
getStoreComponents,
|
||||
getStoreSavedComponents,
|
||||
getStoreTags,
|
||||
|
|
@ -32,25 +32,24 @@ import { cn } from "../../utils/utils";
|
|||
import { MarketCardComponent } from "./components/market-card";
|
||||
export default function StorePage(): JSX.Element {
|
||||
const { setTabId } = useContext(TabsContext);
|
||||
// set null id
|
||||
useEffect(() => {
|
||||
setTabId("");
|
||||
}, []);
|
||||
const [data, setData] = useState<storeComponent[]>([]);
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [filteredCategories, setFilterCategories] = useState<any[]>([]);
|
||||
const [inputText, setInputText] = useState<string>("");
|
||||
const [searchData, setSearchData] = useState(data);
|
||||
const [searchData, setSearchData] = useState<storeComponent[]>([]);
|
||||
const { setErrorData } = useContext(alertContext);
|
||||
const [totalRowsCount, setTotalRowsCount] = useState(0);
|
||||
const [size, setPageSize] = useState(10);
|
||||
const [index, setPageIndex] = useState(1);
|
||||
const [errorApiKey, setErrorApiKey] = useState(false);
|
||||
const { setSavedFlows, savedFlows } = useContext(StoreContext);
|
||||
const { setSavedFlows, savedFlows, hasApiKey } = useContext(StoreContext);
|
||||
const [tags, setTags] = useState<{ id: string; name: string }[]>([]);
|
||||
const tagListId = useRef<{ id: string; name: string }[]>([]);
|
||||
const [renderPagination, setRenderPagination] = useState(false);
|
||||
const filterComponent = useRef<boolean | null>(null);
|
||||
const [tabActive, setTabActive] = useState("Flows");
|
||||
|
||||
useEffect(() => {
|
||||
getStoreTags().then((res) => {
|
||||
|
|
@ -59,6 +58,16 @@ export default function StorePage(): JSX.Element {
|
|||
});
|
||||
}, []);
|
||||
|
||||
useEffect(() => {
|
||||
filterComponent.current = false;
|
||||
getSavedComponents()
|
||||
.finally(() => handleGetComponents())
|
||||
.catch((err) => {
|
||||
setErrorApiKey(true);
|
||||
console.error(err);
|
||||
});
|
||||
}, []);
|
||||
|
||||
async function getSavedComponents() {
|
||||
setLoading(true);
|
||||
const result = await getStoreSavedComponents();
|
||||
|
|
@ -70,17 +79,12 @@ export default function StorePage(): JSX.Element {
|
|||
setErrorApiKey(false);
|
||||
}
|
||||
|
||||
useEffect(() => {
|
||||
getNumberOfComponents().then((res) => {
|
||||
async function getNumberSavedComponents() {
|
||||
getCountComponents(filterComponent.current).then((res) => {
|
||||
setTotalRowsCount(Number(res["count"]));
|
||||
setLoading(false);
|
||||
});
|
||||
getSavedComponents()
|
||||
.finally(() => handleGetComponents())
|
||||
.catch((err) => {
|
||||
setErrorApiKey(true);
|
||||
console.error(err);
|
||||
});
|
||||
}, []);
|
||||
}
|
||||
|
||||
const handleGetComponents = () => {
|
||||
setLoading(true);
|
||||
|
|
@ -89,8 +93,7 @@ export default function StorePage(): JSX.Element {
|
|||
getStoreComponents(index - 1, size, filterComponent.current)
|
||||
.then((res) => {
|
||||
setSearchData(res);
|
||||
setData(res);
|
||||
setLoading(false);
|
||||
getNumberSavedComponents();
|
||||
setErrorApiKey(true);
|
||||
})
|
||||
.catch((err) => {
|
||||
|
|
@ -112,9 +115,9 @@ export default function StorePage(): JSX.Element {
|
|||
searchComponent(inputText).then(
|
||||
(res) => {
|
||||
setSearchData(res);
|
||||
setData(res);
|
||||
setLoading(false);
|
||||
setRenderPagination(false);
|
||||
setTotalRowsCount(res.length);
|
||||
},
|
||||
(error) => {
|
||||
setLoading(false);
|
||||
|
|
@ -127,7 +130,6 @@ export default function StorePage(): JSX.Element {
|
|||
setRenderPagination(true);
|
||||
getStoreComponents(pageIndex, pageSize, filterComponent.current)
|
||||
.then((res) => {
|
||||
setData(res);
|
||||
setSearchData(res);
|
||||
setPageIndex(pageIndex);
|
||||
setPageSize(pageSize);
|
||||
|
|
@ -152,8 +154,8 @@ export default function StorePage(): JSX.Element {
|
|||
searchComponent(null, 1, 10000, null, filterArray).then(
|
||||
(res) => {
|
||||
setSearchData(res);
|
||||
setData(res);
|
||||
setLoading(false);
|
||||
setTotalRowsCount(res.length);
|
||||
},
|
||||
(error) => {
|
||||
setLoading(false);
|
||||
|
|
@ -162,7 +164,32 @@ export default function StorePage(): JSX.Element {
|
|||
);
|
||||
}
|
||||
|
||||
const [tabActive, setTabActive] = useState("Flows");
|
||||
function handleChangeTab(tab: string) {
|
||||
if (tab === "All") {
|
||||
filterComponent.current = null;
|
||||
} else if (tab === "Flows") {
|
||||
filterComponent.current = false;
|
||||
} else if (tab === "Components") {
|
||||
filterComponent.current = true;
|
||||
}
|
||||
setPageIndex(1);
|
||||
setPageSize(10);
|
||||
handleGetComponents();
|
||||
}
|
||||
|
||||
const handleOrderPage = (e) => {
|
||||
let sortedData = cloneDeep(searchData);
|
||||
|
||||
if (e === "Popular") {
|
||||
sortedData = sortedData.sort(
|
||||
(a, b) => Number(b.liked_by_count) - Number(a.liked_by_count)
|
||||
);
|
||||
} else if (e === "Alphabetical") {
|
||||
sortedData = sortedData.sort((a, b) => a.name.localeCompare(b.name));
|
||||
}
|
||||
|
||||
setSearchData([...sortedData]);
|
||||
};
|
||||
|
||||
return (
|
||||
<>
|
||||
|
|
@ -181,7 +208,9 @@ export default function StorePage(): JSX.Element {
|
|||
}}
|
||||
>
|
||||
<Button
|
||||
className={`${errorApiKey ? "animate-pulse border-error" : ""}`}
|
||||
className={`${
|
||||
errorApiKey && !hasApiKey ? "animate-pulse border-error" : ""
|
||||
}`}
|
||||
variant="primary"
|
||||
>
|
||||
<IconComponent name="Key" className="main-page-nav-button" />
|
||||
|
|
@ -209,12 +238,18 @@ export default function StorePage(): JSX.Element {
|
|||
}}
|
||||
value={inputText}
|
||||
/>
|
||||
<Search className="absolute bottom-0 right-4 top-0 my-auto h-6 stroke-1 text-muted-foreground " />
|
||||
<Search
|
||||
onClick={() => {
|
||||
handleSearch(inputText);
|
||||
}}
|
||||
className="absolute bottom-0 right-4 top-0 my-auto h-6 cursor-pointer stroke-1 text-muted-foreground"
|
||||
/>
|
||||
</div>
|
||||
<div className="ml-4 flex w-full gap-2 border-b border-border">
|
||||
<button
|
||||
onClick={() => {
|
||||
setTabActive("All");
|
||||
handleChangeTab("All");
|
||||
}}
|
||||
className={
|
||||
tabActive === "All"
|
||||
|
|
@ -227,6 +262,7 @@ export default function StorePage(): JSX.Element {
|
|||
<button
|
||||
onClick={() => {
|
||||
setTabActive("Flows");
|
||||
handleChangeTab("Flows");
|
||||
}}
|
||||
className={
|
||||
tabActive === "Flows"
|
||||
|
|
@ -239,6 +275,7 @@ export default function StorePage(): JSX.Element {
|
|||
<button
|
||||
onClick={() => {
|
||||
setTabActive("Components");
|
||||
handleChangeTab("Components");
|
||||
}}
|
||||
className={
|
||||
tabActive === "Components"
|
||||
|
|
@ -255,104 +292,7 @@ export default function StorePage(): JSX.Element {
|
|||
</ShadTooltip>
|
||||
</div>
|
||||
</div>
|
||||
{/* <div className="flex items-center gap-3 text-sm">
|
||||
<button className="flex h-8 items-center justify-between rounded-md border border-ring/60 px-4 py-2 text-sm text-primary hover:bg-muted">
|
||||
<IconComponent name="CheckCircle2" className="mr-2 h-4 w-4 " />
|
||||
Installed Locally
|
||||
</button>
|
||||
<Select
|
||||
onValueChange={(value) => {
|
||||
if (value === "Flow") {
|
||||
filterComponent.current = false;
|
||||
} else if (value === "Component") {
|
||||
filterComponent.current = true;
|
||||
} else {
|
||||
filterComponent.current = null;
|
||||
}
|
||||
setPageIndex(1);
|
||||
setPageSize(10);
|
||||
handleGetComponents();
|
||||
}}
|
||||
>
|
||||
<SelectTrigger>
|
||||
<SelectValue placeholder="Component Types" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
<SelectItem value="">Both</SelectItem>
|
||||
<SelectItem value="Flow">Flows</SelectItem>
|
||||
<SelectItem value="Component">Components</SelectItem>
|
||||
</SelectContent>
|
||||
</Select>
|
||||
<Select
|
||||
onValueChange={(value) => {
|
||||
if (value === "Flow") {
|
||||
filterComponent.current = false;
|
||||
} else if (value === "Component") {
|
||||
filterComponent.current = true;
|
||||
} else {
|
||||
filterComponent.current = null;
|
||||
}
|
||||
setPageIndex(1);
|
||||
setPageSize(10);
|
||||
handleGetComponents();
|
||||
}}
|
||||
>
|
||||
<SelectTrigger>
|
||||
<SelectValue placeholder="Use Cases" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
<SelectItem value="">Both</SelectItem>
|
||||
<SelectItem value="Flow">Flows</SelectItem>
|
||||
<SelectItem value="Component">Components</SelectItem>
|
||||
</SelectContent>
|
||||
</Select>
|
||||
<Select
|
||||
onValueChange={(value) => {
|
||||
if (value === "Flow") {
|
||||
filterComponent.current = false;
|
||||
} else if (value === "Component") {
|
||||
filterComponent.current = true;
|
||||
} else {
|
||||
filterComponent.current = null;
|
||||
}
|
||||
setPageIndex(1);
|
||||
setPageSize(10);
|
||||
handleGetComponents();
|
||||
}}
|
||||
>
|
||||
<SelectTrigger>
|
||||
<SelectValue placeholder="Models" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
<SelectItem value="">Both</SelectItem>
|
||||
<SelectItem value="Flow">Flows</SelectItem>
|
||||
<SelectItem value="Component">Components</SelectItem>
|
||||
</SelectContent>
|
||||
</Select>
|
||||
<Select
|
||||
onValueChange={(value) => {
|
||||
if (value === "Flow") {
|
||||
filterComponent.current = false;
|
||||
} else if (value === "Component") {
|
||||
filterComponent.current = true;
|
||||
} else {
|
||||
filterComponent.current = null;
|
||||
}
|
||||
setPageIndex(1);
|
||||
setPageSize(10);
|
||||
handleGetComponents();
|
||||
}}
|
||||
>
|
||||
<SelectTrigger>
|
||||
<SelectValue placeholder="Payment" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
<SelectItem value="">Both</SelectItem>
|
||||
<SelectItem value="Flow">Flows</SelectItem>
|
||||
<SelectItem value="Component">Components</SelectItem>
|
||||
</SelectContent>
|
||||
</Select>
|
||||
</div> */}
|
||||
|
||||
<div className="flex items-center gap-2 px-2">
|
||||
{!loading &&
|
||||
tags.map((i, idx) => (
|
||||
|
|
@ -383,29 +323,25 @@ export default function StorePage(): JSX.Element {
|
|||
</div>
|
||||
<div className="flex items-end justify-between">
|
||||
<span className="px-0.5 text-sm text-muted-foreground">
|
||||
2,117 results
|
||||
{!loading && (
|
||||
<>
|
||||
{totalRowsCount} {totalRowsCount > 0 ? "results" : "result"}
|
||||
</>
|
||||
)}
|
||||
</span>
|
||||
|
||||
<Select
|
||||
onValueChange={(value) => {
|
||||
if (value === "Flow") {
|
||||
filterComponent.current = false;
|
||||
} else if (value === "Component") {
|
||||
filterComponent.current = true;
|
||||
} else {
|
||||
filterComponent.current = null;
|
||||
}
|
||||
setPageIndex(1);
|
||||
setPageSize(10);
|
||||
handleGetComponents();
|
||||
onValueChange={(e) => {
|
||||
handleOrderPage(e);
|
||||
}}
|
||||
>
|
||||
<SelectTrigger>
|
||||
<SelectValue placeholder="Sort By" />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
<SelectItem value="">Most Popular</SelectItem>
|
||||
<SelectItem value="Flow">Most Recent</SelectItem>
|
||||
<SelectItem value="Component">Alphabetical</SelectItem>
|
||||
<SelectItem value="Popular">Most Popular</SelectItem>
|
||||
{/* <SelectItem value="Recent">Most Recent</SelectItem> */}
|
||||
<SelectItem value="Alphabetical">Alphabetical</SelectItem>
|
||||
</SelectContent>
|
||||
</Select>
|
||||
</div>
|
||||
|
|
@ -418,17 +354,15 @@ export default function StorePage(): JSX.Element {
|
|||
<SkeletonCardComponent />
|
||||
</>
|
||||
) : (
|
||||
searchData
|
||||
.filter(
|
||||
(f) =>
|
||||
filteredCategories?.length === 0 ||
|
||||
filteredCategories.some(
|
||||
(category) => category === f.is_component
|
||||
)
|
||||
)
|
||||
.map((item, idx) => (
|
||||
<MarketCardComponent key={idx} data={item} />
|
||||
))
|
||||
searchData.map((item, idx) => {
|
||||
console.log(item);
|
||||
|
||||
return (
|
||||
<>
|
||||
<MarketCardComponent key={idx} data={item} />
|
||||
</>
|
||||
);
|
||||
})
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
|
|
|
|||
|
|
@ -3,4 +3,6 @@ export type storeContextType = {
|
|||
setSavedFlows: (newState: Set<string>) => void;
|
||||
setHasStore: (store: boolean) => void;
|
||||
hasStore: boolean;
|
||||
setHasApiKey: (key: boolean) => void;
|
||||
hasApiKey: boolean;
|
||||
};
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue