refine tipping

This commit is contained in:
keyan 2021-09-12 11:55:38 -05:00
parent 650ad03de5
commit e4c1c2f1e1
15 changed files with 116 additions and 24 deletions

View File

@ -209,7 +209,7 @@ export default {
return await updateItem(parent, { id, data: { text } }, { me, models }) return await updateItem(parent, { id, data: { text } }, { me, models })
}, },
act: async (parent, { id, act, sats }, { me, models }) => { act: async (parent, { id, act, sats, tipDefault }, { me, models }) => {
// need to make sure we are logged in // need to make sure we are logged in
if (!me) { if (!me) {
throw new AuthenticationError('you must be logged in') throw new AuthenticationError('you must be logged in')
@ -228,6 +228,10 @@ export default {
if (item) { if (item) {
throw new UserInputError('cannot tip your self') throw new UserInputError('cannot tip your self')
} }
// if tipDefault, set on user
if (tipDefault) {
await models.user.update({ where: { id: me.id }, data: { tipDefault: sats } })
}
} }
await serialize(models, models.$queryRaw`SELECT item_act(${Number(id)}, ${me.id}, ${act}, ${Number(sats)})`) await serialize(models, models.$queryRaw`SELECT item_act(${Number(id)}, ${me.id}, ${act}, ${Number(sats)})`)

View File

@ -27,7 +27,7 @@ export default gql`
updateDiscussion(id: ID!, title: String!, text: String): Item! updateDiscussion(id: ID!, title: String!, text: String): Item!
createComment(text: String!, parentId: ID!): Item! createComment(text: String!, parentId: ID!): Item!
updateComment(id: ID!, text: String!): Item! updateComment(id: ID!, text: String!): Item!
act(id: ID!, act: ItemAct!, sats: Int): ItemActResult! act(id: ID!, act: ItemAct!, sats: Int, tipDefault: Boolean): ItemActResult!
} }
type Items { type Items {

View File

@ -21,6 +21,7 @@ export default gql`
freePosts: Int! freePosts: Int!
freeComments: Int! freeComments: Int!
hasNewNotes: Boolean! hasNewNotes: Boolean!
tipDefault: Int!
sats: Int! sats: Int!
msats: Int! msats: Int!
} }

View File

@ -1,12 +1,15 @@
import { useFormikContext } from 'formik' import { useFormikContext } from 'formik'
import { OverlayTrigger, Tooltip } from 'react-bootstrap' import { OverlayTrigger, Tooltip } from 'react-bootstrap'
export default function ActionTooltip ({ children, notForm, overlayText }) { export default function ActionTooltip ({ children, notForm, disable, overlayText }) {
// if we're in a form, we want to hide tooltip on submit // if we're in a form, we want to hide tooltip on submit
let formik let formik
if (!notForm) { if (!notForm) {
formik = useFormikContext() formik = useFormikContext()
} }
if (disable) {
return children
}
return ( return (
<OverlayTrigger <OverlayTrigger
placement='bottom' placement='bottom'

View File

@ -15,12 +15,12 @@ export const AdvPostInitial = {
export default function AdvPostForm () { export default function AdvPostForm () {
return ( return (
<AccordianItem <AccordianItem
header={<div className='font-weight-bold'>advanced</div>} header={<div style={{ fontWeight: 'bold', fontSize: '92%' }}>options</div>}
body={ body={
<Input <Input
label='boost' label='boost'
name='boost' name='boost'
hint={<span className='text-muted'>boost ranks posts higher temporarily depending on the amount</span>} hint={<span className='text-muted'>boost ranks posts higher temporarily based on the amount</span>}
append={<InputGroup.Text className='text-monospace'>sats</InputGroup.Text>} append={<InputGroup.Text className='text-monospace'>sats</InputGroup.Text>}
/> />
} }

View File

@ -82,11 +82,9 @@ export function DiscussionForm ({ item, editThreshold }) {
: null} : null}
/> />
{!item && <AdvPostForm />} {!item && <AdvPostForm />}
<div className='d-flex'>
<ActionTooltip> <ActionTooltip>
<SubmitButton variant='secondary' className='mt-2 ml-auto'>{item ? 'save' : 'post'}</SubmitButton> <SubmitButton variant='secondary' className='mt-3'>{item ? 'save' : 'post'}</SubmitButton>
</ActionTooltip> </ActionTooltip>
</div>
</Form> </Form>
) )
} }

View File

@ -175,6 +175,36 @@ export function Input ({ label, groupClassName, ...props }) {
) )
} }
export function Checkbox ({ children, label, extra, handleChange, ...props }) {
// React treats radios and checkbox inputs differently other input types, select, and textarea.
// Formik does this too! When you specify `type` to useField(), it will
// return the correct bag of props for you
const [field, { value }] = useField({ ...props, type: 'checkbox' })
return (
<div className={value ? styles.checkboxChecked : styles.checkboxUnchecked}>
<BootstrapForm.Check
custom
id={props.id || props.name}
>
<BootstrapForm.Check.Input
{...field} {...props} type='checkbox' onChange={(e) => {
field.onChange(e)
handleChange && handleChange(e.target.checked)
}}
/>
<BootstrapForm.Check.Label className='d-flex'>
<div className='flex-grow-1'>{label}</div>
{extra &&
<div className={styles.checkboxExtra}>
{extra}
</div>}
</BootstrapForm.Check.Label>
</BootstrapForm.Check>
{children}
</div>
)
}
export function Form ({ export function Form ({
initial, schema, onSubmit, children, initialError, validateImmediately, ...props initial, schema, onSubmit, children, initialError, validateImmediately, ...props
}) { }) {

View File

@ -1,7 +1,8 @@
import { InputGroup, Modal } from 'react-bootstrap' import { InputGroup, Modal } from 'react-bootstrap'
import React, { useState, useCallback, useContext, useRef, useEffect } from 'react' import React, { useState, useCallback, useContext, useRef, useEffect } from 'react'
import * as Yup from 'yup' import * as Yup from 'yup'
import { Form, Input, SubmitButton } from './form' import { Checkbox, Form, Input, SubmitButton } from './form'
import { useMe } from './me'
export const ItemActContext = React.createContext({ export const ItemActContext = React.createContext({
item: null, item: null,
@ -36,6 +37,7 @@ export const ActSchema = Yup.object({
export function ItemActModal () { export function ItemActModal () {
const { item, setItem } = useItemAct() const { item, setItem } = useItemAct()
const inputRef = useRef(null) const inputRef = useRef(null)
const me = useMe()
useEffect(() => { useEffect(() => {
inputRef.current?.focus() inputRef.current?.focus()
@ -51,11 +53,19 @@ export function ItemActModal () {
<Modal.Body> <Modal.Body>
<Form <Form
initial={{ initial={{
amount: 21 amount: me?.tipDefault || 21,
default: false
}} }}
schema={ActSchema} schema={ActSchema}
onSubmit={async ({ amount, submit }) => { onSubmit={async ({ amount, tipDefault, submit }) => {
await item.act({ variables: { id: item.itemId, act: submit, sats: Number(amount) } }) await item.act({
variables: {
id: item.itemId,
act: submit,
sats: Number(amount),
tipDefault
}
})
await item.strike() await item.strike()
setItem(null) setItem(null)
}} }}
@ -68,6 +78,12 @@ export function ItemActModal () {
autoFocus autoFocus
append={<InputGroup.Text className='text-monospace'>sats</InputGroup.Text>} append={<InputGroup.Text className='text-monospace'>sats</InputGroup.Text>}
/> />
<Checkbox
label='set as default'
name='tipDefault'
required
autoFocus
/>
<div className='d-flex'> <div className='d-flex'>
<SubmitButton variant='success' className='ml-auto mt-1 px-4' value='TIP'>tip</SubmitButton> <SubmitButton variant='success' className='ml-auto mt-1 px-4' value='TIP'>tip</SubmitButton>
</div> </div>

View File

@ -109,11 +109,9 @@ export function LinkForm ({ item, editThreshold }) {
/> />
{!item && <AdvPostForm />} {!item && <AdvPostForm />}
<div className='d-flex'>
<ActionTooltip> <ActionTooltip>
<SubmitButton variant='secondary' className='mt-2 ml-auto'>{item ? 'save' : 'post'}</SubmitButton> <SubmitButton variant='secondary' className='mt-3'>{item ? 'save' : 'post'}</SubmitButton>
</ActionTooltip> </ActionTooltip>
</div>
</Form> </Form>
) )
} }

View File

@ -16,6 +16,7 @@ export function MeProvider ({ children }) {
freePosts freePosts
freeComments freeComments
hasNewNotes hasNewNotes
tipDefault
} }
}` }`
const { data } = useQuery(query, { pollInterval: 1000 }) const { data } = useQuery(query, { pollInterval: 1000 })

View File

@ -7,15 +7,17 @@ import { useFundError } from './fund-error'
import ActionTooltip from './action-tooltip' import ActionTooltip from './action-tooltip'
import { useItemAct } from './item-act' import { useItemAct } from './item-act'
import Window from '../svgs/window-2-fill.svg' import Window from '../svgs/window-2-fill.svg'
import { useMe } from './me'
export default function UpVote ({ item, className }) { export default function UpVote ({ item, className }) {
const [session] = useSession() const [session] = useSession()
const { setError } = useFundError() const { setError } = useFundError()
const { setItem } = useItemAct() const { setItem } = useItemAct()
const me = useMe()
const [act] = useMutation( const [act] = useMutation(
gql` gql`
mutation act($id: ID!, $act: ItemAct! $sats: Int!) { mutation act($id: ID!, $act: ItemAct! $sats: Int!, $tipDefault: Boolean) {
act(id: $id, act: $act, sats: $sats) { act(id: $id, act: $act, sats: $sats, tipDefault: $tipDefault) {
act, act,
sats sats
} }
@ -68,29 +70,58 @@ export default function UpVote ({ item, className }) {
} }
) )
const overlayText = () => {
if (item?.meVote) {
if (me?.tipDefault) {
return `tip ${me.tipDefault}`
}
return <Window style={{ fill: '#fff' }} width={18} height={18} />
}
return '1 sat'
}
const noSelfTips = item?.meVote && item?.user?.id === me?.id
return ( return (
<LightningConsumer> <LightningConsumer>
{({ strike }) => {({ strike }) =>
<ActionTooltip notForm overlayText={item?.meVote ? <Window style={{ fill: '#fff' }} /> : '1 sat'}> <ActionTooltip notForm disable={noSelfTips} overlayText={overlayText()}>
<UpArrow <UpArrow
width={24} width={24}
height={24} height={24}
className={ className={
`${styles.upvote} `${styles.upvote}
${className || ''} ${className || ''}
${noSelfTips ? styles.noSelfTips : ''}
${item?.meVote ? styles.voted : ''}` ${item?.meVote ? styles.voted : ''}`
} }
onClick={ onClick={
session session
? async (e) => { ? async (e) => {
e.stopPropagation() e.stopPropagation()
if (!item) return
// we can't tip ourselves
if (noSelfTips) {
return
}
if (item?.meVote) { if (item?.meVote) {
if (me?.tipDefault) {
try {
await act({ variables: { id: item.id, act: 'TIP', sats: me.tipDefault } })
strike()
} catch (e) {
console.log(e)
}
return
}
setItem({ itemId: item.id, act, strike }) setItem({ itemId: item.id, act, strike })
return return
} }
strike() strike()
if (!item) return
try { try {
await act({ variables: { id: item.id, act: 'VOTE', sats: 1 } }) await act({ variables: { id: item.id, act: 'VOTE', sats: 1 } })

View File

@ -8,6 +8,10 @@
cursor: pointer; cursor: pointer;
} }
.noSelfTips:hover {
cursor: default !important;
}
.upvote.voted { .upvote.voted {
fill: #F6911D; fill: #F6911D;
filter: drop-shadow(0 0 7px #F6911D); filter: drop-shadow(0 0 7px #F6911D);

View File

@ -0,0 +1,5 @@
-- AlterTable
ALTER TABLE "ItemAct" ALTER COLUMN "sats" DROP DEFAULT;
-- AlterTable
ALTER TABLE "users" ADD COLUMN "tipDefault" INTEGER NOT NULL DEFAULT 0;

View File

@ -28,6 +28,7 @@ model User {
freeComments Int @default(5) freeComments Int @default(5)
freePosts Int @default(2) freePosts Int @default(2)
checkedNotesAt DateTime? checkedNotesAt DateTime?
tipDefault Int @default(0)
pubkey String? @unique pubkey String? @unique
@@map(name: "users") @@map(name: "users")