Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Trello 19 #31

Open
wants to merge 11 commits into
base: master
Choose a base branch
from
2 changes: 2 additions & 0 deletions src/App.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,13 +4,15 @@ import './App.scss';
import BaseLayout from './components/layouts/BaseLayout';
import Ipfs from './pages/ipfs/Ipfs';
import IpfsList from './pages/ipfs/IpfsList';
import TrackedContractsList from './pages/tracked-contracts/TrackedContractsList';

const App = (): JSX.Element => {
return (
<Router>
<BaseLayout>
<Switch>
<Route path="/ipfs/:hash" component={Ipfs} />
<Route path="/tracked-contracts" component={TrackedContractsList} />
<Route path={['/', '/ipfs']} component={IpfsList} />
</Switch>
</BaseLayout>
Expand Down
7 changes: 7 additions & 0 deletions src/components/common/Card/Card.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import React from 'react';

const Card: React.FC<{ className?: string }> = ({ className = '', children }) => {
return <div className={`inline-block bg-light shadow-lg overflow-hidden sm:rounded-lg ${className}`}>{children}</div>;
};

export default Card;
7 changes: 7 additions & 0 deletions src/components/common/Card/CardBody.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import React from 'react';

const CardBody: React.FC<{}> = ({ children }) => {
return <div className="px-4 py-5 sm:px-6">{children}</div>;
};

export default CardBody;
7 changes: 7 additions & 0 deletions src/components/common/Card/CardFooter.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import React from 'react';

const CardFooter: React.FC<{}> = ({ children }) => {
return <div className="px-4 py-3 bg-gray-50 text-right sm:px-6">{children}</div>;
};

export default CardFooter;
7 changes: 7 additions & 0 deletions src/components/common/Card/CardHeader.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import React from 'react';

const CardHeader: React.FC<{}> = ({ children }) => {
return <div className="mx-4 px-4 py-3 text-left text-2xl border-b border-gray-200">{children}</div>;
};

export default CardHeader;
4 changes: 4 additions & 0 deletions src/components/common/Card/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
export { default as Card } from './Card';
export { default as CardBody } from './CardBody';
export { default as CardFooter } from './CardFooter';
export { default as CardHeader } from './CardHeader';
46 changes: 46 additions & 0 deletions src/components/common/Input/SearchInput.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,46 @@
import React from 'react';
import { FaSearch, FaTimes } from 'react-icons/fa';

interface SearchInput {
value: any;
onChange: (value: string) => void;
name?: string;
id?: string;
className?: string;
placeholder?: string;
}

const SearchInput: React.FC<SearchInput> = ({
value,
onChange,
name = 'search',
id = 'search',
className = '',
placeholder = '',
}): JSX.Element => {
return (
<div className="mt-1 relative">
<div className="absolute inset-y-0 right-0 p-3 text-gray-600 flex items-center">
{/** TODO: Remove this and actually use the other icons library once we get the query manager merged in */}
{value ? (
<button type="button" onClick={() => onChange('')}>
<FaTimes size={12} />
</button>
) : (
<FaSearch size={12} />
)}
</div>
<input
type="text"
name={name}
id={id}
value={value}
placeholder={placeholder}
className={`focus:outline-none focus:ring-2 focus:ring-opacity-20 focus:ring-primary block w-full pl-7 pr-12 sm:text-sm border-gray-200 bg-gray-200 text-gray-600 rounded-full ${className}`}
onChange={(event) => onChange(event.target.value)}
/>
</div>
);
};

export default SearchInput;
3 changes: 3 additions & 0 deletions src/components/common/Input/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
import SearchInput from './SearchInput';

export default SearchInput;
7 changes: 7 additions & 0 deletions src/components/common/Table/Table.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import React from 'react';

const Table: React.FC<{ className?: string }> = ({ className = '', children }): JSX.Element => {
return <table className={`min-w-full divide-y divide-gray-200 ${className}`}>{children}</table>;
};

export default Table;
7 changes: 7 additions & 0 deletions src/components/common/Table/TableBody.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import React from 'react';

const TableBody: React.FC<{ className?: string }> = ({ className = '', children }): JSX.Element => {
return <tbody className={`bg-white divide-y divide-gray-200 ${className}`}>{children}</tbody>;
};

export default TableBody;
7 changes: 7 additions & 0 deletions src/components/common/Table/TableCell.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import React from 'react';

const TableCell: React.FC<{ className?: string }> = ({ className = '', children }): JSX.Element => {
return <td className={`px-6 py-4 whitespace-nowrap ${className}`}>{children}</td>;
};

export default TableCell;
7 changes: 7 additions & 0 deletions src/components/common/Table/TableHeader.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import React from 'react';

const TableHeader: React.FC<{ className?: string }> = ({ className = '', children }): JSX.Element => {
return <thead className={`bg-gray-50 ${className}`}>{children}</thead>;
};

export default TableHeader;
14 changes: 14 additions & 0 deletions src/components/common/Table/TableHeaderCell.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
import React from 'react';

const TableHeaderCell: React.FC<{ className?: string }> = ({ className = '', children }): JSX.Element => {
return (
<th
scope="col"
className={`px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider ${className}`}
>
{children}
</th>
);
};

export default TableHeaderCell;
7 changes: 7 additions & 0 deletions src/components/common/Table/TableRow.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
import React from 'react';

const TableRow: React.FC<{ className?: string }> = ({ className = '', children }): JSX.Element => {
return <tr className={className}>{children}</tr>;
};

export default TableRow;
6 changes: 6 additions & 0 deletions src/components/common/Table/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
export { default as Table } from './Table';
export { default as TableBody } from './TableBody';
export { default as TableCell } from './TableCell';
export { default as TableHeader } from './TableHeader';
export { default as TableHeaderCell } from './TableHeaderCell';
export { default as TableRow } from './TableRow';
2 changes: 2 additions & 0 deletions src/components/common/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
export * from './Card';
export * from './Table';
3 changes: 3 additions & 0 deletions src/components/layouts/NavigationBar/NavigationBar.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,9 @@ const NavigationBar = (): JSX.Element => {
<li className="navigation-bar__link">
<Link to="/ipfs">IPFS</Link>
</li>
<li className="navigation-bar__link">
<Link to="/tracked-contracts">Tracked Contracts</Link>
</li>
</ul>
</div>
</div>
Expand Down
8 changes: 3 additions & 5 deletions src/pages/ipfs/Ipfs.tsx
Original file line number Diff line number Diff line change
@@ -1,16 +1,14 @@
import React, { ReactNode } from 'react';
import { Link } from 'react-router-dom';
import { Button, Grid, Header, Message, Segment } from 'semantic-ui-react';

import ace from 'brace';
import 'brace/mode/json';
import 'brace/theme/github';

// No typings for jsoneditor-react for now just using @ts-ignore for getting around the issue
// Later on with advanced design we might not used that library. If we do we can as well define typings for it
// @ts-ignore
import { JsonEditor as Editor } from 'jsoneditor-react';
import 'jsoneditor-react/es/editor.min.css';
import React, { ReactNode } from 'react';
import { Link } from 'react-router-dom';
import { Button, Grid, Header, Message, Segment } from 'semantic-ui-react';

interface IpfsProps {
match: { params: { hash: string } };
Expand Down
23 changes: 23 additions & 0 deletions src/pages/tracked-contracts/TrackedChain.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
import React from 'react';

interface TrackedChainProps {
name: string;
active: boolean;
onClick: () => void;
}

const TrackedChain: React.FC<TrackedChainProps> = ({ name, active, onClick }) => {
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Moved it here, in the same folder, I don't think it's gonna be used anywhere else so no need to put it inside components

return (
<div
className={`focus:outline-none py-4 capitalize cursor-pointer ${active ? 'font-bold text-primary' : ''}`}
role="button"
onClick={() => onClick()}
onKeyPress={() => onClick()}
tabIndex={0}
>
{name}
</div>
);
};

export default TrackedChain;
74 changes: 74 additions & 0 deletions src/pages/tracked-contracts/TrackedContractsList.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,74 @@
import React, { useEffect, useState } from 'react';
import { Card, CardBody, CardHeader } from '../../components/common';
import SearchInput from '../../components/common/Input';
import mockDataChain from './mock-data';
import TrackedChain from './TrackedChain';

type ChainType = 'evm' | 'substrate';

interface Chain {
name: string;
type: ChainType;
project: string;
url: string;
credentials: {
// TODO: Ask potentially BE to have structure camelCase in case or we can just disable the warning in eslint
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've changed eslint settings in Query Builder to display a warning

/* eslint-disable camelcase */
private_key?: string;
public_key?: string;
};
tracked_contracts: string[];
}

const TrackedContractsList: React.FC<{}> = () => {
const [chains, setChains] = useState<Chain[]>([]);
const [activeChain, setActiveChain] = useState<Chain>();
const [searchChain, setSearchChain] = useState('');

useEffect(() => {
setChains(mockDataChain as Chain[]);
}, []);

// Filter out by the search but also keep the current active in the search
const allChains = chains
.filter((chain) => chain.name.includes(searchChain.trim().toLocaleLowerCase()) || activeChain?.name === chain.name)
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just this filter works actually no need to check for if defined or null. Also added the fact that if a chan is already active you might still want to see it despite searching

.map((chain) => (
<TrackedChain active={chain.name === activeChain?.name} name={chain.name} onClick={() => setActiveChain(chain)} />
));

const activeContracts =
activeChain && activeChain.tracked_contracts.length ? activeChain.tracked_contracts : 'No Tracked Contracts';

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This should also work

const activeContracts =
    activeChain && activeChain.tracked_contracts.length
      ? activeChain.tracked_contracts
      : 'No Tracked Contracts';

return (
<div className="container mx-auto">
<div className="flex">
<div className="flex-none mr-10 w-1/4 md:w-auto">
<Card>
<div className="mx-4 px-4 py-3 max-w-md">
<SearchInput
value={searchChain}
onChange={(val: string) => setSearchChain(val)}
placeholder="Search"
className="py-3 text-lg leading-relaxed max-w-full"
/>
</div>
<CardBody>
<div className="divide-y divide-gray-100 text-secondary">{allChains}</div>
</CardBody>
</Card>
</div>
<div className="flex-1">
<Card className="w-full">
<CardBody>
<CardHeader>Tracked Contracts</CardHeader>
{/* <Table></Table> */}
{activeContracts}
</CardBody>
</Card>
</div>
</div>
</div>
);
};

export default TrackedContractsList;
1 change: 1 addition & 0 deletions src/pages/tracked-contracts/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export { default as TrackedContractsList } from './TrackedContractsList';
33 changes: 33 additions & 0 deletions src/pages/tracked-contracts/mock-data.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
// We can remove this when we can hook to the BE and get that data it's just for getting the UX ready for it
const mockDataChain = [
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Just mocking the sample we got from Jure

{
name: 'eth.mainnet',
type: 'evm',
project: 'eth',
url: 'https://mainnet.infura.io/v3/<project_id>',
credentials: {},
tracked_contracts: [],
},
{
name: 'eth.dev-mainnet-fork',
type: 'evm',
project: 'eth',
url: 'ws://localhost:8545',
credentials: {
private_key: '<private_key>',
},
tracked_contracts: ['0x3194cBDC3dbcd3E11a07892e7bA5c3394048Cc87'],
},
{
name: 'dev-canvas',
type: 'substrate',
project: 'canvas',
url: 'ws://127.0.0.1:9944',
credentials: {
private_key: '<private_key>',
public_key: '<public_key>',
},
tracked_contracts: ['5FnDzvXcnu3RCtQ3f3RFqaQnVLfcWLZ9FvUURNoPMdmsKZXP'],
},
];
export default mockDataChain;
34 changes: 18 additions & 16 deletions tailwind.config.js
Original file line number Diff line number Diff line change
@@ -1,23 +1,25 @@
module.exports = {
theme: {
colors: {
primary: {
DEFAULT: 'var(--color-primary)',
light: 'var(--color-primary-light)',
extend: {
colors: {
primary: {
DEFAULT: 'var(--color-primary)',
light: 'var(--color-primary-light)',
},
secondary: {
DEFAULT: 'var(--color-secondary)',
light: 'var(--color-secondary-light)',
lightest: 'var(--color-secondary-lightest)',
darkest: 'var(--color-secondary-darkest)',
},
dark: 'var(--color-dark)',
light: 'var(--color-light)',
},
secondary: {
DEFAULT: 'var(--color-secondary)',
light: 'var(--color-secondary-light)',
lightest: 'var(--color-secondary-lightest)',
darkest: 'var(--color-secondary-darkest)',
fontFamily: {
sans: ['Poppins'],
serif: ['Georgia'],
mono: ['Roboto'],
},
dark: 'var(--color-dark)',
light: 'var(--color-light)',
},
fontFamily: {
sans: ['Poppins'],
serif: ['Georgia'],
mono: ['Roboto'],
},
},
purge: ['./src/**/*.{js,jsx,ts,tsx}', './public/index.html'],
Expand Down