2024-02-09 15:42:26 +00:00
|
|
|
import { createContext, useCallback, useContext, useEffect, useState } from 'react'
|
2024-02-08 18:33:13 +00:00
|
|
|
import { LNbitsProvider, useLNbits } from './lnbits'
|
|
|
|
import { NWCProvider, useNWC } from './nwc'
|
2024-03-20 00:37:31 +00:00
|
|
|
import { useToast, withToastFlow } from '@/components/toast'
|
2024-02-08 18:33:13 +00:00
|
|
|
import { gql, useMutation } from '@apollo/client'
|
|
|
|
|
|
|
|
const WebLNContext = createContext({})
|
|
|
|
|
2024-02-09 15:42:26 +00:00
|
|
|
const syncProvider = (array, provider) => {
|
|
|
|
const idx = array.findIndex(({ name }) => provider.name === name)
|
|
|
|
if (idx === -1) {
|
|
|
|
// add provider to end if enabled
|
|
|
|
return provider.enabled ? [...array, provider] : array
|
2024-02-08 18:33:13 +00:00
|
|
|
}
|
2024-02-09 15:42:26 +00:00
|
|
|
return [
|
|
|
|
...array.slice(0, idx),
|
|
|
|
// remove provider if not enabled
|
|
|
|
...provider.enabled ? [provider] : [],
|
|
|
|
...array.slice(idx + 1)
|
|
|
|
]
|
2024-02-08 18:33:13 +00:00
|
|
|
}
|
|
|
|
|
2024-02-09 15:42:26 +00:00
|
|
|
const storageKey = 'webln:providers'
|
2024-02-08 18:33:13 +00:00
|
|
|
|
|
|
|
function RawWebLNProvider ({ children }) {
|
|
|
|
const lnbits = useLNbits()
|
|
|
|
const nwc = useNWC()
|
2024-02-09 15:42:26 +00:00
|
|
|
const availableProviders = [lnbits, nwc]
|
|
|
|
const [enabledProviders, setEnabledProviders] = useState([])
|
2024-02-08 18:33:13 +00:00
|
|
|
|
2024-02-09 15:42:26 +00:00
|
|
|
// restore order on page reload
|
|
|
|
useEffect(() => {
|
|
|
|
const storedOrder = window.localStorage.getItem(storageKey)
|
|
|
|
if (!storedOrder) return
|
|
|
|
const providerNames = JSON.parse(storedOrder)
|
|
|
|
setEnabledProviders(providers => {
|
|
|
|
return providerNames.map(name => {
|
|
|
|
for (const p of availableProviders) {
|
|
|
|
if (p.name === name) return p
|
|
|
|
}
|
|
|
|
console.warn(`Stored provider with name ${name} not available`)
|
|
|
|
return null
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}, [])
|
|
|
|
|
|
|
|
// keep list in sync with underlying providers
|
|
|
|
useEffect(() => {
|
|
|
|
setEnabledProviders(providers => {
|
|
|
|
// Sync existing provider state with new provider state
|
|
|
|
// in the list while keeping the order they are in.
|
|
|
|
// If provider does not exist but is enabled, it is just added to the end of the list.
|
|
|
|
// This can be the case if we're syncing from a page reload
|
|
|
|
// where the providers are initially not enabled.
|
|
|
|
// If provider is no longer enabled, it is removed from the list.
|
|
|
|
const isInitialized = p => p.initialized
|
|
|
|
const newProviders = availableProviders.filter(isInitialized).reduce(syncProvider, providers)
|
|
|
|
const newOrder = newProviders.map(({ name }) => name)
|
|
|
|
window.localStorage.setItem(storageKey, JSON.stringify(newOrder))
|
|
|
|
return newProviders
|
|
|
|
})
|
|
|
|
}, [lnbits, nwc])
|
|
|
|
|
|
|
|
// sanity check
|
|
|
|
for (const p of enabledProviders) {
|
|
|
|
if (!p.enabled && p.initialized) {
|
|
|
|
console.warn('Expected provider to be enabled but is not:', p.name)
|
|
|
|
}
|
2024-02-08 18:33:13 +00:00
|
|
|
}
|
2024-02-09 15:42:26 +00:00
|
|
|
|
|
|
|
// first provider in list is the default provider
|
|
|
|
// TODO: implement fallbacks via provider priority
|
|
|
|
const provider = enabledProviders[0]
|
2024-02-08 18:33:13 +00:00
|
|
|
|
|
|
|
const toaster = useToast()
|
|
|
|
const [cancelInvoice] = useMutation(gql`
|
|
|
|
mutation cancelInvoice($hash: String!, $hmac: String!) {
|
|
|
|
cancelInvoice(hash: $hash, hmac: $hmac) {
|
|
|
|
id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
`)
|
|
|
|
|
2024-02-20 01:03:30 +00:00
|
|
|
const sendPaymentWithToast = withToastFlow(toaster)(
|
2024-02-24 20:33:08 +00:00
|
|
|
({ bolt11, hash, hmac, expiresAt, flowId }) => {
|
|
|
|
const expiresIn = (+new Date(expiresAt)) - (+new Date())
|
2024-02-20 01:03:30 +00:00
|
|
|
return {
|
2024-02-22 00:48:42 +00:00
|
|
|
flowId: flowId || hash,
|
2024-02-20 01:03:30 +00:00
|
|
|
type: 'payment',
|
|
|
|
onPending: () => provider.sendPayment(bolt11),
|
|
|
|
// hash and hmac are only passed for JIT invoices
|
2024-02-24 20:33:08 +00:00
|
|
|
onCancel: () => hash && hmac ? cancelInvoice({ variables: { hash, hmac } }) : undefined,
|
|
|
|
timeout: expiresIn
|
2024-02-08 18:33:13 +00:00
|
|
|
}
|
2024-02-20 01:03:30 +00:00
|
|
|
}
|
|
|
|
)
|
2024-02-08 18:33:13 +00:00
|
|
|
|
2024-02-09 15:42:26 +00:00
|
|
|
const setProvider = useCallback((defaultProvider) => {
|
|
|
|
// move provider to the start to set it as default
|
|
|
|
setEnabledProviders(providers => {
|
|
|
|
const idx = providers.findIndex(({ name }) => defaultProvider.name === name)
|
|
|
|
if (idx === -1) {
|
|
|
|
console.warn(`tried to set unenabled provider ${defaultProvider.name} as default`)
|
|
|
|
return providers
|
|
|
|
}
|
|
|
|
return [defaultProvider, ...providers.slice(0, idx), ...providers.slice(idx + 1)]
|
|
|
|
})
|
|
|
|
}, [setEnabledProviders])
|
|
|
|
|
|
|
|
const value = { provider: { ...provider, sendPayment: sendPaymentWithToast }, enabledProviders, setProvider }
|
2024-02-08 18:33:13 +00:00
|
|
|
return (
|
2024-02-09 15:42:26 +00:00
|
|
|
<WebLNContext.Provider value={value}>
|
2024-02-08 18:33:13 +00:00
|
|
|
{children}
|
|
|
|
</WebLNContext.Provider>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function WebLNProvider ({ children }) {
|
|
|
|
return (
|
|
|
|
<LNbitsProvider>
|
|
|
|
<NWCProvider>
|
|
|
|
<RawWebLNProvider>
|
|
|
|
{children}
|
|
|
|
</RawWebLNProvider>
|
|
|
|
</NWCProvider>
|
|
|
|
</LNbitsProvider>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
export function useWebLN () {
|
2024-02-09 15:42:26 +00:00
|
|
|
const { provider } = useContext(WebLNContext)
|
|
|
|
return provider
|
|
|
|
}
|
|
|
|
|
|
|
|
export function useWebLNConfigurator () {
|
2024-02-08 18:33:13 +00:00
|
|
|
return useContext(WebLNContext)
|
|
|
|
}
|