From 48c07ad1c0e00e63cff24b4ccb7b837c64561b35 Mon Sep 17 00:00:00 2001
From: artur <brugeman.artur@gmail.com>
Date: Thu, 8 Feb 2024 14:15:45 +0300
Subject: [PATCH] Implement connectApp logic, add app url and icon

---
 .../ModalConfirmConnect.tsx                   |  82 ++++++--
 .../ModalConfirmEvent/ModalConfirmEvent.tsx   |   2 +-
 src/modules/backend.ts                        | 194 ++++++++++++------
 .../components/Permissions/ItemPermission.tsx |   2 +-
 src/pages/CreatePage/Create.Page.tsx          | 100 +++++++++
 src/pages/CreatePage/styled.tsx               |  26 +++
 src/pages/HomePage/Home.Page.tsx              |   2 +-
 src/pages/KeyPage/components/Apps.tsx         |   2 +-
 src/pages/KeyPage/components/ItemApp.tsx      |  15 +-
 src/routes/AppRoutes.tsx                      |   2 +
 src/utils/helpers/helpers.ts                  |  13 +-
 11 files changed, 350 insertions(+), 90 deletions(-)
 create mode 100644 src/pages/CreatePage/Create.Page.tsx
 create mode 100644 src/pages/CreatePage/styled.tsx

diff --git a/src/components/Modal/ModalConfirmConnect/ModalConfirmConnect.tsx b/src/components/Modal/ModalConfirmConnect/ModalConfirmConnect.tsx
index bc82cd9..9c955a4 100644
--- a/src/components/Modal/ModalConfirmConnect/ModalConfirmConnect.tsx
+++ b/src/components/Modal/ModalConfirmConnect/ModalConfirmConnect.tsx
@@ -1,9 +1,9 @@
 import { useModalSearchParams } from '@/hooks/useModalSearchParams'
 import { Modal } from '@/shared/Modal/Modal'
 import { MODAL_PARAMS_KEYS } from '@/types/modal'
-import { call, getAppIconTitle, getShortenNpub } from '@/utils/helpers/helpers'
+import { call, getAppIconTitle, getDomain, getShortenNpub } from '@/utils/helpers/helpers'
 import { Avatar, Box, Stack, Typography } from '@mui/material'
-import { useParams, useSearchParams } from 'react-router-dom'
+import { useNavigate, useParams, useSearchParams } from 'react-router-dom'
 import { useAppSelector } from '@/store/hooks/redux'
 import { selectAppsByNpub } from '@/store'
 import { StyledButton, StyledToggleButtonsGroup } from './styled'
@@ -11,25 +11,33 @@ import { ActionToggleButton } from './сomponents/ActionToggleButton'
 import { useState } from 'react'
 import { swicCall } from '@/modules/swic'
 import { ACTION_TYPE } from '@/utils/consts'
+import { useEnqueueSnackbar } from '@/hooks/useEnqueueSnackbar'
 
 export const ModalConfirmConnect = () => {
   const { getModalOpened, createHandleCloseReplace } = useModalSearchParams()
+  const notify = useEnqueueSnackbar()
+	const navigate = useNavigate()
   const isModalOpened = getModalOpened(MODAL_PARAMS_KEYS.CONFIRM_CONNECT)
 
-  const { npub = '' } = useParams<{ npub: string }>()
+  const [searchParams] = useSearchParams()
+	const paramNpub = searchParams.get('npub') || ''
+  const { npub = paramNpub } = useParams<{ npub: string }>()
   const apps = useAppSelector((state) => selectAppsByNpub(state, npub))
 
   const [selectedActionType, setSelectedActionType] = useState<ACTION_TYPE>(ACTION_TYPE.BASIC)
 
-  const [searchParams] = useSearchParams()
   const appNpub = searchParams.get('appNpub') || ''
   const pendingReqId = searchParams.get('reqId') || ''
   const isPopup = searchParams.get('popup') === 'true'
+  const token = searchParams.get('token') || ''
 
   const triggerApp = apps.find((app) => app.appNpub === appNpub)
-  const { name, icon = '' } = triggerApp || {}
-  const appName = name || getShortenNpub(appNpub)
-	const appAvatarTitle = getAppIconTitle(name, appNpub)
+  const { name, url = '', icon = '' } = triggerApp || {}
+  const appUrl = url || searchParams.get('appUrl') || ''
+	const appDomain = getDomain(appUrl)
+  const appName = name || appDomain || getShortenNpub(appNpub)
+	const appAvatarTitle = getAppIconTitle(name || appDomain, appNpub)
+	const appIcon = icon || (appDomain ? `https://${appDomain}/favicon.ico` : '')
 
   const handleActionTypeChange = (_: any, value: ACTION_TYPE | null) => {
     if (!value) return undefined
@@ -47,6 +55,9 @@ export const ModalConfirmConnect = () => {
     onClose: (sp) => {
       sp.delete('appNpub')
       sp.delete('reqId')
+      sp.delete('popup')
+      sp.delete('npub')
+      sp.delete('appUrl')
     },
   })
 
@@ -59,16 +70,57 @@ export const ModalConfirmConnect = () => {
     if (isPopup) window.close()
   }
 
-  const allow = () => {
-    const options: any = {}
-    if (selectedActionType === ACTION_TYPE.BASIC) options.perms = [ACTION_TYPE.BASIC]
-    // else
-    // 	options.perms = ['connect','get_public_key'];
-    confirmPending(pendingReqId, true, true, options)
+  const allow = async () => {
+		let perms = ['connect','get_public_key'];
+		if (selectedActionType === ACTION_TYPE.BASIC) perms = [ACTION_TYPE.BASIC]
+
+		if (pendingReqId) {
+			const options = { perms }
+			await confirmPending(pendingReqId, true, true, options)	
+		} else {
+			try {
+				await swicCall('enablePush')
+				console.log('enablePush done')	
+			} catch (e: any) {
+				console.log('error', e)
+				notify('Please enable Notifications in website settings!', 'error')
+				return;
+			}
+
+			try {
+				await swicCall('connectApp', { npub, appNpub, appUrl, perms })
+				console.log('connectApp done', npub, appNpub, appUrl, perms)
+			} catch (e: any) {
+				notify(e.toString(), 'error')
+				return;
+			}
+
+			if (token) {
+				try {
+					await swicCall('redeemToken', npub, token)
+					console.log('redeemToken done')	
+				} catch (e) {
+					console.log("error", e);
+					notify('App did not reply. Please try to log in now.', 'error')
+					navigate(`/key/${npub}`, { replace: true })
+					return;
+				}
+			}
+
+			notify('App connected! Closing...', 'success')
+
+			if (isPopup)
+				setTimeout(() => window.close(), 3000);
+			else
+				navigate(`/key/${npub}`, { replace: true })
+		}
   }
 
   const disallow = () => {
-    confirmPending(pendingReqId, false, true)
+		if (pendingReqId)
+	    confirmPending(pendingReqId, false, true)
+		else
+			closeModalAfterRequest()
   }
 
   if (isPopup) {
@@ -91,7 +143,7 @@ export const ModalConfirmConnect = () => {
               width: 56,
               height: 56,
             }}
-            src={icon}
+            src={appIcon}
           >
 						{appAvatarTitle}
 					</Avatar>
diff --git a/src/components/Modal/ModalConfirmEvent/ModalConfirmEvent.tsx b/src/components/Modal/ModalConfirmEvent/ModalConfirmEvent.tsx
index 5be7df3..583efff 100644
--- a/src/components/Modal/ModalConfirmEvent/ModalConfirmEvent.tsx
+++ b/src/components/Modal/ModalConfirmEvent/ModalConfirmEvent.tsx
@@ -1,7 +1,7 @@
 import { useModalSearchParams } from '@/hooks/useModalSearchParams'
 import { Modal } from '@/shared/Modal/Modal'
 import { MODAL_PARAMS_KEYS } from '@/types/modal'
-import { call, getAppIconTitle, getReqActionName, getShortenNpub, getSignReqKind } from '@/utils/helpers/helpers'
+import { call, getAppIconTitle, getReqActionName, getShortenNpub } from '@/utils/helpers/helpers'
 import { Avatar, Box, List, ListItem, ListItemIcon, ListItemText, Stack, Typography } from '@mui/material'
 import { useParams, useSearchParams } from 'react-router-dom'
 import { useAppSelector } from '@/store/hooks/redux'
diff --git a/src/modules/backend.ts b/src/modules/backend.ts
index 6bd6bac..bff1bda 100644
--- a/src/modules/backend.ts
+++ b/src/modules/backend.ts
@@ -246,12 +246,12 @@ export class NoauthBackend {
     return Buffer.from(await this.swg.crypto.subtle.digest('SHA-256', Buffer.from(s))).toString('hex')
   }
 
-	private async fetchNpubName(npub: string) {
+  private async fetchNpubName(npub: string) {
     const url = `${NOAUTHD_URL}/name?npub=${npub}`
     const r = await fetch(url)
-		const d = await r.json()
-		return d?.names?.length ? d.names[0] as string : ''
-	}
+    const d = await r.json()
+    return d?.names?.length ? (d.names[0] as string) : ''
+  }
 
   private async sendPost({ url, method, headers, body }: { url: string; method: string; headers: any; body: string }) {
     const r = await fetch(url, {
@@ -407,6 +407,23 @@ export class NoauthBackend {
     throw new Error('Too many requests, retry later')
   }
 
+  private async sendTokenToServer(npub: string, token: string) {
+    const body = JSON.stringify({
+      npub,
+      token,
+    })
+
+    const method = 'POST'
+    const url = `${NOAUTHD_URL}/created`
+
+    return this.sendPostAuthd({
+      npub,
+      url,
+      method,
+      body,
+    })
+  }
+
   private notify() {
     // FIXME collect info from accessBuffer and confirmBuffer
     // and update the notifications
@@ -550,6 +567,50 @@ export class NoauthBackend {
     return perm?.value || ''
   }
 
+  private async connectApp({
+		npub,
+		appNpub,
+		appUrl,
+		perms,
+		appName = '',
+		appIcon = ''
+	}: { 
+		npub: string, 
+		appNpub: string, 
+		appUrl: string,
+		appName?: string,
+		appIcon?: string, 
+		perms: string[] 
+	}) {
+
+		await dbi.addApp({
+      appNpub: appNpub,
+      npub: npub,
+      timestamp: Date.now(),
+      name: appName,
+      icon: appIcon,
+      url: appUrl,
+    })
+
+    // reload
+    this.apps = await dbi.listApps()
+
+    // write new perms confirmed by user
+    for (const p of perms) {
+      await dbi.addPerm({
+        id: Math.random().toString(36).substring(7),
+        npub: npub,
+        appNpub: appNpub,
+        perm: p,
+        value: '1',
+        timestamp: Date.now(),
+      })
+    }
+
+    // reload
+    this.perms = await dbi.listPerms()
+  }
+
   private async allowPermitCallback({
     backend,
     npub,
@@ -566,13 +627,13 @@ export class NoauthBackend {
     }
 
     const appNpub = nip19.npubEncode(remotePubkey)
-		const connected = !!this.apps.find(a => a.appNpub === appNpub)
-		if (!connected && method !== 'connect') {
+    const connected = !!this.apps.find((a) => a.appNpub === appNpub)
+    if (!connected && method !== 'connect') {
       console.log('ignoring request before connect', method, id, appNpub, npub)
       return false
-		}
+    }
 
-		const req: DbPending = {
+    const req: DbPending = {
       id,
       npub,
       appNpub,
@@ -588,24 +649,24 @@ export class NoauthBackend {
         // confirm
         console.log(Date.now(), allow ? 'allowed' : 'disallowed', npub, method, options, params)
 
-				if (manual) {
+        if (manual) {
           await dbi.confirmPending(id, allow)
 
-					// add app on 'allow connect'
+          // add app on 'allow connect'
           if (method === 'connect' && allow) {
             // if (!(await dbi.getApp(req.appNpub))) {
-						await dbi.addApp({
-							appNpub: req.appNpub,
-							npub: req.npub,
-							timestamp: Date.now(),
-							name: '',
-							icon: '',
-							url: '',
-						})
+            await dbi.addApp({
+              appNpub: req.appNpub,
+              npub: req.npub,
+              timestamp: Date.now(),
+              name: '',
+              icon: '',
+              url: '',
+            })
 
-						// reload
-						self.apps = await dbi.listApps()
-					}
+            // reload
+            self.apps = await dbi.listApps()
+          }
         } else {
           // just send to db w/o waiting for it
           dbi.addConfirmed({
@@ -625,7 +686,7 @@ export class NoauthBackend {
           let newPerms = [getReqPerm(req)]
           if (allow && options && options.perms) newPerms = options.perms
 
-					// write new perms confirmed by user
+          // write new perms confirmed by user
           for (const p of newPerms) {
             await dbi.addPerm({
               id: req.id,
@@ -635,14 +696,14 @@ export class NoauthBackend {
               value: allow ? '1' : '0',
               timestamp: Date.now(),
             })
-					}
+          }
 
-					// reload
+          // reload
           this.perms = await dbi.listPerms()
 
-					// confirm pending requests that might now have
-					// the proper perms
-					const otherReqs = self.confirmBuffer.filter((r) => r.req.appNpub === req.appNpub)
+          // confirm pending requests that might now have
+          // the proper perms
+          const otherReqs = self.confirmBuffer.filter((r) => r.req.appNpub === req.appNpub)
           console.log('updated perms', this.perms, 'otherReqs', otherReqs, 'connected', connected)
           for (const r of otherReqs) {
             let perm = this.getPerm(r.req)
@@ -790,6 +851,11 @@ export class NoauthBackend {
     return k
   }
 
+  private async redeemToken(npub: string, token: string) {
+    console.log('redeeming token', npub, token)
+    await this.sendTokenToServer(npub, token)
+  }
+
   private async importKey(name: string, nsec: string) {
     const k = await this.addKey({ name, nsec })
     this.updateUI()
@@ -821,42 +887,42 @@ export class NoauthBackend {
     const key = this.enckeys.find((k) => k.npub === npub)
     if (key) return this.keyInfo(key)
 
-		let name = ''
-		let existingName = true
-		// check name - user might have provided external nip05,
-		// or just his npub - we must fetch their name from our 
-		// server, and if not exists - try to assign one
-		const npubName = await this.fetchNpubName(npub)
-		if (npubName) {
-			// already have name for this npub
-			console.log("existing npub name", npub, npubName)
-			name = npubName
-		} else if (nip05.includes('@')) {
-			// no name for them?
-			const [nip05name, domain] = nip05.split('@')
-			if (domain === DOMAIN) {
-				// wtf? how did we learn their npub if 
-				// it's the name on our server but we can't fetch it?
-				console.log("existing name", nip05name)
-				name = nip05name
-			} else {
-				// try to take same name on our domain
-				existingName = false
-				name = nip05name
-				let takenName = await fetchNip05(`${name}@${DOMAIN}`)
-				if (takenName) {
-					// already taken? try name_domain as name
-					name = `${nip05name}_${domain}`
-					takenName = await fetchNip05(`${name}@${DOMAIN}`)
-				}
-				if (takenName) {
-					console.log("All names taken, leave without a name?")
-					name = ''
-				}
-			}	
-		}
+    let name = ''
+    let existingName = true
+    // check name - user might have provided external nip05,
+    // or just his npub - we must fetch their name from our
+    // server, and if not exists - try to assign one
+    const npubName = await this.fetchNpubName(npub)
+    if (npubName) {
+      // already have name for this npub
+      console.log('existing npub name', npub, npubName)
+      name = npubName
+    } else if (nip05.includes('@')) {
+      // no name for them?
+      const [nip05name, domain] = nip05.split('@')
+      if (domain === DOMAIN) {
+        // wtf? how did we learn their npub if
+        // it's the name on our server but we can't fetch it?
+        console.log('existing name', nip05name)
+        name = nip05name
+      } else {
+        // try to take same name on our domain
+        existingName = false
+        name = nip05name
+        let takenName = await fetchNip05(`${name}@${DOMAIN}`)
+        if (takenName) {
+          // already taken? try name_domain as name
+          name = `${nip05name}_${domain}`
+          takenName = await fetchNip05(`${name}@${DOMAIN}`)
+        }
+        if (takenName) {
+          console.log('All names taken, leave without a name?')
+          name = ''
+        }
+      }
+    }
 
-		console.log("fetch", { name, existingName })
+    console.log('fetch', { name, existingName })
 
     // add new key
     const nsec = await this.keysModule.decryptKeyPass({
@@ -926,6 +992,8 @@ export class NoauthBackend {
       let result = undefined
       if (method === 'generateKey') {
         result = await this.generateKey(args[0])
+      } else if (method === 'redeemToken') {
+        result = await this.redeemToken(args[0], args[1])
       } else if (method === 'importKey') {
         result = await this.importKey(args[0], args[1])
       } else if (method === 'saveKey') {
@@ -934,6 +1002,8 @@ export class NoauthBackend {
         result = await this.fetchKey(args[0], args[1], args[2])
       } else if (method === 'confirm') {
         result = await this.confirm(args[0], args[1], args[2], args[3])
+      } else if (method === 'connectApp') {
+        result = await this.connectApp(args[0])
       } else if (method === 'deleteApp') {
         result = await this.deleteApp(args[0])
       } else if (method === 'deletePerm') {
diff --git a/src/pages/AppPage/components/Permissions/ItemPermission.tsx b/src/pages/AppPage/components/Permissions/ItemPermission.tsx
index 961fa19..f03b22d 100644
--- a/src/pages/AppPage/components/Permissions/ItemPermission.tsx
+++ b/src/pages/AppPage/components/Permissions/ItemPermission.tsx
@@ -15,7 +15,7 @@ type ItemPermissionProps = {
 }
 
 export const ItemPermission: FC<ItemPermissionProps> = ({ permission }) => {
-  const { perm, value, timestamp, id } = permission || {}
+  const { value, timestamp, id } = permission || {}
 
   const { anchorEl, handleClose, handleOpen, open } = useOpenMenu()
 
diff --git a/src/pages/CreatePage/Create.Page.tsx b/src/pages/CreatePage/Create.Page.tsx
new file mode 100644
index 0000000..5ce3a97
--- /dev/null
+++ b/src/pages/CreatePage/Create.Page.tsx
@@ -0,0 +1,100 @@
+import { Stack, Typography } from '@mui/material'
+import { GetStartedButton, LearnMoreButton } from './styled'
+import { DOMAIN } from '@/utils/consts'
+import { useSearchParams } from 'react-router-dom'
+import { swicCall } from '@/modules/swic'
+import { useEnqueueSnackbar } from '@/hooks/useEnqueueSnackbar'
+import { ModalConfirmConnect } from '@/components/Modal/ModalConfirmConnect/ModalConfirmConnect'
+import { useModalSearchParams } from '@/hooks/useModalSearchParams'
+import { MODAL_PARAMS_KEYS } from '@/types/modal'
+
+const CreatePage = () => {
+  const notify = useEnqueueSnackbar()
+  const { handleOpen } = useModalSearchParams()
+
+  const [searchParams] = useSearchParams()
+
+  const name = searchParams.get('name') || ''
+  const token = searchParams.get('token') || ''
+  const appNpub = searchParams.get('appNpub') || ''
+  const isValid = name && token && appNpub
+
+  const nip05 = `${name}@${DOMAIN}`
+
+  const handleLearnMore = () => {
+    // @ts-ignore
+    window.open(`https://${DOMAIN}`, '_blank').focus()
+  }
+
+  const handleClickAddAccount = async () => {
+    try {
+      const key: any = await swicCall('generateKey', name)
+
+      let appUrl = ''
+      if (window.document.referrer) {
+        try {
+          const u = new URL(window.document.referrer)
+          appUrl = u.origin
+        } catch {}
+      }
+
+      console.log('Created', key.npub, 'app', appUrl)
+
+      handleOpen(MODAL_PARAMS_KEYS.CONFIRM_CONNECT, {
+        search: {
+          npub: key.npub,
+          appNpub,
+          appUrl,
+          token,
+          // will close after all done
+          popup: 'true'
+        },
+      });
+
+    } catch (error: any) {
+      notify(error.message || error.toString(), 'error')
+    }
+  }
+
+  if (!isValid) {
+    return (
+      <Stack maxHeight={'100%'} overflow={'auto'}>
+        <Typography textAlign={'center'} variant="h6" paddingTop="1em">
+          Bad parameters.
+        </Typography>
+      </Stack>
+    )
+  }
+
+  return (
+    <>
+      <Stack maxHeight={'100%'} overflow={'auto'}>
+        <Typography textAlign={'center'} variant="h4" paddingTop="0.5em">
+          Welcome to Nostr!
+        </Typography>
+        <Stack gap={'0.5rem'} overflow={'auto'}>
+          <Typography textAlign={'left'} variant="h6" paddingTop="0.5em">
+            Chosen name: <b>{nip05}</b>
+          </Typography>
+          <GetStartedButton onClick={handleClickAddAccount}>Create account</GetStartedButton>
+
+          <Typography textAlign={'left'} variant="h5" paddingTop="1em">
+            What you need to know:
+          </Typography>
+
+          <ol style={{ marginLeft: '1em' }}>
+            <li>Nostr accounts are based on cryptographic keys.</li>
+            <li>All your actions on Nostr will be signed by your keys.</li>
+            <li>Nsec.app is one of many services to manage Nostr keys.</li>
+            <li>When you create an account, a new key will be created.</li>
+            <li>This key can later be used with other Nostr websites.</li>
+          </ol>
+          <LearnMoreButton onClick={handleLearnMore}>Learn more</LearnMoreButton>
+        </Stack>
+      </Stack>
+      <ModalConfirmConnect />
+    </>
+  )
+}
+
+export default CreatePage
diff --git a/src/pages/CreatePage/styled.tsx b/src/pages/CreatePage/styled.tsx
new file mode 100644
index 0000000..35ff4c6
--- /dev/null
+++ b/src/pages/CreatePage/styled.tsx
@@ -0,0 +1,26 @@
+import { AppButtonProps, Button } from '@/shared/Button/Button'
+import { styled } from '@mui/material'
+import PersonAddAltRoundedIcon from '@mui/icons-material/PersonAddAltRounded'
+import PlayArrowOutlinedIcon from '@mui/icons-material/PlayArrowOutlined'
+import HelpOutlineOutlinedIcon from '@mui/icons-material/HelpOutlineOutlined'
+
+export const AddAccountButton = styled((props: AppButtonProps) => (
+  <Button {...props} startIcon={<PersonAddAltRoundedIcon />} />
+))(() => ({
+  alignSelf: 'center',
+  padding: '0.35rem 1rem',
+}))
+
+export const GetStartedButton = styled((props: AppButtonProps) => (
+  <Button {...props} startIcon={<PlayArrowOutlinedIcon />} />
+))(() => ({
+  alignSelf: 'left',
+  padding: '0.35rem 1rem',
+}))
+
+export const LearnMoreButton = styled((props: AppButtonProps) => (
+  <Button {...props} startIcon={<HelpOutlineOutlinedIcon />} />
+))(() => ({
+  alignSelf: 'left',
+  padding: '0.35rem 1rem',
+}))
diff --git a/src/pages/HomePage/Home.Page.tsx b/src/pages/HomePage/Home.Page.tsx
index 4024bec..c69627a 100644
--- a/src/pages/HomePage/Home.Page.tsx
+++ b/src/pages/HomePage/Home.Page.tsx
@@ -18,7 +18,7 @@ const HomePage = () => {
 
   const handleLearnMore = () => {
     // @ts-ignore
-    window.open(`https://info.${DOMAIN}`, '_blank').focus()
+    window.open(`https://${DOMAIN}`, '_blank').focus()
   }
 
   return (
diff --git a/src/pages/KeyPage/components/Apps.tsx b/src/pages/KeyPage/components/Apps.tsx
index d7e3e10..bba05bc 100644
--- a/src/pages/KeyPage/components/Apps.tsx
+++ b/src/pages/KeyPage/components/Apps.tsx
@@ -15,7 +15,7 @@ type AppsProps = {
   npub: string
 }
 
-export const Apps: FC<AppsProps> = ({ apps = [], npub = '' }) => {
+export const Apps: FC<AppsProps> = ({ apps = [] }) => {
   const notify = useEnqueueSnackbar()
 
   // eslint-disable-next-line
diff --git a/src/pages/KeyPage/components/ItemApp.tsx b/src/pages/KeyPage/components/ItemApp.tsx
index ae69eb7..a016137 100644
--- a/src/pages/KeyPage/components/ItemApp.tsx
+++ b/src/pages/KeyPage/components/ItemApp.tsx
@@ -2,15 +2,16 @@ import { DbApp } from '@/modules/db'
 import { Avatar, Stack, Typography } from '@mui/material'
 import { FC } from 'react'
 import { Link } from 'react-router-dom'
-// import ImageOutlinedIcon from '@mui/icons-material/ImageOutlined'
-import { getAppIconTitle, getShortenNpub } from '@/utils/helpers/helpers'
+import { getAppIconTitle, getDomain, getShortenNpub } from '@/utils/helpers/helpers'
 import { StyledItemAppContainer } from './styled'
 
 type ItemAppProps = DbApp
 
-export const ItemApp: FC<ItemAppProps> = ({ npub, appNpub, icon, name }) => {
-  const appName = name || getShortenNpub(appNpub)
-	const appAvatarTitle = getAppIconTitle(name, appNpub)
+export const ItemApp: FC<ItemAppProps> = ({ npub, appNpub, icon, name, url }) => {
+  const appDomain = getDomain(url)
+  const appName = name || appDomain || getShortenNpub(appNpub)
+  const appIcon = icon || `https://${appDomain}/favicon.ico`
+	const appAvatarTitle = getAppIconTitle(name || appDomain, appNpub)
   return (
     <StyledItemAppContainer
       direction={'row'}
@@ -23,8 +24,8 @@ export const ItemApp: FC<ItemAppProps> = ({ npub, appNpub, icon, name }) => {
       <Avatar 
         variant="rounded" 
         sx={{ width: 56, height: 56 }} 
-        src={icon} 
-        alt={name}
+        src={appIcon} 
+        alt={appName}
       >
         {appAvatarTitle}
       </Avatar>
diff --git a/src/routes/AppRoutes.tsx b/src/routes/AppRoutes.tsx
index 069e366..72df1de 100644
--- a/src/routes/AppRoutes.tsx
+++ b/src/routes/AppRoutes.tsx
@@ -4,6 +4,7 @@ import HomePage from '../pages/HomePage/Home.Page'
 import WelcomePage from '../pages/Welcome.Page'
 import { Layout } from '../layout/Layout'
 import { CircularProgress, Stack } from '@mui/material'
+import CreatePage from '@/pages/CreatePage/Create.Page'
 
 const KeyPage = lazy(() => import('../pages/KeyPage/Key.Page'))
 const ConfirmPage = lazy(() => import('../pages/Confirm.Page'))
@@ -26,6 +27,7 @@ const AppRoutes = () => {
           <Route path="/key/:npub" element={<KeyPage />} />
           <Route path="/key/:npub/app/:appNpub" element={<AppPage />} />
           <Route path="/key/:npub/:req_id" element={<ConfirmPage />} />
+          <Route path="/create" element={<CreatePage />} />
         </Route>
         <Route path="*" element={<Navigate to={'/home'} />} />
       </Routes>
diff --git a/src/utils/helpers/helpers.ts b/src/utils/helpers/helpers.ts
index eadc8c1..811b344 100644
--- a/src/utils/helpers/helpers.ts
+++ b/src/utils/helpers/helpers.ts
@@ -3,14 +3,23 @@ import { ACTIONS, ACTION_TYPE, NIP46_RELAYS } from '../consts'
 import { DbPending, DbPerm } from '@/modules/db'
 import { MetaEvent } from '@/types/meta-event'
 
-export async function call(cb: () => any) {
+export async function call(cb: () => any, err?: (e: string) => void) {
   try {
     return await cb()
-  } catch (e) {
+  } catch (e: any) {
     console.log(`Error: ${e}`)
+		err?.(e.toString());
   }
 }
 
+export const getDomain = (url: string) => {
+	try {
+		return new URL(url).hostname
+	} catch {
+		return ''
+	}
+}
+
 export const getShortenNpub = (npub = '') => {
   return npub.substring(0, 10) + '...' + npub.slice(-4)
 }