One thing to keep in mind when working with decentralized systems is that there's very minimal support for things like APIs and databases. Instead, we use a combination of on-chain data (via NFTs and GraphQL tags on Arweave) and Koii Tasks running on decentralized nodes.
When constructing a leaderboard or list of content (such as a collection of an Artist's NFTs, or a personalized newsfeed), it's usually necessary to either configure an index or hijack an existing one.
Koii Tasks will go live in Q2 2022, at which point you'll be able to fully customize your system, but until then, the best course of action is likely to process the current Koii Leaderboard APIs clientside and construct your own newsfeed using filters.
View Filters
Filters can be applied to any list component as shown:
📦src ┣ 📂components ┃ ┗ 📂filters ┃ ┗ 📂TimeFilter// Set up Time Filter components /src/components/TimeFilterfunctionTimeFilter() {const [timeframe,setTimeframe] =useQueryParam<string>("t",withDefault(StringParam,"1w"));constonTimeframeChange= (newValue:string) => {setTimeframe(newValue,"replaceIn");return ( <Flex> <ToggleButtonGroupvalue={timeframe} defaultValue={timeframe} onChange={onTimeframeChange} name="timeframe"isAttachedvariant="outline"aria-label="Change timeframe"> <ToggleButtonvalue="24h"aria-label="24 hours"children="24h" /> <ToggleButtonvalue="1w"aria-label="1 week"children="1w" /> <ToggleButtonvalue="1m"aria-label="1 month"children="1m" /> <ToggleButtonvalue="1y"aria-label="1 year"children="1y" /> <ToggleButtonvalue="all"aria-label="all time"children="all" /> </ToggleButtonGroup> </Flex> );};📦src ┣ 📂api ┃ ┗ 📂hooks ┃ ┗ 📜useNfts.ts// Use Time Filter /src/api/hooks/useNfts.tsconstfetchNfts=async (timeframe:string="1w") => {try {let nsfwList:any[] = [];/* Get nsfw list */awaitfetchNsfwList().then(res => { nsfwList =res?.data?.NSFWList || []; }).catch(() => {}); // We don't want to catch anything./* Get nfts based on the timeframe */const { data } =awaitaxios.get(`/attention/nft-summaries?period=${timeframe}`);constdataWithNsfwTag=data?.map((nft:any) => ({ ...nft, isNsfw: [...nsfwList].includes(nft?.id) }));return dataWithNsfwTag; } catch (error) {returnundefined; }};exportfunctionuseNfts({ timeframe ="1w" }:Props) {const [isNsfw] =useQueryParam("nsfw");returnuseQuery(`nfts-${timeframe}`, () =>fetchNfts(timeframe), { staleTime:60*1000*5,// 5min cache refetchOnWindowFocus:undefined,select: data => {return isNsfw ? data : [...data].filter((nft:any) =>!nft.isNsfw); } });}
The default list component fetches the fully Koii NFT list, so configuring views can allow you to select any content you like without rebuilding the Tasks API. If this doesn't work for your use case, please contact our support team and we can help you deploy an Indexing Task of your own!
NSFW
Koii currently manages an NSFW blocked content list which can be accessed using the same API as the main feed. NSFW content can be toggled in list views like this:
📦src ┣ 📂components ┃ ┗ 📂filters ┃ ┗ 📂NsfwFilter// Set up NSFW components /src/components/TimeFilterfunctionNsfwFilter() {const [isOn,setIsOn] =useQueryParam("nsfw",withDefault(BooleanParam,false));constonNsfwChange=async (e:ChangeEvent<HTMLInputElement>) => {setIsOn(e.target.checked ||undefined,"replaceIn"); };return ( <> <FormControldisplay="flex"alignItems="center"justifyContent="center"> <FormLabelhtmlFor="isNsfw"m="0 6px 0 0"fontSize="xs"userSelect="none"> Show NSFW content </FormLabel> <Switchid="isNsfw"size="sm"defaultChecked={isOn} value={isOn ?1:0} onChange={onNsfwChange} /> </FormControl> </> );}📦src ┣ 📂api ┃ ┗ 📂hooks ┃ ┗ 📜useNsfw.ts// Use NSFW /src/api/hooks/useNsfw.tsconstfetchNsfw=async () => {try {const { data } =awaitaxios.get(`/getNSFWList`, { baseURL:process.env.REACT_APP_API_URL });constnsfwList=data?.NSFWList || [];return nsfwList; } catch (error) {returnundefined; }};exportfunctionuseNsfw() {returnuseQuery(`nsfw-list`, fetchNsfw, { staleTime:60*1000*60,// 1hr cache, since the nft details does not change. refetchOnWindowFocus:undefined });}
Search
Search is currently implemented using clientside libraries which draw from the same core NFT list, and can be customized by editing the Search Component, shown here.
Thumbnails
Since your app will live on decentralized storage, it can sometimes be difficult to generate thumbnail images on the fly for your NFT assets, and it can be even harder to set OpenGraph tags for sharing content.
To streamline this process, Koii has provided a thumbnail generation API powered by Koii Tasks, to which you can POST new NFTs so that they will have nicely formatted previews when sharing on social media.
📦src ┣ 📂api ┃ ┗ 📜upload.ts line:137-143// The code will send POST request to https://api.koii.live/generateCardWithData to get a thumbnailconstbody= { data: {...initialState, id:tx.id }, media};// Your media tag should look like this:// "media" : {// "url" : "NFT file in decode in base64 format"// }// For example of the body://{// "data": {// "id": "QfZ3CRZYX3dvkmAwml6qTfJZJL2atFCyF1_mM7ChggU",// "owner": null,// "title": "Tons of code",// "name": "Neo Matrix",// "description": "Lots of unused vars in my code\n",// "ticker": "KOINFT",// "balances": {// "null": 1// },// "contentType": "image/png",// "createdAt": "1632748295",// "tags": ["The one"],// "isNsfw": false// },// "media":{// "url": "/9j/4AAQSkZJRgABAQAAAQABAAD/......"// }//}📦src ┣ 📂api ┃ ┗ 📜upload.ts line:188-194constgenerateCardWithData=async (body:any) => {returnawaitaxios.post(`https://api.koii.live/generateCardWithData`, body, {transformRequest: (data, headers:any) => {headers.common["Access-Control-Allow-Origin"] ="*";return data; }, baseURL:undefined });};