withdraw design

This commit is contained in:
Sebastian 2021-09-13 13:32:58 -03:00
parent 217f34397f
commit c24db59be8
No known key found for this signature in database
GPG Key ID: BE4FF68352439FC1
5 changed files with 937 additions and 34 deletions

View File

@ -0,0 +1,57 @@
/*
This file is part of GNU Taler
(C) 2019 Taler Systems SA
GNU Taler is free software; you can redistribute it and/or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation; either version 3, or (at your option) any later version.
GNU Taler is distributed in the hope that it will be useful, but WITHOUT ANY
WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE. See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License along with
GNU Taler; see the file COPYING. If not, see <http://www.gnu.org/licenses/>
*/
import { JSX } from "preact/jsx-runtime";
import { Outlined, StyledCheckboxLabel } from "./styled/index";
interface Props {
enabled: boolean;
onToggle: () => void;
label: string;
name: string;
}
const Tick = () => <svg
xmlns="http://www.w3.org/2000/svg"
viewBox="0 0 24 24"
aria-hidden="true"
focusable="false"
style={{ backgroundColor: 'green' }}
>
<path
fill="none"
stroke="white"
stroke-width="3"
d="M1.73 12.91l6.37 6.37L22.79 4.59"
/>
</svg>
export function CheckboxOutlined({ name, enabled, onToggle, label }: Props): JSX.Element {
return (
<Outlined>
<StyledCheckboxLabel onClick={onToggle}>
<span>
<input type="checkbox" name={name} checked={enabled} disabled={false} />
<div>
<Tick />
</div>
<label for={name}>{label}</label>
</span>
</StyledCheckboxLabel>
</Outlined>
);
}

View File

@ -0,0 +1,51 @@
import { Fragment, VNode } from "preact"
import { useState } from "preact/hooks"
import { JSXInternal } from "preact/src/jsx"
export function ExchangeXmlTos({ doc }: { doc: Document }) {
const termsNode = doc.querySelector('[ids=terms-of-service]')
if (!termsNode) {
return <div>not found</div>
}
return <Fragment>
{Array.from(termsNode.children).map(renderChild)}
</Fragment>
}
function renderChild(child: Element): VNode {
const children = Array.from(child.children)
switch (child.nodeName) {
case 'title': return <header>{child.textContent}</header>
case '#text': return <Fragment />
case 'paragraph': return <p>{child.textContent}</p>
case 'section': {
return <AnchorWithOpenState href={`#terms-${child.getAttribute('ids')}`}>
{children.map(renderChild)}
</AnchorWithOpenState>
}
case 'bullet_list': {
return <ul>{children.map(renderChild)}</ul>
}
case 'enumerated_list': {
return <ol>{children.map(renderChild)}</ol>
}
case 'list_item': {
return <li>{children.map(renderChild)}</li>
}
case 'block_quote': {
return <div>{children.map(renderChild)}</div>
}
default: return <div style={{ color: 'red', display: 'hidden' }}>unknown tag {child.nodeName} <a></a></div>
}
}
function AnchorWithOpenState(props: JSXInternal.HTMLAttributes<HTMLAnchorElement>) {
const [open, setOpen] = useState<boolean>(false)
function doClick(e: JSXInternal.TargetedMouseEvent<HTMLAnchorElement>) {
setOpen(!open);
e.stopPropagation();
e.preventDefault();
}
return <a data-open={JSON.stringify(open)} onClick={doClick} {...props} />
}

View File

@ -12,6 +12,10 @@ export const PaymentStatus = styled.div<{ color: string }>`
`
export const WalletAction = styled.section`
display: flex;
flex-direction: column;
justify-content: space-between;
align-items: center;
max-width: 50%;
margin: auto;
@ -20,6 +24,9 @@ export const WalletAction = styled.section`
& h1:first-child {
margin-top: 0;
}
section {
margin-bottom: 2em;
}
`
export const WalletActionOld = styled.section`
border: solid 5px black;
@ -211,6 +218,51 @@ export const Button = styled.button<{ upperCased?: boolean }>`
}
`;
export const Link = styled.a<{ upperCased?: boolean }>`
display: inline-block;
zoom: 1;
line-height: normal;
white-space: nowrap;
vertical-align: middle;
text-align: center;
cursor: pointer;
user-select: none;
box-sizing: border-box;
text-transform: ${({ upperCased }) => upperCased ? 'uppercase' : 'none'};
font-family: inherit;
font-size: 100%;
padding: 0.5em 1em;
background-color: transparent;
text-decoration: none;
:focus {
outline: 0;
}
&:disabled {
border: none;
background-image: none;
/* csslint ignore:start */
filter: alpha(opacity=40);
/* csslint ignore:end */
opacity: 0.4;
cursor: not-allowed;
box-shadow: none;
pointer-events: none;
}
:hover {
text-decoration: underline;
/* filter: alpha(opacity=90);
background-image: linear-gradient(
transparent,
rgba(0, 0, 0, 0.05) 40%,
rgba(0, 0, 0, 0.1)
); */
}
`;
export const FontIcon = styled.div`
font-family: monospace;
font-size: x-large;
@ -220,7 +272,7 @@ export const FontIcon = styled.div`
`
export const ButtonBox = styled(Button)`
padding: .5em;
width: 2em;
width: fit-content;
height: 2em;
& > ${FontIcon} {
@ -255,6 +307,9 @@ export const ButtonBoxPrimary = styled(ButtonBox)`
export const ButtonSuccess = styled(ButtonVariant)`
background-color: #388e3c;
`
export const LinkSuccess = styled(Link)`
color: #388e3c;
`
export const ButtonBoxSuccess = styled(ButtonBox)`
color: #388e3c;
border-color: #388e3c;
@ -504,3 +559,144 @@ export const NiceSelect = styled.div`
display: none;
}
`
export const Outlined = styled.div`
border: 2px solid #388e3c;
padding: 0.5em 1em;
width: fit-content;
border-radius: 2px;
color: #388e3c;
`
/* { width: "1.5em", height: "1.5em", verticalAlign: "middle" } */
export const CheckboxSuccess = styled.input`
vertical-align: center;
`
export const TermsSection = styled.a`
border: 1px solid black;
border-radius: 5px;
padding: 1em;
margin-top: 2px;
margin-bottom: 2px;
text-decoration: none;
color: inherit;
flex-direction: column;
display: flex;
&[data-open="true"] {
display: flex;
}
&[data-open="false"] > *:not(:first-child) {
display: none;
}
header {
display: flex;
flex-direction: row;
font-weight: bold;
justify-content: space-between;
height: auto;
}
&[data-open="true"] header:after {
content: '\\2227';
}
&[data-open="false"] header:after {
content: '\\2228';
}
`;
export const TermsOfService = styled.div`
display: flex;
flex-direction: column;
text-align: left;
& > header {
text-align: center;
font-size: 2em;
}
a {
border: 1px solid black;
border-radius: 5px;
padding: 1em;
margin-top: 2px;
margin-bottom: 2px;
text-decoration: none;
color: inherit;
flex-direction: column;
display: flex;
&[data-open="true"] {
display: flex;
}
&[data-open="false"] > *:not(:first-child) {
display: none;
}
header {
display: flex;
flex-direction: row;
font-weight: bold;
justify-content: space-between;
height: auto;
}
&[data-open="true"] header:after {
content: '\\2227';
}
&[data-open="false"] header:after {
content: '\\2228';
}
}
`
export const StyledCheckboxLabel = styled.div`
color: green;
text-transform: uppercase;
/* font-weight: bold; */
text-align: center;
span {
input {
display: none;
opacity: 0;
width: 1em;
height: 1em;
}
div {
display: inline-grid;
width: 1em;
height: 1em;
margin-right: 1em;
border-radius: 2px;
border: 2px solid currentColor;
svg {
transition: transform 0.1s ease-in 25ms;
transform: scale(0);
transform-origin: bottom left;
}
}
label {
padding: 0px;
font-size: small;
}
}
input:checked + div svg {
transform: scale(1);
}
input:disabled + div {
color: #959495;
};
input:disabled + div + label {
color: #959495;
};
input:focus + div + label {
box-shadow: 0 0 0 0.05em #fff, 0 0 0.15em 0.1em currentColor;
}
`

View File

@ -22,6 +22,7 @@
import { amountFractionalBase, Amounts } from '@gnu-taler/taler-util';
import { ExchangeRecord } from '@gnu-taler/taler-wallet-core';
import { ExchangeWithdrawDetails } from '@gnu-taler/taler-wallet-core/src/operations/withdraw';
import { getMaxListeners } from 'process';
import { createExample } from '../test-utils';
import { View as TestedComponent } from './Withdraw';
@ -33,20 +34,353 @@ export default {
},
};
export const WithdrawWithFee = createExample(TestedComponent, {
details: {
exchangeInfo: {
baseUrl: 'exchange.demo.taler.net'
} as ExchangeRecord,
withdrawFee: {
currency: 'USD',
fraction: amountFractionalBase*0.5,
value: 0
},
} as ExchangeWithdrawDetails,
amount: 'USD:2',
})
export const WithdrawWithoutFee = createExample(TestedComponent, {
const termsXml = `<?xml version="1.0" encoding="utf-8"?>
<document source="/home/grothoff/research/taler/exchange/contrib/tos/tos.rst">
<section ids="terms-of-service" names="terms\ of\ service">
<title>Terms Of Service</title>
<paragraph>Last Updated: 12.4.2019</paragraph>
<paragraph>Welcome! Taler Systems SA (we, our, or us) provides a payment service
through our Internet presence (collectively the Services). Before using our
Services, please read the Terms of Service (the Terms or the Agreement)
carefully.</paragraph>
<section ids="overview" names="overview">
<title>Overview</title>
<paragraph>This section provides a brief summary of the highlights of this
Agreement. Please note that when you accept this Agreement, you are accepting
all of the terms and conditions and not just this section. We and possibly
other third parties provide Internet services which interact with the Taler
Wallets self-hosted personal payment application. When using the Taler Wallet
to interact with our Services, you are agreeing to our Terms, so please read
carefully.</paragraph>
<section ids="highlights" names="highlights:">
<title>Highlights:</title>
<block_quote>
<bullet_list bullet="•">
<list_item>
<paragraph>You are responsible for keeping the data in your Taler Wallet at all times
under your control. Any losses arising from you not being in control of
your private information are your problem.</paragraph>
</list_item>
<list_item>
<paragraph>We will try to transfer funds we hold in escrow for our users to any legal
recipient to the best of our ability within the limitations of the law and
our implementation. However, the Services offered today are highly
experimental and the set of recipients of funds is severely restricted.</paragraph>
</list_item>
<list_item>
<paragraph>For our Services, we may charge transaction fees. The specific fee structure
is provided based on the Taler protocol and should be shown to you when you
withdraw electronic coins using a Taler Wallet. You agree and understand
that the Taler protocol allows for the fee structure to change.</paragraph>
</list_item>
<list_item>
<paragraph>You agree to not intentionally overwhelm our systems with requests and
follow responsible disclosure if you find security issues in our services.</paragraph>
</list_item>
<list_item>
<paragraph>We cannot be held accountable for our Services not being available due to
circumstances beyond our control. If we modify or terminate our services,
we will try to give you the opportunity to recover your funds. However,
given the experimental state of the Services today, this may not be
possible. You are strongly advised to limit your use of the Service
to small-scale experiments expecting total loss of all funds.</paragraph>
</list_item>
</bullet_list>
</block_quote>
<paragraph>These terms outline approved uses of our Services. The Services and these
Terms are still at an experimental stage. If you have any questions or
comments related to this Agreement, please send us a message to
<reference refuri="mailto:legal@taler-systems.com">legal@taler-systems.com</reference>. If you do not agree to this Agreement, you must not
use our Services.</paragraph>
</section>
</section>
<section ids="how-you-accept-this-policy" names="how\ you\ accept\ this\ policy">
<title>How you accept this policy</title>
<paragraph>By sending funds to us (to top-up your Taler Wallet), you acknowledge that you
have read, understood, and agreed to these Terms. We reserve the right to
change these Terms at any time. If you disagree with the change, we may in the
future offer you with an easy option to recover your unspent funds. However,
in the current experimental period you acknowledge that this feature is not
yet available, resulting in your funds being lost unless you accept the new
Terms. If you continue to use our Services other than to recover your unspent
funds, your continued use of our Services following any such change will
signify your acceptance to be bound by the then current Terms. Please check
the effective date above to determine if there have been any changes since you
have last reviewed these Terms.</paragraph>
</section>
<section ids="services" names="services">
<title>Services</title>
<paragraph>We will try to transfer funds that we hold in escrow for our users to any
legal recipient to the best of our ability and within the limitations of the
law and our implementation. However, the Services offered today are highly
experimental and the set of recipients of funds is severely restricted. The
Taler Wallet can be loaded by exchanging fiat currencies against electronic
coins. We are providing this exchange service. Once your Taler Wallet is
loaded with electronic coins they can be spent for purchases if the seller is
accepting Taler as a means of payment. We are not guaranteeing that any seller
is accepting Taler at all or a particular seller. The seller or recipient of
deposits of electronic coins must specify the target account, as per the
design of the Taler protocol. They are responsible for following the protocol
and specifying the correct bank account, and are solely liable for any losses
that may arise from specifying the wrong account. We will allow the government
to link wire transfers to the underlying contract hash. It is the
responsibility of recipients to preserve the full contracts and to pay
whatever taxes and charges may be applicable. Technical issues may lead to
situations where we are unable to make transfers at all or lead to incorrect
transfers that cannot be reversed. We will only refuse to execute transfers if
the transfers are prohibited by a competent legal authority and we are ordered
to do so.</paragraph>
</section>
<section ids="fees" names="fees">
<title>Fees</title>
<paragraph>You agree to pay the fees for exchanges and withdrawals completed via the
Taler Wallet (Fees) as defined by us, which we may change from time to
time. With the exception of wire transfer fees, Taler transaction fees are set
for any electronic coin at the time of withdrawal and fixed throughout the
validity period of the respective electronic coin. Your wallet should obtain
and display applicable fees when withdrawing funds. Fees for coins obtained as
change may differ from the fees applicable to the original coin. Wire transfer
fees that are independent from electronic coins may change annually. You
authorize us to charge or deduct applicable fees owed in connection with
deposits, exchanges and withdrawals following the rules of the Taler protocol.
We reserve the right to provide different types of rewards to users either in
the form of discount for our Services or in any other form at our discretion
and without prior notice to you.</paragraph>
</section>
<section ids="eligibility" names="eligibility">
<title>Eligibility</title>
<paragraph>To be eligible to use our Services, you must be able to form legally binding
contracts or have the permission of your legal guardian. By using our
Services, you represent and warrant that you meet all eligibility requirements
that we outline in these Terms.</paragraph>
</section>
<section ids="financial-self-responsibility" names="financial\ self-responsibility">
<title>Financial self-responsibility</title>
<paragraph>You will be responsible for maintaining the availability, integrity and
confidentiality of the data stored in your wallet. When you setup a Taler
Wallet, you are strongly advised to follow the precautionary measures offered
by the software to minimize the chances to losse access to or control over
your Wallet data. We will not be liable for any loss or damage arising from
your failure to comply with this paragraph.</paragraph>
</section>
<section ids="copyrights-and-trademarks" names="copyrights\ and\ trademarks">
<title>Copyrights and trademarks</title>
<paragraph>The Taler Wallet is released under the terms of the GNU General Public License
(GNU GPL). You have the right to access, use, and share the Taler Wallet, in
modified or unmodified form. However, the GPL is a strong copyleft license,
which means that any derivative works must be distributed under the same
license terms as the original software. If you have any questions, you should
review the GNU GPLs full terms and conditions at
<reference refuri="https://www.gnu.org/licenses/gpl-3.0.en.html">https://www.gnu.org/licenses/gpl-3.0.en.html</reference>. “Taler” itself is a trademark
of Taler Systems SA. You are welcome to use the name in relation to processing
payments using the Taler protocol, assuming your use is compatible with an
official release from the GNU Project that is not older than two years.</paragraph>
</section>
<section ids="your-use-of-our-services" names="your\ use\ of\ our\ services">
<title>Your use of our services</title>
<paragraph>When using our Services, you agree to not take any action that intentionally
imposes an unreasonable load on our infrastructure. If you find security
problems in our Services, you agree to first report them to
<reference refuri="mailto:security@taler-systems.com">security@taler-systems.com</reference> and grant us the right to publish your report. We
warrant that we will ourselves publicly disclose any issues reported within 3
months, and that we will not prosecute anyone reporting security issues if
they did not exploit the issue beyond a proof-of-concept, and followed the
above responsible disclosure practice.</paragraph>
</section>
<section ids="limitation-of-liability-disclaimer-of-warranties" names="limitation\ of\ liability\ &amp;\ disclaimer\ of\ warranties">
<title>Limitation of liability &amp; disclaimer of warranties</title>
<paragraph>You understand and agree that we have no control over, and no duty to take any
action regarding: Failures, disruptions, errors, or delays in processing that
you may experience while using our Services; The risk of failure of hardware,
software, and Internet connections; The risk of malicious software being
introduced or found in the software underlying the Taler Wallet; The risk that
third parties may obtain unauthorized access to information stored within your
Taler Wallet, including, but not limited to your Taler Wallet coins or backup
encryption keys. You release us from all liability related to any losses,
damages, or claims arising from:</paragraph>
<enumerated_list enumtype="loweralpha" prefix="(" suffix=")">
<list_item>
<paragraph>user error such as forgotten passwords, incorrectly constructed
transactions;</paragraph>
</list_item>
<list_item>
<paragraph>server failure or data loss;</paragraph>
</list_item>
<list_item>
<paragraph>unauthorized access to the Taler Wallet application;</paragraph>
</list_item>
<list_item>
<paragraph>bugs or other errors in the Taler Wallet software; and</paragraph>
</list_item>
<list_item>
<paragraph>any unauthorized third party activities, including, but not limited to,
the use of viruses, phishing, brute forcing, or other means of attack
against the Taler Wallet. We make no representations concerning any
Third Party Content contained in or accessed through our Services.</paragraph>
</list_item>
</enumerated_list>
<paragraph>Any other terms, conditions, warranties, or representations associated with
such content, are solely between you and such organizations and/or
individuals.</paragraph>
</section>
<section ids="limitation-of-liability" names="limitation\ of\ liability">
<title>Limitation of liability</title>
<paragraph>To the fullest extent permitted by applicable law, in no event will we or any
of our officers, directors, representatives, agents, servants, counsel,
employees, consultants, lawyers, and other personnel authorized to act,
acting, or purporting to act on our behalf (collectively the Taler Parties)
be liable to you under contract, tort, strict liability, negligence, or any
other legal or equitable theory, for:</paragraph>
<enumerated_list enumtype="loweralpha" prefix="(" suffix=")">
<list_item>
<paragraph>any lost profits, data loss, cost of procurement of substitute goods or
* TLSv1.3 (IN), TLS Unknown, Unknown (23):
services, or direct, indirect, incidental, special, punitive, compensatory,
or consequential damages of any kind whatsoever resulting from:</paragraph>
</list_item>
</enumerated_list>
<block_quote>
<enumerated_list enumtype="lowerroman" prefix="(" suffix=")">
<list_item>
<paragraph>your use of, or conduct in connection with, our services;</paragraph>
</list_item>
<list_item>
<paragraph>any unauthorized use of your wallet and/or private key due to your
failure to maintain the confidentiality of your wallet;</paragraph>
</list_item>
<list_item>
<paragraph>any interruption or cessation of transmission to or from the services; or</paragraph>
</list_item>
<list_item>
<paragraph>any bugs, viruses, trojan horses, or the like that are found in the Taler
Wallet software or that may be transmitted to or through our services by
any third party (regardless of the source of origination), or</paragraph>
</list_item>
</enumerated_list>
</block_quote>
<enumerated_list enumtype="loweralpha" prefix="(" start="2" suffix=")">
<list_item>
<paragraph>any direct damages.</paragraph>
</list_item>
</enumerated_list>
<paragraph>These limitations apply regardless of legal theory, whether based on tort,
strict liability, breach of contract, breach of warranty, or any other legal
theory, and whether or not we were advised of the possibility of such
damages. Some jurisdictions do not allow the exclusion or limitation of
liability for consequential or incidental damages, so the above limitation may
not apply to you.</paragraph>
</section>
<section ids="warranty-disclaimer" names="warranty\ disclaimer">
<title>Warranty disclaimer</title>
<paragraph>Our services are provided as is and without warranty of any kind. To the
maximum extent permitted by law, we disclaim all representations and
warranties, express or implied, relating to the services and underlying
software or any content on the services, whether provided or owned by us or by
any third party, including without limitation, warranties of merchantability,
fitness for a particular purpose, title, non-infringement, freedom from
computer virus, and any implied warranties arising from course of dealing,
course of performance, or usage in trade, all of which are expressly
disclaimed. In addition, we do not represent or warrant that the content
accessible via the services is accurate, complete, available, current, free of
viruses or other harmful components, or that the results of using the services
will meet your requirements. Some states do not allow the disclaimer of
implied warranties, so the foregoing disclaimers may not apply to you. This
paragraph gives you specific legal rights and you may also have other legal
rights that vary from state to state.</paragraph>
</section>
<section ids="indemnity" names="indemnity">
<title>Indemnity</title>
<paragraph>To the extent permitted by applicable law, you agree to defend, indemnify, and
hold harmless the Taler Parties from and against any and all claims, damages,
obligations, losses, liabilities, costs or debt, and expenses (including, but
not limited to, attorneys fees) arising from: (a) your use of and access to
the Services; (b) any feedback or submissions you provide to us concerning the
Taler Wallet; (c) your violation of any term of this Agreement; or (d) your
violation of any law, rule, or regulation, or the rights of any third party.</paragraph>
</section>
<section ids="time-limitation-on-claims" names="time\ limitation\ on\ claims">
<title>Time limitation on claims</title>
<paragraph>You agree that any claim you may have arising out of or related to your
relationship with us must be filed within one year after such claim arises,
otherwise, your claim in permanently barred.</paragraph>
</section>
<section ids="governing-law" names="governing\ law">
<title>Governing law</title>
<paragraph>No matter where youre located, the laws of Switzerland will govern these
Terms. If any provisions of these Terms are inconsistent with any applicable
law, those provisions will be superseded or modified only to the extent such
provisions are inconsistent. The parties agree to submit to the ordinary
courts in Zurich, Switzerland for exclusive jurisdiction of any dispute
arising out of or related to your use of the Services or your breach of these
Terms.</paragraph>
</section>
<section ids="termination" names="termination">
<title>Termination</title>
<paragraph>In the event of termination concerning your use of our Services, your
obligations under this Agreement will still continue.</paragraph>
</section>
<section ids="discontinuance-of-services" names="discontinuance\ of\ services">
<title>Discontinuance of services</title>
<paragraph>We may, in our sole discretion and without cost to you, with or without prior
notice, and at any time, modify or discontinue, temporarily or permanently,
any portion of our Services. We will use the Taler protocols provisions to
notify Wallets if our Services are to be discontinued. It is your
responsibility to ensure that the Taler Wallet is online at least once every
three months to observe these notifications. We shall not be held responsible
or liable for any loss of funds in the event that we discontinue or depreciate
the Services and your Taler Wallet fails to transfer out the coins within a
three months notification period.</paragraph>
</section>
<section ids="no-waiver" names="no\ waiver">
<title>No waiver</title>
<paragraph>Our failure to exercise or delay in exercising any right, power, or privilege
under this Agreement shall not operate as a waiver; nor shall any single or
partial exercise of any right, power, or privilege preclude any other or
further exercise thereof.</paragraph>
</section>
<section ids="severability" names="severability">
<title>Severability</title>
<paragraph>If it turns out that any part of this Agreement is invalid, void, or for any
reason unenforceable, that term will be deemed severable and limited or
eliminated to the minimum extent necessary.</paragraph>
</section>
<section ids="force-majeure" names="force\ majeure">
<title>Force majeure</title>
<paragraph>We shall not be held liable for any delays, failure in performance, or
interruptions of service which result directly or indirectly from any cause or
condition beyond our reasonable control, including but not limited to: any
delay or failure due to any act of God, act of civil or military authorities,
act of terrorism, civil disturbance, war, strike or other labor dispute, fire,
interruption in telecommunications or Internet services or network provider
services, failure of equipment and/or software, other catastrophe, or any
other occurrence which is beyond our reasonable control and shall not affect
the validity and enforceability of any remaining provisions.</paragraph>
</section>
<section ids="assignment" names="assignment">
<title>Assignment</title>
<paragraph>You agree that we may assign any of our rights and/or transfer, sub-contract,
or delegate any of our obligations under these Terms.</paragraph>
</section>
<section ids="entire-agreement" names="entire\ agreement">
<title>Entire agreement</title>
<paragraph>This Agreement sets forth the entire understanding and agreement as to the
subject matter hereof and supersedes any and all prior discussions,
agreements, and understandings of any kind (including, without limitation, any
prior versions of this Agreement) and every nature between us. Except as
provided for above, any modification to this Agreement must be in writing and
must be signed by both parties.</paragraph>
</section>
<section ids="questions-or-comments" names="questions\ or\ comments">
<title>Questions or comments</title>
<paragraph>We welcome comments, questions, concerns, or suggestions. Please send us a
message on our contact page at <reference refuri="mailto:legal@taler-systems.com">legal@taler-systems.com</reference>.</paragraph>
</section>
</section>
</document>
`;
export const WithdrawNewTermsXML = createExample(TestedComponent, {
details: {
exchangeInfo: {
baseUrl: 'exchange.demo.taler.net'
@ -58,4 +392,155 @@ export const WithdrawWithoutFee = createExample(TestedComponent, {
},
} as ExchangeWithdrawDetails,
amount: 'USD:2',
terms: {
value : {
type: 'xml',
document: new DOMParser().parseFromString(termsXml, "text/xml"),
},
status: 'new'
},
})
export const WithdrawNewTermsReviewingXML = createExample(TestedComponent, {
details: {
exchangeInfo: {
baseUrl: 'exchange.demo.taler.net'
} as ExchangeRecord,
withdrawFee: {
currency: 'USD',
fraction: 0,
value: 0
},
} as ExchangeWithdrawDetails,
amount: 'USD:2',
terms: {
value : {
type: 'xml',
document: new DOMParser().parseFromString(termsXml, "text/xml"),
},
status: 'new'
},
reviewing: true
})
export const WithdrawNewTermsAcceptedXML = createExample(TestedComponent, {
details: {
exchangeInfo: {
baseUrl: 'exchange.demo.taler.net'
} as ExchangeRecord,
withdrawFee: {
currency: 'USD',
fraction: 0,
value: 0
},
} as ExchangeWithdrawDetails,
amount: 'USD:2',
terms: {
value : {
type: 'xml',
document: new DOMParser().parseFromString(termsXml, "text/xml"),
},
status: 'new'
},
accepted: true
})
export const WithdrawNewTermsShowAfterAcceptedXML = createExample(TestedComponent, {
details: {
exchangeInfo: {
baseUrl: 'exchange.demo.taler.net'
} as ExchangeRecord,
withdrawFee: {
currency: 'USD',
fraction: 0,
value: 0
},
} as ExchangeWithdrawDetails,
amount: 'USD:2',
terms: {
value : {
type: 'xml',
document: new DOMParser().parseFromString(termsXml, "text/xml"),
},
status: 'new'
},
accepted: true,
reviewing: true,
})
export const WithdrawChangedTermsXML = createExample(TestedComponent, {
details: {
exchangeInfo: {
baseUrl: 'exchange.demo.taler.net'
} as ExchangeRecord,
withdrawFee: {
currency: 'USD',
fraction: 0,
value: 0
},
} as ExchangeWithdrawDetails,
amount: 'USD:2',
terms: {
value : {
type: 'xml',
document: new DOMParser().parseFromString(termsXml, "text/xml"),
},
status: 'changed'
},
})
export const WithdrawNotFoundTermsXML = createExample(TestedComponent, {
details: {
exchangeInfo: {
baseUrl: 'exchange.demo.taler.net'
} as ExchangeRecord,
withdrawFee: {
currency: 'USD',
fraction: 0,
value: 0
},
} as ExchangeWithdrawDetails,
amount: 'USD:2',
terms: {
status: 'notfound'
},
})
export const WithdrawAcceptedTermsXML = createExample(TestedComponent, {
details: {
exchangeInfo: {
baseUrl: 'exchange.demo.taler.net'
} as ExchangeRecord,
withdrawFee: {
currency: 'USD',
fraction: amountFractionalBase * 0.5,
value: 0
},
} as ExchangeWithdrawDetails,
amount: 'USD:2',
terms: {
status: 'accepted'
},
})
export const WithdrawAcceptedTermsWithoutFee = createExample(TestedComponent, {
details: {
exchangeInfo: {
baseUrl: 'exchange.demo.taler.net'
} as ExchangeRecord,
withdrawFee: {
currency: 'USD',
fraction: 0,
value: 0
},
} as ExchangeWithdrawDetails,
amount: 'USD:2',
terms: {
value : {
type: 'xml',
document: new DOMParser().parseFromString(termsXml, "text/xml"),
},
status: 'accepted',
}
})

View File

@ -24,10 +24,11 @@
import { AmountLike, Amounts, i18n, WithdrawUriInfoResponse } from '@gnu-taler/taler-util';
import { ExchangeWithdrawDetails } from '@gnu-taler/taler-wallet-core/src/operations/withdraw';
import { useEffect, useState } from "preact/hooks";
import { JSX } from "preact/jsx-runtime";
import { CheckboxOutlined } from '../components/CheckboxOutlined';
import { ExchangeXmlTos } from '../components/ExchangeToS';
import { LogoHeader } from '../components/LogoHeader';
import { Part } from '../components/Part';
import { ButtonSuccess, WalletAction } from '../components/styled';
import { ButtonDestructive, ButtonSuccess, ButtonWarning, LinkSuccess, TermsOfService, WalletAction } from '../components/styled';
import {
acceptWithdrawal, getExchangeWithdrawalInfo, getWithdrawalDetailsForUri, onUpdateNotification
} from "../wxApi";
@ -40,19 +41,42 @@ interface Props {
export interface ViewProps {
details: ExchangeWithdrawDetails;
amount: string;
accept: () => Promise<void>;
setCancelled: (b: boolean) => void;
setSelecting: (b: boolean) => void;
onWithdraw: () => Promise<void>;
// setCancelled: (b: boolean) => void;
// setSelecting: (b: boolean) => void;
onReview: (b: boolean) => void;
onAccept: (b: boolean) => void;
reviewing: boolean;
accepted: boolean;
terms: {
value?: TermsDocument;
status: TermsStatus;
}
};
type TermsStatus = 'new' | 'accepted' | 'changed' | 'notfound';
type TermsDocument = TermsDocumentXml | TermsDocumentHtml;
interface TermsDocumentXml {
type: 'xml',
document: Document,
}
interface TermsDocumentHtml {
type: 'html',
href: string,
}
function amountToString(text: AmountLike) {
const aj = Amounts.jsonifyAmount(text)
const amount = Amounts.stringifyValue(aj)
return `${amount} ${aj.currency}`
}
export function View({ details, amount, accept, setCancelled, setSelecting }: ViewProps) {
export function View({ details, amount, onWithdraw, terms, reviewing, onReview, onAccept, accepted }: ViewProps) {
const needsReview = terms.status === 'changed' || terms.status === 'new'
return (
<WalletAction style={{ textAlign: 'center' }}>
@ -70,17 +94,101 @@ export function View({ details, amount, accept, setCancelled, setSelecting }: Vi
<Part title="Exchange" text={details.exchangeInfo.baseUrl} kind='neutral' big />
</div>
</section>
<section>
<div>
<ButtonSuccess
{!reviewing &&
<section>
<LinkSuccess
upperCased
disabled={!details.exchangeInfo.baseUrl}
onClick={accept}
>
{i18n.str`Accept fees and withdraw`}
</ButtonSuccess>
</div>
{i18n.str`Edit exchange`}
</LinkSuccess>
</section>
}
{!reviewing && accepted &&
<section>
<LinkSuccess
upperCased
onClick={() => onReview(true)}
>
{i18n.str`Show terms of service`}
</LinkSuccess>
</section>
}
{reviewing &&
<section>
<TermsOfService>
{terms.status !== 'accepted' && terms.value && terms.value.type === 'xml' && <ExchangeXmlTos doc={terms.value.document} />}
</TermsOfService>
</section>}
{reviewing && accepted &&
<section>
<LinkSuccess
upperCased
onClick={() => onReview(false)}
>
{i18n.str`Hide terms of service`}
</LinkSuccess>
</section>
}
{(reviewing || accepted) &&
<section>
<div>
<CheckboxOutlined
name="terms"
enabled={accepted}
label={i18n.str`I accept the exchange terms of service`}
onToggle={() => {
onAccept(!accepted)
onReview(false)
}}
/>
</div>
</section>
}
<section>
{terms.status === 'new' && !accepted &&
<div>
<ButtonSuccess
upperCased
disabled={!details.exchangeInfo.baseUrl}
onClick={() => onReview(true)}
>
{i18n.str`Review exchange terms of service`}
</ButtonSuccess>
</div>
}
{terms.status === 'changed' && !accepted &&
<div>
<ButtonWarning
upperCased
disabled={!details.exchangeInfo.baseUrl}
onClick={() => onReview(true)}
>
{i18n.str`Review new version of terms of service`}
</ButtonWarning>
</div>
}
{(terms.status === 'accepted' || (needsReview && accepted)) &&
<div>
<ButtonSuccess
upperCased
disabled={!details.exchangeInfo.baseUrl}
onClick={onWithdraw}
>
{i18n.str`Confirm withdrawal`}
</ButtonSuccess>
</div>
}
{terms.status === 'notfound' &&
<div>
<ButtonDestructive
upperCased
disabled={true}
>
{i18n.str`Exchange doesn't have terms of service`}
</ButtonDestructive>
</div>
}
</section>
</WalletAction>
)
@ -93,6 +201,8 @@ export function WithdrawPage({ talerWithdrawUri, ...rest }: Props): JSX.Element
const [selecting, setSelecting] = useState(false);
const [error, setError] = useState<boolean>(false);
const [updateCounter, setUpdateCounter] = useState(1);
const [reviewing, setReviewing] = useState<boolean>(false)
const [accepted, setAccepted] = useState<boolean>(false)
useEffect(() => {
return onUpdateNotification(() => {
@ -132,7 +242,7 @@ export function WithdrawPage({ talerWithdrawUri, ...rest }: Props): JSX.Element
return <span><i18n.Translate>missing withdraw uri</i18n.Translate></span>;
}
const accept = async (): Promise<void> => {
const onWithdraw = async (): Promise<void> => {
if (!details) {
throw Error("can't accept, no exchange selected");
}
@ -157,9 +267,13 @@ export function WithdrawPage({ talerWithdrawUri, ...rest }: Props): JSX.Element
return <span><i18n.Translate>Getting withdrawal details.</i18n.Translate></span>;
}
return <View accept={accept}
setCancelled={setCancelled} setSelecting={setSelecting}
return <View onWithdraw={onWithdraw}
// setCancelled={setCancelled} setSelecting={setSelecting}
details={details} amount={uriInfo.amount}
terms={{} as any}
accepted={accepted} onAccept={setAccepted}
reviewing={reviewing} onReview={setReviewing}
// terms={[]}
/>
}