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

feat: add migration card in wallet screen #555

Merged
merged 6 commits into from
Aug 29, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
69 changes: 69 additions & 0 deletions frontend/src/components/TransferFundsButton.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,69 @@
import React from "react";
import { ButtonProps, LoadingButton } from "src/components/ui/loading-button";
import { useToast } from "src/components/ui/use-toast";
import { AlbyBalance, Channel } from "src/types";
import { request } from "src/utils/request";

type TransferFundsButtonProps = {
channels: Channel[] | undefined;
albyBalance: AlbyBalance;
reloadAlbyBalance: () => void;
} & ButtonProps;

export function TransferFundsButton({
channels,
albyBalance,
reloadAlbyBalance,
children,
...props
}: TransferFundsButtonProps) {
const [loading, setLoading] = React.useState(false);

const { toast } = useToast();

return (
<LoadingButton
loading={loading}
onClick={async () => {
if (!albyBalance) {
return;
}
if (
!channels?.some(
(channel) => channel.remoteBalance / 1000 > albyBalance.sats
)
) {
toast({
title: "Please increase your receiving capacity first",
});
return;
}
setLoading(true);
try {
await request("/api/alby/drain", {
method: "POST",
headers: {
"Content-Type": "application/json",
},
});
await reloadAlbyBalance();
toast({
title:
"🎉 Funds from Alby shared wallet transferred to your Alby Hub!",
});
} catch (error) {
console.error(error);
toast({
variant: "destructive",
description: "Something went wrong: " + error,
});
} finally {
setLoading(false);
}
}}
{...props}
>
{children}
</LoadingButton>
);
}
48 changes: 48 additions & 0 deletions frontend/src/components/icons/SelfCustodyIcon.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
import { createLucideIcon } from "lucide-react";

export const SelfCustodyIcon = createLucideIcon("RowIcon", [
[
"path",
{
d: "M29.8624 25.3804C20.5432 25.3804 12.6577 31.14 9.96325 39.1491C9.07335 41.72 10.5812 44.4144 13.1026 45.4526C18.4173 47.6032 24.1275 48.7156 29.8624 48.7156C35.5974 48.7156 41.2829 47.6032 46.6223 45.4526C49.1437 44.4144 50.6516 41.72 49.7617 39.1491C47.0672 31.14 39.157 25.3804 29.8624 25.3804ZM22.6444 41.72C19.9499 41.72 17.7746 39.9896 17.7746 37.839C17.7746 35.6884 19.9746 33.958 22.6444 33.958C25.3141 33.958 27.5141 35.6884 27.5141 37.839C27.5141 39.9896 25.3388 41.72 22.6444 41.72ZM36.5614 41.72C33.8917 41.72 31.7164 39.9896 31.7164 37.839C31.7164 35.6884 33.867 33.958 36.5614 33.958C39.2559 33.958 41.4312 35.6884 41.4312 37.839C41.4312 39.9896 39.2559 41.72 36.5614 41.72ZM48.4021 21.5983L50.6516 19.3488C52.5055 20.387 54.7797 20.3376 56.5842 19.2005C59.3281 17.4949 60.1438 13.9105 58.4382 11.1667C58.0332 10.5165 57.5039 9.95274 56.8806 9.50756C56.2574 9.06237 55.5524 8.74455 54.8061 8.57232C54.0597 8.4001 53.2868 8.37684 52.5314 8.50389C51.7761 8.63094 51.0533 8.90579 50.4044 9.3127C49.753 9.71638 49.1876 10.2445 48.7405 10.867C48.2935 11.4894 47.9736 12.1938 47.7991 12.94C47.6246 13.6862 47.5989 14.4595 47.7236 15.2156C47.8483 15.9718 48.1208 16.6959 48.5257 17.3465L46.2515 19.6207C41.6289 15.8387 35.9929 13.5892 29.8624 13.5892C23.732 13.5892 17.9724 15.8881 13.3004 19.7691L10.8284 17.2971C11.9408 15.4679 11.9655 13.1937 10.9026 11.3397C10.5215 10.6755 10.0133 10.0929 9.40685 9.62527C8.80045 9.15763 8.10782 8.81413 7.36856 8.61439C6.6293 8.41465 5.85792 8.3626 5.09853 8.46122C4.33913 8.55983 3.60663 8.80718 2.94291 9.18911C2.27734 9.56882 1.6931 10.0759 1.22356 10.6815C0.75402 11.287 0.408375 11.9792 0.206369 12.7183C0.00436257 13.4575 -0.0500464 14.2292 0.0462495 14.9894C0.142545 15.7496 0.387659 16.4833 0.767591 17.1488C1.53923 18.4947 2.81319 19.4795 4.31002 19.8873C5.80684 20.2951 7.40431 20.0926 8.75199 19.3241L11.1745 21.7466C6.40364 26.5422 3.01707 33.1423 1.70693 40.5334C1.01479 44.6369 3.11594 48.6661 6.87331 50.3965C14.0914 53.8325 21.9769 55.6123 29.9613 55.5876C37.8716 55.5876 45.6829 53.8572 52.8269 50.4954C54.423 49.7664 55.7754 48.5931 56.7223 47.1158C57.6692 45.6385 58.1705 43.9197 58.1663 42.1649C58.1663 41.6705 58.1416 41.1514 58.0427 40.6323C56.7078 33.1176 53.2718 26.4186 48.4021 21.5983ZM51.5909 47.8751C44.8278 51.0663 37.4395 52.7128 29.9613 52.6954C22.3972 52.7202 14.9319 51.0392 8.134 47.7762C6.88628 47.2138 5.86014 46.2537 5.21608 45.0461C4.57202 43.8385 4.34636 42.4515 4.57439 41.102C7.1205 26.8635 17.6263 16.5308 29.8624 16.5308C42.0986 16.5308 52.6538 26.913 55.1752 41.1761C55.6449 43.9447 54.1864 46.6639 51.5909 47.8751Z",
fill: "currentColor",
},
],
[
"path",
{
d: "M104.44 22.7127C104.857 22.2961 105.532 22.2961 105.949 22.7127L114.482 31.246C114.899 31.6626 114.899 32.338 114.482 32.7545L105.949 41.288C105.532 41.7044 104.857 41.7044 104.44 41.288C104.024 40.8713 104.024 40.1959 104.44 39.7793L111.153 33.067H92.3947C91.8056 33.067 91.328 32.5894 91.328 32.0003C91.328 31.4112 91.8056 30.9336 92.3947 30.9336H111.153L104.44 24.2212C104.024 23.8046 104.024 23.1293 104.44 22.7127Z",
fill: "currentColor",
fillRule: "evenodd",
clipRule: "evenodd",
},
],
[
"path",
{
d: "M193.696 28.5841C193.459 27.5279 193.107 26.5165 192.659 25.5564C192.16 24.481 191.539 23.4696 190.809 22.5414C190.054 21.5813 189.19 20.7171 188.231 19.9618C187.303 19.232 186.292 18.6111 185.216 18.1118C184.257 17.6638 183.245 17.3181 182.19 17.0749C181.095 16.8188 179.95 16.6908 178.779 16.6908C177.608 16.6908 176.462 16.8188 175.368 17.0749C175.208 17.8622 175.054 18.6879 174.92 19.5521C174.67 21.0884 174.465 22.7463 174.306 24.5322C174.229 25.4283 174.165 26.3629 174.114 27.3295C174.037 28.8081 173.992 30.3636 173.992 32.0023C173.992 33.641 174.037 35.1901 174.114 36.6751C175.592 36.752 177.147 36.7968 178.785 36.7968C180.423 36.7968 181.972 36.752 183.457 36.6751C184.417 36.6239 185.351 36.5599 186.247 36.4831C188.039 36.3231 189.696 36.1182 191.232 35.8686C192.096 35.7278 192.921 35.5805 193.708 35.4205C193.958 34.3259 194.092 33.1801 194.092 32.0087C194.092 30.8373 193.964 29.6915 193.708 28.5969L193.696 28.5841ZM189.261 27.4255C188.87 27.4831 188.493 27.5791 188.122 27.7071C187.456 27.9376 186.842 28.2769 186.285 28.7057C184.896 29.7619 183.898 31.3494 183.61 33.0585C183.61 33.0905 183.578 33.1033 183.559 33.0969C183.54 33.0969 183.521 33.0777 183.521 33.0585C183.066 30.146 180.711 27.8736 177.87 27.4191C177.819 27.4127 177.819 27.3359 177.87 27.3295C178.068 27.2975 178.26 27.2591 178.452 27.2079C180.052 26.8046 181.441 25.8124 182.369 24.4938C182.669 24.0713 182.919 23.6168 183.117 23.1303C183.309 22.6631 183.45 22.1702 183.527 21.6645C183.533 21.6133 183.61 21.6133 183.617 21.6645C183.789 22.7655 184.237 23.7705 184.871 24.6282C185.236 25.1211 185.658 25.5564 186.138 25.934C187.034 26.6446 188.103 27.1374 189.261 27.3359C189.312 27.3423 189.312 27.4191 189.261 27.4255Z",
fill: "currentColor",
},
],
[
"path",
{
d: "M210.795 30.9087L210.769 33.0915C210.769 36.3561 207.256 39.3262 200.882 41.4514C198.636 42.2003 196.108 42.8212 193.388 43.3077C193.19 43.3461 192.992 43.3781 192.793 43.4165C192.051 43.5382 191.302 43.6534 190.528 43.7558C189.773 43.8582 189.005 43.9478 188.231 44.031C187.322 44.1271 186.407 44.2103 185.472 44.2807C184.538 44.3447 183.597 44.4023 182.637 44.4407C181.364 44.4919 180.078 44.5175 178.779 44.5175C177.48 44.5175 176.187 44.4919 174.92 44.4407C173.96 44.3959 173.019 44.3447 172.085 44.2807C171.151 44.2103 170.236 44.1335 169.333 44.031C168.552 43.9478 167.791 43.8582 167.036 43.7558C166.268 43.6534 165.513 43.5382 164.777 43.4165C164.572 43.3845 164.374 43.3461 164.175 43.3077C161.456 42.8276 158.928 42.2067 156.682 41.4514C150.308 39.3262 146.795 36.3561 146.795 33.0915V30.8959C146.795 27.6313 150.308 24.6612 156.682 22.536C158.928 21.7871 161.456 21.1662 164.175 20.6797C164.143 20.9486 164.111 21.211 164.079 21.4862C163.913 22.8881 163.785 24.3155 163.695 25.775C162.991 25.9158 162.313 26.0567 161.654 26.2167C155.447 27.6761 151.345 29.8397 150.282 31.9969C151.345 34.1541 155.447 36.3177 161.654 37.7707C162.313 37.9244 162.991 38.0716 163.695 38.2124C164.092 38.2892 164.495 38.366 164.911 38.4365C165.436 38.5261 165.967 38.6157 166.511 38.6925C168.092 38.9293 169.762 39.1278 171.515 39.2686C172.431 39.3454 173.365 39.4094 174.318 39.4542C175.765 39.5246 177.262 39.5631 178.798 39.5631C180.334 39.5631 181.825 39.5246 183.277 39.4542C184.231 39.4094 185.165 39.3454 186.074 39.2686C187.827 39.1214 189.498 38.9293 191.078 38.6925C191.622 38.6093 192.16 38.5261 192.678 38.4365C193.088 38.366 193.491 38.2892 193.894 38.2124C194.598 38.078 195.276 37.9308 195.935 37.7771C202.143 36.3177 206.245 34.1541 207.307 31.9969C206.245 29.8397 202.143 27.6761 195.935 26.2231C195.359 24.514 194.54 22.9265 193.51 21.4926C193.286 21.179 193.056 20.8781 192.812 20.5837C193.011 20.6157 193.216 20.6541 193.414 20.6925C196.134 21.1726 198.661 21.7935 200.908 22.5488C207.281 24.674 210.795 27.6441 210.795 30.9087Z",
fill: "currentColor",
},
],
[
"path",
{
d: "M190.182 17.9744C189.888 17.7311 189.587 17.5007 189.274 17.2767C187.84 16.2461 186.247 15.4267 184.545 14.8506C183.092 8.64153 180.929 4.5384 178.772 3.47582C176.616 4.5384 174.453 8.64153 173 14.8506C172.84 15.51 172.699 16.1885 172.559 16.8926C172.482 17.2895 172.405 17.6991 172.335 18.1088C172.245 18.6337 172.155 19.165 172.079 19.7091C171.842 21.2902 171.643 22.9609 171.503 24.7148C171.426 25.6237 171.362 26.5583 171.317 27.5121C171.247 28.9587 171.208 30.4566 171.208 31.9929C171.208 33.5291 171.247 35.0206 171.317 36.4737C169.525 36.3136 167.868 36.1088 166.338 35.8592C166.287 34.5853 166.262 33.2987 166.262 31.9993C166.262 30.6998 166.287 29.4068 166.338 28.1394C166.383 27.1792 166.434 26.2383 166.505 25.3037C166.569 24.3691 166.652 23.4474 166.748 22.5512C166.831 21.7703 166.921 21.0021 167.023 20.2468C167.125 19.4787 167.241 18.7233 167.362 17.9872C167.394 17.7888 167.433 17.5839 167.471 17.3855C167.951 14.665 168.572 12.1366 169.327 9.88975C171.451 3.51422 174.421 0 177.684 0H179.879C183.143 0 186.112 3.51422 188.237 9.88975C188.986 12.1366 189.606 14.665 190.093 17.3855C190.131 17.5839 190.163 17.7888 190.202 17.9872L190.182 17.9744Z",
fill: "currentColor",
},
],
[
"path",
{
d: "M190.08 46.6151C189.601 49.3356 188.98 51.864 188.225 54.1108C186.1 60.4863 183.131 64.0006 179.867 64.0006H177.672C174.408 64.0006 171.439 60.4863 169.315 54.1108C168.566 51.864 167.945 49.3356 167.459 46.6151C167.728 46.6471 167.99 46.6791 168.265 46.7111C169.667 46.8775 171.094 47.0055 172.553 47.0952C172.693 47.7993 172.834 48.4778 172.994 49.1371C174.447 55.3462 176.61 59.4494 178.766 60.5119C180.923 59.4494 183.086 55.3462 184.545 49.1371C184.699 48.4778 184.846 47.7993 184.98 47.0952C186.439 46.9991 187.873 46.8711 189.268 46.7111C189.537 46.6855 189.805 46.6471 190.074 46.6151H190.08Z",
fill: "currentColor",
},
],
]);
21 changes: 20 additions & 1 deletion frontend/src/screens/Intro.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import React, { ReactElement } from "react";
import { useNavigate } from "react-router-dom";
import Cloud from "src/assets/images/cloud.png";
import Cloud2 from "src/assets/images/cloud2.png";
import { SelfCustodyIcon } from "src/components/icons/SelfCustodyIcon";
import { Button } from "src/components/ui/button";
import {
Carousel,
Expand Down Expand Up @@ -87,6 +88,14 @@ export function Intro() {
</div>
</div>
</CarouselItem>
<CarouselItem>
rolznz marked this conversation as resolved.
Show resolved Hide resolved
<Slide
api={api}
icon={SelfCustodyIcon}
title="Take Self-Custody in One Click"
description="Easily move your funds from Alby hosted balance to your own, self-custodial wallet."
/>
</CarouselItem>
<CarouselItem>
<Slide
api={api}
Expand Down Expand Up @@ -143,7 +152,17 @@ function Slide({

return (
<div className="flex flex-col justify-center items-center h-screen gap-8 p-5">
<Icon className="w-16 h-16 text-primary-background" />
{Icon === SelfCustodyIcon ? (
<Icon
width="211"
height="64"
viewBox="0 0 211 64"
rolznz marked this conversation as resolved.
Show resolved Hide resolved
strokeWidth="0"
className="text-primary-background"
/>
) : (
<Icon className="w-16 h-16 text-primary-background" />
)}
<div className="flex flex-col gap-4 text-center items-center max-w-lg">
<div className="text-3xl font-semibold text-primary-background">
{title}
Expand Down
49 changes: 7 additions & 42 deletions frontend/src/screens/channels/Channels.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ import { ChannelsCards } from "src/components/channels/ChannelsCards.tsx";
import { ChannelsTable } from "src/components/channels/ChannelsTable.tsx";
import EmptyState from "src/components/EmptyState.tsx";
import ExternalLink from "src/components/ExternalLink";
import { TransferFundsButton } from "src/components/TransferFundsButton";
import {
Alert,
AlertDescription,
Expand All @@ -39,7 +40,6 @@ import {
DropdownMenuSeparator,
DropdownMenuTrigger,
} from "src/components/ui/dropdown-menu.tsx";
import { LoadingButton } from "src/components/ui/loading-button.tsx";
import { CircleProgress } from "src/components/ui/progress.tsx";
import {
Tooltip,
Expand Down Expand Up @@ -73,8 +73,6 @@ export default function Channels() {
const [nodes, setNodes] = React.useState<Node[]>([]);

const { toast } = useToast();
const [drainingAlbySharedFunds, setDrainingAlbySharedFunds] =
React.useState(false);
const isDesktop = useIsDesktop();

const nodeHealth = channels ? getNodeHealth(channels) : 0;
Expand Down Expand Up @@ -252,51 +250,18 @@ export default function Channels() {
</CardHeader>
<CardContent className="flex-grow">
<div className="text-2xl font-bold">
{new Intl.NumberFormat().format(albyBalance?.sats)} sats
{new Intl.NumberFormat().format(albyBalance.sats)} sats
</div>
</CardContent>
<CardFooter className="flex justify-end space-x-1">
<LoadingButton
loading={drainingAlbySharedFunds}
onClick={async () => {
if (
!channels?.some(
(channel) =>
channel.remoteBalance / 1000 > albyBalance.sats
)
) {
toast({
title: "Please increase your receiving capacity first",
});
return;
}

setDrainingAlbySharedFunds(true);
try {
await request("/api/alby/drain", {
method: "POST",
headers: {
"Content-Type": "application/json",
},
});
await reloadAlbyBalance();
toast({
description:
"🎉 Funds from Alby shared wallet transferred to your Alby Hub!",
});
} catch (error) {
console.error(error);
toast({
variant: "destructive",
description: "Something went wrong: " + error,
});
}
setDrainingAlbySharedFunds(false);
}}
<TransferFundsButton
variant="outline"
channels={channels}
albyBalance={albyBalance}
reloadAlbyBalance={reloadAlbyBalance}
>
Transfer
</LoadingButton>
</TransferFundsButton>
</CardFooter>
</Card>
)}
Expand Down
45 changes: 45 additions & 0 deletions frontend/src/screens/wallet/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,28 +8,73 @@ import { Link } from "react-router-dom";
import AppHeader from "src/components/AppHeader";
import BreezRedeem from "src/components/BreezRedeem";
import ExternalLink from "src/components/ExternalLink";
import { SelfCustodyIcon } from "src/components/icons/SelfCustodyIcon";
import Loading from "src/components/Loading";
import TransactionsList from "src/components/TransactionsList";
import { TransferFundsButton } from "src/components/TransferFundsButton";
import {
Alert,
AlertDescription,
AlertTitle,
} from "src/components/ui/alert.tsx";
import { Button } from "src/components/ui/button";
import { ALBY_HIDE_HOSTED_BALANCE_BELOW as ALBY_HIDE_HOSTED_BALANCE_LIMIT } from "src/constants.ts";
import { useAlbyBalance } from "src/hooks/useAlbyBalance";
import { useBalances } from "src/hooks/useBalances";
import { useChannels } from "src/hooks/useChannels";
import { useInfo } from "src/hooks/useInfo";

function Wallet() {
const { data: info, hasChannelManagement } = useInfo();
const { data: balances } = useBalances();
const { data: channels } = useChannels();
const { data: albyBalance, mutate: reloadAlbyBalance } = useAlbyBalance();

if (!info || !balances) {
return <Loading />;
}

const showMigrateCard =
albyBalance && albyBalance.sats > ALBY_HIDE_HOSTED_BALANCE_LIMIT;

return (
<>
<AppHeader title="Wallet" description="" />
{showMigrateCard && (
<div className="flex flex-1 items-center justify-center rounded-lg border border-dashed shadow-sm p-8">
<div className="flex flex-col items-center gap-1 text-center max-w-md">
<SelfCustodyIcon
width="211"
height="64"
viewBox="0 0 211 64"
strokeWidth="0"
className="text-primary-background"
/>
<h3 className="mt-4 text-lg font-semibold">
Your funds ({new Intl.NumberFormat().format(albyBalance.sats)}{" "}
sats) are still hosted by Alby.
</h3>
<p className="text-sm text-muted-foreground mb-4">
{channels && channels.length > 0
? "Transfer funds from your Alby hosted balance to your self-custodial wallet."
: "Migrate funds from your Alby hosted balance to start using your self-custodial wallet."}
</p>
{channels && channels.length > 0 ? (
<TransferFundsButton
channels={channels}
Copy link
Contributor

Choose a reason for hiding this comment

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

you could have used the SWR functions within the component rather than passing these as args. But I think it's ok

albyBalance={albyBalance}
reloadAlbyBalance={reloadAlbyBalance}
>
Transfer Funds
</TransferFundsButton>
) : (
<Link to="/channels/first">
<Button className="mt-4">Migrate Funds</Button>
</Link>
)}
</div>
</div>
)}
{hasChannelManagement && !balances.lightning.totalSpendable && (
<Alert>
<AlertTriangle className="h-4 w-4" />
Expand Down
Loading