|
|
|
import React from 'react'
|
|
|
|
import PropTypes from 'prop-types'
|
|
|
|
import ReactModal from 'react-modal'
|
|
|
|
import { MdClose } from 'react-icons/lib/md'
|
|
|
|
import { FaCircle } from 'react-icons/lib/fa'
|
|
|
|
import styles from './ContactsForm.scss'
|
|
|
|
|
|
|
|
class ContactsForm extends React.Component {
|
|
|
|
constructor(props) {
|
|
|
|
super(props)
|
|
|
|
|
|
|
|
this.state = {
|
|
|
|
editing: false,
|
|
|
|
manualFormInput: ''
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
|
|
|
const {
|
|
|
|
contactsform,
|
|
|
|
closeContactsForm,
|
|
|
|
updateContactFormSearchQuery,
|
|
|
|
updateContactCapacity,
|
|
|
|
openChannel,
|
|
|
|
|
|
|
|
activeChannelPubkeys,
|
|
|
|
nonActiveChannelPubkeys,
|
|
|
|
pendingOpenChannelPubkeys,
|
|
|
|
filteredNetworkNodes,
|
|
|
|
loadingChannelPubkeys,
|
|
|
|
showManualForm
|
|
|
|
} = this.props
|
|
|
|
|
|
|
|
const { editing, manualFormInput } = this.state
|
|
|
|
|
|
|
|
|
|
|
|
const renderRightSide = (node) => {
|
|
|
|
if (loadingChannelPubkeys.includes(node.pub_key)) {
|
|
|
|
return (
|
|
|
|
<span className={styles.inactive}>
|
|
|
|
<div className={styles.loading}>
|
|
|
|
<div className={styles.spinner} />
|
|
|
|
</div>
|
|
|
|
</span>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (activeChannelPubkeys.includes(node.pub_key)) {
|
|
|
|
return (
|
|
|
|
<span className={`${styles.online} ${styles.inactive}`}>
|
|
|
|
<FaCircle style={{ verticalAlign: 'top' }} /> <span>Online</span>
|
|
|
|
</span>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (nonActiveChannelPubkeys.includes(node.pub_key)) {
|
|
|
|
return (
|
|
|
|
<span className={`${styles.offline} ${styles.inactive}`}>
|
|
|
|
<FaCircle style={{ verticalAlign: 'top' }} /> <span>Offline</span>
|
|
|
|
</span>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pendingOpenChannelPubkeys.includes(node.pub_key)) {
|
|
|
|
return (
|
|
|
|
<span className={`${styles.pending} ${styles.inactive}`}>
|
|
|
|
<FaCircle style={{ verticalAlign: 'top' }} /> <span>Pending</span>
|
|
|
|
</span>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!node.addresses.length) {
|
|
|
|
return (
|
|
|
|
<span className={`${styles.private} ${styles.inactive}`}>
|
|
|
|
Private
|
|
|
|
</span>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<span
|
|
|
|
className={`${styles.connect} hint--left`}
|
|
|
|
data-hint='Connect with 0.1 BTC'
|
|
|
|
onClick={() => openChannel({ pubkey: node.pub_key, host: node.addresses[0].addr, local_amt: contactsform.contactCapacity })}
|
|
|
|
>
|
|
|
|
Connect
|
|
|
|
</span>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
const inputClicked = () => {
|
|
|
|
if (editing) { return }
|
|
|
|
|
|
|
|
this.setState({ editing: true })
|
|
|
|
}
|
|
|
|
|
|
|
|
const manualFormSubmit = () => {
|
|
|
|
if (!manualFormInput.length) { return }
|
|
|
|
if (!manualFormInput.includes('@')) { return }
|
|
|
|
|
|
|
|
const [pubkey, host] = manualFormInput && manualFormInput.split('@')
|
|
|
|
|
|
|
|
openChannel({ pubkey, host, local_amt: contactsform.contactCapacity })
|
|
|
|
|
|
|
|
this.setState({ manualFormInput: '' })
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div>
|
|
|
|
<ReactModal
|
|
|
|
isOpen={contactsform.isOpen}
|
|
|
|
contentLabel='No Overlay Click Modal'
|
|
|
|
ariaHideApp
|
|
|
|
shouldCloseOnOverlayClick
|
|
|
|
onRequestClose={() => closeContactsForm}
|
|
|
|
parentSelector={() => document.body}
|
|
|
|
className={styles.modal}
|
|
|
|
>
|
|
|
|
<header>
|
|
|
|
<div>
|
|
|
|
<h1>Add Contact</h1>
|
|
|
|
</div>
|
|
|
|
<div onClick={closeContactsForm} className={styles.modalClose}>
|
|
|
|
<MdClose />
|
|
|
|
</div>
|
|
|
|
</header>
|
|
|
|
|
|
|
|
<div className={styles.form}>
|
|
|
|
<div className={styles.search}>
|
|
|
|
<input
|
|
|
|
type='text'
|
|
|
|
placeholder='Find contact by alias or pubkey'
|
|
|
|
className={styles.searchInput}
|
|
|
|
value={contactsform.searchQuery}
|
|
|
|
onChange={event => updateContactFormSearchQuery(event.target.value)}
|
|
|
|
autoFocus
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<ul className={styles.networkResults}>
|
|
|
|
{
|
|
|
|
contactsform.searchQuery.length > 0 && filteredNetworkNodes.map(node => {
|
|
|
|
return (
|
|
|
|
<li key={node.pub_key}>
|
|
|
|
<section>
|
|
|
|
{
|
|
|
|
node.alias.length > 0 ?
|
|
|
|
<h2>
|
|
|
|
<span>{node.alias.trim()}</span>
|
|
|
|
<span>({node.pub_key.substr(0, 10)}...{node.pub_key.substr(node.pub_key.length - 10)})</span>
|
|
|
|
</h2>
|
|
|
|
:
|
|
|
|
<h2>
|
|
|
|
<span>{node.pub_key}</span>
|
|
|
|
</h2>
|
|
|
|
}
|
|
|
|
</section>
|
|
|
|
<section>
|
|
|
|
{renderRightSide(node)}
|
|
|
|
</section>
|
|
|
|
</li>
|
|
|
|
)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
{
|
|
|
|
showManualForm &&
|
|
|
|
<div className={styles.manualForm}>
|
|
|
|
<h2>Hm, looks like we can't see that contact from here. Want to try and manually connect?</h2>
|
|
|
|
<section>
|
|
|
|
<input
|
|
|
|
type='text'
|
|
|
|
placeholder='pubkey@host'
|
|
|
|
value={manualFormInput}
|
|
|
|
onChange={event => this.setState({ manualFormInput: event.target.value })}
|
|
|
|
/>
|
|
|
|
<div className={styles.submit} onClick={manualFormSubmit}>Submit</div>
|
|
|
|
</section>
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
|
|
|
|
<footer className={styles.footer}>
|
|
|
|
<div>
|
|
|
|
<span className={styles.amount}>
|
|
|
|
<input
|
|
|
|
type='text'
|
|
|
|
value={contactsform.contactCapacity}
|
|
|
|
onChange={event => updateContactCapacity(event.target.value)}
|
|
|
|
onClick={inputClicked}
|
|
|
|
onKeyPress={event => event.charCode === 13 && this.setState({ editing: false })}
|
|
|
|
readOnly={!editing}
|
|
|
|
style={{ width: `${editing ? 20 : contactsform.contactCapacity.toString().length + 1}%` }}
|
|
|
|
/>
|
|
|
|
</span>
|
|
|
|
<span className={styles.caption}>
|
|
|
|
BTC per contact
|
|
|
|
</span>
|
|
|
|
</div>
|
|
|
|
</footer>
|
|
|
|
</ReactModal>
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ContactsForm.propTypes = {
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
export default ContactsForm
|