import { Checkbox, Form, Input, SubmitButton, Select, VariableInput } from '../components/form' import * as Yup from 'yup' import { Alert, Button, InputGroup, Modal } from 'react-bootstrap' import LayoutCenter from '../components/layout-center' import { useState } from 'react' import { gql, useMutation, useQuery } from '@apollo/client' import { getGetServerSideProps } from '../api/ssrApollo' import LoginButton from '../components/login-button' import { signIn } from 'next-auth/client' import ModalButton from '../components/modal-button' import { LightningAuth, SlashtagsAuth } from '../components/lightning-auth' import { SETTINGS, SET_SETTINGS } from '../fragments/users' import { useRouter } from 'next/router' import Info from '../components/info' import { CURRENCY_SYMBOLS } from '../components/price' import Link from 'next/link' import AccordianItem from '../components/accordian-item' import { MAX_NOSTR_RELAY_NUM } from '../lib/constants' import { WS_REGEXP } from '../lib/url' import { bech32 } from 'bech32' export const getServerSideProps = getGetServerSideProps(SETTINGS) const supportedCurrencies = Object.keys(CURRENCY_SYMBOLS) const HEX64 = /^[0-9a-fA-F]{64}$/ const NPUB = /^npub1[02-9ac-hj-np-z]+$/ Yup.addMethod(Yup.string, 'or', function (schemas, msg) { return this.test({ name: 'or', message: msg, test: value => { if (Array.isArray(schemas) && schemas.length > 1) { const resee = schemas.map(schema => schema.isValidSync(value)) return resee.some(res => res) } else { throw new TypeError('Schemas is not correct array schema') } }, exclusive: false }) }) export const SettingsSchema = Yup.object({ tipDefault: Yup.number().typeError('must be a number').required('required') .positive('must be positive').integer('must be whole'), fiatCurrency: Yup.string().required('required').oneOf(supportedCurrencies), nostrPubkey: Yup.string() .or([ Yup.string().matches(HEX64, 'must be 64 hex chars'), Yup.string().matches(NPUB, 'invalid bech32 encoding')], 'invalid pubkey'), nostrRelays: Yup.array().of( Yup.string().matches(WS_REGEXP, 'invalid web socket address') ).max(MAX_NOSTR_RELAY_NUM, ({ max, value }) => `${Math.abs(max - value.length)} too many`) }) const warningMessage = 'If I logout, even accidentally, I will never be able to access my account again' export const WarningSchema = Yup.object({ warning: Yup.string().matches(warningMessage, 'does not match').required('required') }) function bech32encode (hexString) { return bech32.encode('npub', bech32.toWords(Buffer.from(hexString, 'hex'))) } export default function Settings ({ data: { settings } }) { const [success, setSuccess] = useState() const [setSettings] = useMutation(SET_SETTINGS, { update (cache, { data: { setSettings } }) { cache.modify({ id: 'ROOT_QUERY', fields: { settings () { return setSettings } } }) } } ) const { data } = useQuery(SETTINGS) if (data) { ({ settings } = data) } return (

settings

{ if (nostrPubkey.length === 0) { nostrPubkey = null } else { if (NPUB.test(nostrPubkey)) { const { words } = bech32.decode(nostrPubkey) nostrPubkey = Buffer.from(bech32.fromWords(words)).toString('hex') } } const nostrRelaysFiltered = nostrRelays?.filter(word => word.trim().length > 0) await setSettings({ variables: { tipDefault: Number(tipDefault), nostrPubkey, nostrRelays: nostrRelaysFiltered, ...values } }) setSuccess('settings saved') }} > {success && setSuccess(undefined)} dismissible>{success}} sats} hint={note: you can also press and hold the lightning bolt to tip custom amounts} />
advanced
} body={turbo tipping
  • Makes every additional bolt click raise your total tip to another 10x multiple of your default tip
  • e.g. if your tip default is 10 sats
    • 1st click: 10 sats total tipped
    • 2nd click: 100 sats total tipped
    • 3rd click: 1000 sats total tipped
    • 4th click: 10000 sats total tipped
    • and so on ...
  • You can still custom tip via long press
    • the next bolt click rounds up to the next greatest 10x multiple of your default
} />} /> pubkey optional} name='nostrPubkey' clear /> relays optional} name='nostrRelays' clear min={0} max={MAX_NOSTR_RELAY_NUM} /> } />
save
saturday newsletter
{settings?.authMethods && }
) } function AuthMethods ({ methods }) { const router = useRouter() const [unlinkAuth] = useMutation( gql` mutation unlinkAuth($authType: String!) { unlinkAuth(authType: $authType) { lightning email twitter github } }`, { update (cache, { data: { unlinkAuth } }) { cache.modify({ id: 'ROOT_QUERY', fields: { settings (existing) { return { ...existing, authMethods: { ...unlinkAuth } } } } }) } } ) const [obstacle, setObstacle] = useState() const providers = Object.keys(methods).filter(k => k !== '__typename') const unlink = async type => { // if there's only one auth method left const links = providers.reduce((t, p) => t + (methods[p] ? 1 : 0), 0) if (links === 1) { setObstacle(type) } else { await unlinkAuth({ variables: { authType: type } }) } } return ( <> setObstacle(null)} >
setObstacle(null)}>X
You are removing your last auth method. It is recommended you link another auth method before removing your last auth method. If you'd like to proceed anyway, type the following below
If I logout, even accidentally, I will never be able to access my account again
{ await unlinkAuth({ variables: { authType: obstacle } }) router.push('/settings') setObstacle(null) }} > do it
auth methods
{providers && providers.map(provider => { switch (provider) { case 'email': return methods.email ? (
) :
case 'lightning': return methods.lightning ? { await unlink('lightning') } } /> : ( }>
) case 'slashtags': return methods.slashtags ? { await unlink('slashtags') } } /> : ( }>
) default: return ( { if (methods[provider]) { await unlink(provider) } else { signIn(provider) } }} text={methods[provider] ? 'Unlink' : 'Link'} /> ) } })} ) } export const EmailSchema = Yup.object({ email: Yup.string().email('email is no good').required('required') }) export function EmailLinkForm ({ callbackUrl }) { const [linkUnverifiedEmail] = useMutation( gql` mutation linkUnverifiedEmail($email: String!) { linkUnverifiedEmail(email: $email) }` ) return (
{ // add email to user's account // then call signIn const { data } = await linkUnverifiedEmail({ variables: { email } }) if (data.linkUnverifiedEmail) { signIn('email', { email, callbackUrl }) } }} >
Link Email
) }