Skip to content

Commit f591294

Browse files
feature: NFT Gallery + transfer modal UX (CityOfZion#2429)
* Add icon, route and resize/style sidebar component * Fix redux stuff, add GET logic and UX * Remove dead code add placeholder * CSS and UI polish across various panel components * Remove redux cruft * Update snaps * Remove hard coded test address * Remove dead code, fix naming * Fix typos * feature: support for transferring NFTs (CityOfZion#2430) * Bump @walletconnect/sign-client (CityOfZion#2428) * Working MVP * lint * improve error handling and add notitications * Add pending transaction UX, fix UI bug in activity tab * lint and cleanup * Implement figma mocks, update watchOnly sidebar * remove comments * remove comments * fix sidebar add ledger support * update snaps * Fix wallet connect * Fix conflicts * Revert changes to yarn.lock * remove logging * update yarn.lock * Revert changes to yarn.lock * Revert changes to yarn.lock * Bump version * Improve header UI in gallery, adds defensive programming, fixes dependency issue (CityOfZion#2441)
1 parent 6271e7e commit f591294

38 files changed

+1240
-261
lines changed

__tests__/components/__snapshots__/News.test.js.snap

-2
Original file line numberDiff line numberDiff line change
@@ -767,12 +767,10 @@ exports[`News renders without crashing 1`] = `
767767
<Panel
768768
className="newsPanel"
769769
contentClassName="newsPanelContent"
770-
onScroll={[Function]}
771770
renderHeader={null}
772771
>
773772
<div
774773
className="panel newsPanel"
775-
onScroll={[Function]}
776774
>
777775
<Content
778776
className="content newsPanelContent"

__tests__/components/__snapshots__/Sidebar.test.js.snap

+38
Original file line numberDiff line numberDiff line change
@@ -1147,6 +1147,44 @@ exports[`Sidebar renders without crashing 1`] = `
11471147
</Link>
11481148
</Route>
11491149
</NavLink>
1150+
<NavLink
1151+
activeClassName="active"
1152+
aria-current="page"
1153+
className="navItem"
1154+
id="NFT"
1155+
to="/NFT"
1156+
>
1157+
<Route
1158+
path="\\\\/NFT"
1159+
>
1160+
<Link
1161+
aria-current={null}
1162+
className="navItem"
1163+
id="NFT"
1164+
replace={false}
1165+
to="/NFT"
1166+
>
1167+
<a
1168+
aria-current={null}
1169+
className="navItem"
1170+
href="/NFT"
1171+
id="NFT"
1172+
onClick={[Function]}
1173+
>
1174+
<Component
1175+
height="20px"
1176+
>
1177+
<svg
1178+
height="20px"
1179+
/>
1180+
</Component>
1181+
<div>
1182+
NFTs
1183+
</div>
1184+
</a>
1185+
</Link>
1186+
</Route>
1187+
</NavLink>
11501188
<NavLink
11511189
activeClassName="active"
11521190
aria-current="page"

app/actions/accountActions.js

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import transactionHistoryActions from './transactionHistoryActions'
88
import { getPendingTransactionInfo } from './pendingTransactionActions'
99
import { blockHeightActions } from './blockHeightActions'
1010
import nftActions from './nftActions'
11+
import nftGalleryActions from './nftGalleryActions'
1112

1213
export const ID = 'account'
1314

@@ -19,4 +20,5 @@ export default createBatchActions(ID, {
1920
prices: pricesActions,
2021
blockHeight: blockHeightActions,
2122
nft: nftActions,
23+
nftGallery: nftGalleryActions,
2224
})

app/actions/dashboardActions.js

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import priceHistoryActions from './priceHistoryActions'
88
import { blockHeightActions } from './blockHeightActions'
99
import { getPendingTransactionInfo } from './pendingTransactionActions'
1010
import nftActions from './nftActions'
11+
import nftGalleryActions from './nftGalleryActions'
1112

1213
export const ID = 'dashboard'
1314

@@ -19,4 +20,5 @@ export default createBatchActions(ID, {
1920
priceHistory: priceHistoryActions,
2021
getPendingTransactionInfo,
2122
height: blockHeightActions,
23+
nftGallery: nftGalleryActions,
2224
})

app/actions/nftGalleryActions.js

+94
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,94 @@
1+
// @flow
2+
import axios from 'axios'
3+
import { createActions } from 'spunky'
4+
5+
export const ID = 'nft-gallery'
6+
7+
const GHOST_MARKET = 'GHOST_MARKET'
8+
const NFT_PROVIDER = GHOST_MARKET
9+
10+
export type NftGalleryItem = {
11+
metadata: {
12+
attributes: any,
13+
description: string,
14+
image: string,
15+
media_type: string,
16+
name: string,
17+
},
18+
series: {
19+
chain: string,
20+
contrract: string,
21+
creator: string,
22+
current_supply: number,
23+
description: string,
24+
},
25+
tokenId: string,
26+
contract: string,
27+
collection: {
28+
name: string,
29+
},
30+
}
31+
32+
export type NftGalleryResults = {
33+
results: NftGalleryItem[],
34+
count: number,
35+
page: number,
36+
}
37+
38+
const DEFAULT_NFT_GALLERY_RESULTS = { results: [], page: 0, count: 0 }
39+
40+
export async function parseGhostMarketResults({
41+
address,
42+
page = 0,
43+
previousResults = [],
44+
}: {
45+
address: string,
46+
page: number,
47+
previousResults: NftGalleryItem[],
48+
}): Promise<NftGalleryResults> {
49+
try {
50+
const LIMIT = 8
51+
const OFFSET = LIMIT * page
52+
53+
const response = await axios.get(
54+
`https://api.ghostmarket.io/api/v1/assets?chain=n3&owner=${address}&limit=${LIMIT}&offset=${OFFSET}&with_total=1`,
55+
)
56+
57+
const items = response?.data?.assets ?? []
58+
const count = response?.data?.total_results ?? 0
59+
if (items.length) {
60+
const results = items.map(asset => {
61+
const parsed = {
62+
metadata: asset.nft.nft_metadata,
63+
series: asset.nft.series,
64+
tokenId: asset.nft.token_id,
65+
contract: asset.nft.contract,
66+
collection: asset.nft.collection,
67+
}
68+
return parsed
69+
})
70+
71+
return { results: previousResults.concat(results), page, count }
72+
}
73+
74+
return { results: previousResults, page: 0, count }
75+
} catch (e) {
76+
console.error('An error occurred fetching data for NFT gallery', { e })
77+
return DEFAULT_NFT_GALLERY_RESULTS
78+
}
79+
}
80+
81+
export default createActions(
82+
ID,
83+
({ address, page, previousResults }) => async (): Promise<
84+
NftGalleryResults,
85+
> => {
86+
switch (true) {
87+
case NFT_PROVIDER === GHOST_MARKET:
88+
return parseGhostMarketResults({ address, page, previousResults })
89+
90+
default:
91+
return parseGhostMarketResults(address)
92+
}
93+
},
94+
)

app/actions/pendingTransactionActions.js

+3-2
Original file line numberDiff line numberDiff line change
@@ -23,19 +23,20 @@ export const parseContractTransaction = async (
2323
chain: string,
2424
): Promise<Array<ParsedPendingTransaction>> => {
2525
const parsedData = []
26-
// eslint-disable-next-line camelcase
26+
2727
const {
2828
confirmations,
2929
txid,
3030
net_fee, // eslint-disable-line camelcase
3131
blocktime = 0,
32+
hash = '',
3233
sendEntries,
3334
} = transaction
3435

3536
for (const send of sendEntries) {
3637
parsedData.push({
3738
confirmations,
38-
txid: txid.substring(2),
39+
txid: txid ? txid.substring(2) : hash.substring(2),
3940
net_fee,
4041
blocktime,
4142
amount: toBigNumber(send.amount).toString(),

app/assets/navigation/canvas.svg

+3
Loading

app/components/Blockchain/Transaction/N3NEP11ReceiveAbstract.jsx

+7-5
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,9 @@
11
// @flow
22
import React from 'react'
33
import classNames from 'classnames'
4-
import { FormattedMessage, IntlShape, injectIntl } from 'react-intl'
5-
import Button from '../../Button'
4+
import { injectIntl } from 'react-intl'
65
import styles from './Transaction.scss'
76
import ReceiveIcon from '../../../assets/icons/receive-tx.svg'
8-
import ContactsAdd from '../../../assets/icons/contacts-add.svg'
9-
import CopyToClipboard from '../../CopyToClipboard'
107

118
type Props = {
129
image: string,
@@ -21,7 +18,12 @@ class N3NEP11ReceiveAbstract extends React.Component<Props> {
2118
render = () => {
2219
const { image, isPending, tokenName, symbol, txDate } = this.props
2320

24-
const logo = image && <img src={image} alt={`${symbol}`} />
21+
const logo = image && (
22+
<img
23+
src={image.replace('ipfs://', 'https://ipfs.io/ipfs/')}
24+
alt={`${symbol}`}
25+
/>
26+
)
2527

2628
return (
2729
<div className={classNames(styles.transactionContainerN3)}>

app/components/Blockchain/Transaction/N3NEP11SendAbstract.jsx

+6-1
Original file line numberDiff line numberDiff line change
@@ -35,7 +35,12 @@ class N3NEP11SendAbstract extends React.Component<Props> {
3535
symbol,
3636
txDate,
3737
} = this.props
38-
const logo = image && <img src={image} alt={`${symbol}`} />
38+
const logo = image && (
39+
<img
40+
src={image.replace('ipfs://', 'https://ipfs.io/ipfs/')}
41+
alt={`${symbol}`}
42+
/>
43+
)
3944
const contactTo = to && findContact(to)
4045
const contactToExists = contactTo !== to
4146
return (

app/components/Blockchain/Transaction/N3NEP17ReceiveAbstract.jsx

+6-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,12 @@ class N3NEP17ReceiveAbstract extends React.Component<Props> {
3333
symbol,
3434
txDate,
3535
} = this.props
36-
const logo = image && <img src={image} alt={`${symbol}`} />
36+
const logo = image && (
37+
<img
38+
src={image.replace('ipfs://', 'https://ipfs.io/ipfs/')}
39+
alt={`${symbol}`}
40+
/>
41+
)
3742
const contactFrom = from && findContact(from)
3843
const contactFromExists = contactFrom !== from
3944
return (

app/components/Blockchain/Transaction/N3NEP17SendAbstract.jsx

+6-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,12 @@ class N3NEP17SendAbstract extends React.Component<Props> {
3333
symbol,
3434
txDate,
3535
} = this.props
36-
const logo = image && <img src={image} alt={`${symbol}`} />
36+
const logo = image && (
37+
<img
38+
src={image.replace('ipfs://', 'https://ipfs.io/ipfs/')}
39+
alt={`${symbol}`}
40+
/>
41+
)
3742
const contactTo = findContact(to)
3843
const contactToExists = contactTo !== to
3944
return (

app/components/ConnectDapp/ConnectionError.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ const ConnectionError = () => {
3636
<p>
3737
{typeof error === 'string'
3838
? error
39-
: 'An unkown error occurred please try again.'}
39+
: 'An unknown error occurred please try again.'}
4040
</p>
4141
<br />
4242
<br />

app/components/Contacts/ContactsPanel/ContactsPanel.scss

+1-3
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,7 @@
11
@import '../../../styles/variables';
22

33
.contactsPanel {
4-
width: 100%;
5-
height: 100%;
6-
overflow: scroll;
4+
overflow-y: auto;
75

86
.name {
97
width: 180px;

app/components/ErrorBoundaries/Main/Main.jsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ export default class ErrorBoundary extends Component<Props, State> {
2828
}
2929

3030
componentDidCatch(error: Error, info: any) {
31-
console.warn('An unkown error has occurred!', { error, info })
31+
console.warn('An unknown error has occurred!', { error, info })
3232
this.setState({ hasError: true })
3333
}
3434

0 commit comments

Comments
 (0)