\n >\n {formatCurrency(Number(data?.gastosExterior || 0))}\n {/* {disabled && (\n }\n onClick={() => handleEditValue(\"gastosExterior\")}\n />\n )} */}\n
\n );\n },\n },\n {\n id: \"multa\",\n label: (onClick: any) => (\n ();\n const [editForm] = Form.useForm();\n const handleEditValue = (id: string) => {\n const value = data?.[id as keyof (IDarf | IDarfBolsa)];\n setEdit({ id, isEditting: true, value });\n editForm.setFieldsValue({ [id]: value });\n };\n\n const handleEditValueCancel = () => {\n setEdit(undefined);\n editForm.resetFields();\n };\n const handleEditValueConfirm = () => {\n if (edit && edit.value >= 0 && edit.value < minDarfPrice) {\n if (id !== \"\") {\n api\n .put(\"dividendos/rendimentos\", {\n _id: id,\n year,\n month: month + 1,\n [edit.id]: edit.value,\n })\n .then(() =>\n (setDataCripto ?? setDataBolsa)?.((data: any) => ({\n ...data,\n [edit.id]: edit.value,\n }))\n )\n .catch(() =>\n message.error(\n \"Algo inesperado aconteceu! Tente novamente mais tarde.\"\n )\n )\n .finally(() => handleEditValueCancel());\n } else {\n api\n .post(\"dividendos/rendimentos\", {\n monthStock,\n year,\n month: month + 1,\n juros: data.juros,\n multa: data.multa,\n userCode: user.user.cpf,\n total: data.impostoTotal,\n impostoDevido: data.impostoDevido,\n totalRendimentos: data.totalRendimentos,\n baseTributaria: data.baseTributaria,\n aliquotaDevida: data.aliquotaDevida,\n totalDependentes: data.totalDependentes,\n totalDespesas: data.totalDespesas,\n totalPrevidencia: data.totalPrevidencia,\n totalPensao: data.totalPensao,\n totalDeducoes: data.totalDeducoes,\n [edit.id]: edit.value,\n })\n .then(() =>\n (setDataCripto ?? setDataBolsa)?.((data: any) => ({\n ...data,\n [edit.id]: edit.value,\n }))\n )\n .catch(() =>\n message.error(\n \"Algo inesperado aconteceu! Tente novamente mais tarde.\"\n )\n )\n .finally(() => handleEditValueCancel());\n }\n } else {\n handleEditValueCancel();\n }\n };\n return edit && edit.isEditting && edit.id === \"impostoAcumulado\" ? (\n \n } onClick={handleEditValueCancel} />\n
\n typeof value === \"number\" &&\n value >= 0 &&\n value < minDarfPrice\n ? Promise.resolve()\n : Promise.reject(),\n },\n ]}>\n \n \n \n = minDarfPrice}>\n Ok\n \n \n ) : (\n \n \n {formatCurrency(Number(data?.impostoAcumulado || 0))}\n \n {disabled && (\n }\n onClick={() => handleEditValue(\"impostoAcumulado\")}\n />\n )}\n
\n );\n },\n },\n {\n id: \"multa\",\n label: (onClick: any) => (\n \n (+) Multa de atraso{\" \"}\n }\n onClick={() => onClick(MultaModalProps)}\n />\n
\n ),\n Component: ({ data }: any) => (\n \n {formatCurrency(\n Number(data?.[\"impostoDevido\"] || 0) +\n Number(data?.[\"impostoAcumulado\"] || 0) <\n minDarfPrice\n ? 0\n : Number(data?.[\"multa\"] || 0)\n )}\n \n ),\n },\n {\n id: \"juros\",\n label: (onClick: any) => (\n \n (+) Juros e/ou encargos de atraso{\" \"}\n }\n onClick={() => onClick(JurosModalProps)}\n />\n
\n ),\n Component: ({ data }: any) => (\n \n {formatCurrency(\n Number(data?.[\"impostoDevido\"] || 0) +\n Number(data?.[\"impostoAcumulado\"] || 0) <\n minDarfPrice\n ? 0\n : Number(data?.[\"juros\"] || 0)\n )}\n \n ),\n },\n];\n\nexport const RendimentosImpostosDevidos = [\n {\n id: \"totalRendimentos\",\n label: () => \"Total rendimentos\",\n render: (data: any) => (\n \n {formatCurrency(data?.totalRendimentos ?? 0)}\n \n ),\n },\n {\n id: \"aliquotaDevida\",\n label: (onClick: any) => (\n \n (x) Alíquota devida\n }\n onClick={() => onClick(AliquotaModalProps)}\n />\n
\n ),\n render: (data: any) => (\n \n {((data?.aliquotaDevida as number) ?? 0).toFixed(2)}%\n \n ),\n },\n {\n id: \"impostoDevido\",\n label: () => \"Imposto devido\",\n render: (data: any) => (\n \n {formatCurrency(data?.impostoDevido ?? 0)}\n \n ),\n },\n];\n\nconst AliquotaModalProps = {\n title: \"Alíquota sobre Ganho Carnê-Leão\",\n content: (\n \n
\n O imposto sobre rendimentos é calculado sobre o total recebido no mês,\n mediante a aplicação da seguinte tabela progressiva mensal:\n
\n
\n a) Até R$ 2.112,00 não há impostos a pagar; \n \n b) De R$ 2.112,01 até R$ 2.826,65, a alíquota é de 7,5% sobre os\n rendimentos, com parcela a deduzir de R$ 142,80;\n \n \n c) De R$ 2.826,66 até R$ 3.751,05, a alíquota é de 15% sobre os\n rendimentos, com parcela a deduzir de R$ 354,80;\n \n \n d) De R$ 3.751,06 até R$ 4.664,68, a alíquota é de 22,5% sobre os\n rendimentos, com parcela a deduzir de R$ 636,13; e\n \n \n e) Acima de R$ 4.664,68, a alíquota é de 27,5% sobre os rendimentos,\n com parcela a deduzir de R$ 869,36.\n \n
\n
\n ),\n};\n\nexport const ModalAutonomos = {\n title: \"Despesas de Autônomos\",\n content: (\n \n
\n As seguintes despesas poderão ser deduzidas dos rendimentos recebidos de\n pessoas físicas por autônomos:\n
\n
- Aluguel do escritório/consultório
\n
- Água do escritório/consultório
\n
- Contribuições obrigatórias a entidades de classe
\n
- Condomínio do escritório/consultório
\n
- Emolumentos pagos a terceiros
\n
- Cópia e autenticação de documentos
\n
- Energia do escritório/consultório
\n
- Gás do escritório/consultório
\n
- IPTU do escritório/consultório quando pago pelo contribuinte
\n
- ISS
\n
- Material de conservação e limpeza do escritório/consultório
\n
- Material de escritório
\n
\n - Remuneração paga a terceiros, com vínculo empregatício, INSS e FGTS\n
\n
- Telefone do escritório/consultório
\n
\n ),\n};\n\nexport const ModalDependentes = {\n title: \"Dependentes\",\n content: (\n \n
A Receita Federal considera como dependentes, mas não só:
\n
\n - Companheiro(a) com quem o contribuinte tenha filho ou viva há mais de\n 5 anos, ou cônjuge;\n
\n
- Filho(a) ou enteado(a), até 21 anos de idade;
\n
\n - Filho(a) ou enteado(a), se ainda estiverem cursando estabelecimento de\n ensino superior ou escola técnica de segundo grau, até 24 anos de idade;\n
\n
\n - Irmão(ã), neto(a) ou bisneto(a), sem arrimo dos pais, de quem o\n contribuinte detenha a guarda judicial, até 21 anos, ou em qualquer\n idade, quando incapacitado física ou mentalmente para o trabalho;\n
\n
\n - Irmão(ã), neto(a) ou bisneto(a), sem arrimo dos pais, com idade de 21\n anos até 24 anos, se ainda estiver cursando estabelecimento de ensino\n superior ou escola técnica de segundo grau, desde que o contribuinte\n tenha detido sua guarda judicial até os 21 anos;\n
\n
\n - Irmão(ã), neto(a) ou bisneto(a) com deficiência, sem arrimo dos pais,\n do(a) qual o contribuinte detém a guarda judicial, em qualquer idade,\n quando a sua remuneração não exceder as deduções autorizadas por lei\n (tendo em vista a decisão do Supremo Tribunal Federal – STF, na Ação\n Direta de Inconstitucionalidade nº 5.583/DF);\n
\n
\n - Pais, avós e bisavós que, em 2021, tenham recebido rendimentos,\n tributáveis ou não, até R$ 22.847,76;\n
\n
\n – Pessoa absolutamente incapaz, da qual o contribuinte seja\n tutor ou curador.\n
\n
\n ),\n};\n\nexport const ModalPensao = {\n title: \"Despesas com Pensão Alimentícia\",\n content: (\n \n
\n Despesas com alimentos ou pensões, quando em cumprimento de decisão\n judicial ou acordo homologado judicialmente, são dedutíveis da base de\n cálculo do imposto sobre rendimentos.\n
\n
\n As despesas médicas e as de instrução, pagas em decorrência de decisão\n judicial ou acordo homologado na Justiça, não podem ser deduzidas na\n determinação da base de cálculo mensal. Esses pagamentos são dedutíveis\n apenas na Declaração de Ajuste Anual, obedecido o limite anual para os\n gastos com instrução.\n
\n
\n ),\n};\n\nexport const ModalPrevidencia = {\n title: \"Despesas com Previdência\",\n content: (\n \n
\n Despesas com contribuição para a Previdência Social da União, dos\n Estados, do Distrito Federal e dos Municípios são dedutíveis da base de\n cálculo do imposto sobre rendimentos.\n
\n
\n As contribuições para as entidades de previdência privada domiciliadas\n no País, bem como as contribuições destinadas ao Fundo de Aposentadoria\n Programada Individual (FAPI), não podem ser deduzidas na determinação da\n base de cálculo mensal. Esses pagamentos são dedutíveis apenas na\n Declaração de Ajuste Anual.\n
\n
\n ),\n};\n","import moment from \"moment\";\nimport { Typography } from \"antd\";\nimport { InfoCircleOutlined } from \"@ant-design/icons\";\nimport React, { Dispatch, SetStateAction } from \"react\";\nimport { AiOutlineMinusCircle, AiOutlinePlusCircle } from \"react-icons/ai\";\nimport Loading from \"../../Loading\";\nimport { PosicaoCardStyled } from \"./styles\";\nimport { Posicao } from \"../../../contexts/CarteiraContext\";\nimport { CustoMedioModal } from \"../../../constants/darfBolsa\";\nimport { formatCurrency, numberToPercentageWallet } from \"../../../utils\";\nimport {\n Area,\n AreaChart,\n CartesianGrid,\n ResponsiveContainer,\n Tooltip,\n XAxis,\n YAxis,\n} from \"recharts\";\n\nconst negativeToPositive = (n: number) => (n < 0 ? n * -1 : n);\n\nconst signal = (n: number) => (n > 0 ? \"+\" : n < 0 ? \"-\" : \"\");\n\nconst className = (n: number) => (n === 0 ? \"\" : n > 0 ? \"success\" : \"error\");\n\nexport const LucroPrejuizo: React.FC<{\n n?: number;\n p?: number;\n cotacao?: number;\n}> = ({ n, p, cotacao }) => (\n \n {signal(n || 0)}\n {formatCurrency(negativeToPositive(n || 0))} ({signal(n || 0)}\n {numberToPercentageWallet(negativeToPositive(p || 0))})\n {cotacao && (\n \n {signal(n || 0)}\n {formatCurrency(negativeToPositive(n ? n / cotacao : 0), \"US$ \")}\n \n )}\n \n);\n\nexport const HidenValue: React.FC<{\n block?: boolean;\n strong?: boolean;\n currency?: boolean;\n}> = ({ block, strong, currency = true }) => {\n const value = `${currency ? \"R$ \" : \"\"}****`;\n const content = strong ? {value} : value;\n return block ? {content}
: {content} ;\n};\n\ninterface PosicaoCardProps {\n key: string;\n posicao?: Partial;\n hideValues?: boolean;\n setHelpModal: Dispatch>;\n details?: any;\n loading?: boolean;\n cotacaoVenda?: number;\n}\n\nexport const PosicaoHeader = (\n posicao?: Partial,\n hideValues?: boolean\n) => {\n return (\n \n
\n {posicao?.codigo || \"GOOGL\"}\n \n {posicao?.categoria === \"Bonds\"\n ? signal(posicao?.lucroPrejuizoTotal || 0)\n : signal(posicao?.lucroPrejuizoDiario || 0)}{\" \"}\n {numberToPercentageWallet(\n posicao?.categoria === \"Bonds\"\n ? negativeToPositive(posicao?.lucroPrejuizoTotalPercentual || 0)\n : negativeToPositive(posicao?.lucroPrejuizoDiarioPercentual || 0)\n )}\n \n
\n
\n {hideValues ? (\n \n ) : (\n (posicao?.precoAtual || posicao?.posicao)\n ? formatCurrency(\n posicao?.categoria === \"TesouroDireto\"\n ? posicao?.precoAtual || 0\n : posicao?.posicao || 0\n )\n : formatCurrency(negativeToPositive(posicao?.lucroPrejuizoTotal || 0))\n )}\n
\n
\n );\n};\n\nexport const PosicaoBody = (\n posicao: Partial,\n setHelpModal: Dispatch>,\n hideValues?: boolean,\n details?: any,\n loading?: boolean,\n cotacaoVenda?: number\n) => {\n return (\n <>\n \n Quantidade: \n \n {hideValues ? : posicao.quantidade}\n \n
\n \n {posicao.categoria === \"TesouroDireto\" ? (\n Valor atualizado: \n ) : (\n Preço atual: \n )}\n \n {hideValues ? (\n \n ) : (\n formatCurrency(posicao.precoAtual || 0)\n )}\n {cotacaoVenda && (\n \n {formatCurrency(\n posicao.precoAtual ? posicao.precoAtual / cotacaoVenda : 0,\n \"US$ \"\n )}\n \n )}\n \n
\n \n {posicao?.loan?.typeLoan === \"Tomador\" && (\n <>\n Preço de venda: \n \n {hideValues ? (\n \n ) : (\n formatCurrency(posicao.precoVenda || 0)\n )}\n \n >\n )}\n
\n {posicao.hasPosition2019 && posicao?.loan?.typeLoan !== \"Tomador\" && (\n \n posicao.custoMedio === 0 || !posicao.custoMedio\n ? setHelpModal(CustoMedioModal)\n : null\n }\n >\n Custo médio de compra: \n \n {posicao.custoMedio === 0 || !posicao.custoMedio ? (\n \n ) : (\n formatCurrency(posicao.custoMedio)\n )}\n {cotacaoVenda && posicao.custoMedio && (\n \n {formatCurrency(\n posicao.custoMedio ? posicao.custoMedio / cotacaoVenda : 0,\n \"US$ \"\n )}\n \n )}\n \n
\n )}\n {!posicao.hasPosition2019 && posicao?.loan?.typeLoan !== \"Tomador\" && (\n \n {posicao.categoria === \"TesouroDireto\" ? (\n Valor Aplicado: \n ) : (\n Custo médio de compra: \n )}\n \n {posicao.custoMedio === 0 || !posicao.custoMedio ? (\n \"N/A\"\n ) : hideValues ? (\n \n ) : (\n formatCurrency(posicao.custoMedio)\n )}\n {cotacaoVenda && posicao.custoMedio && (\n \n {formatCurrency(\n posicao.custoMedio ? posicao.custoMedio / cotacaoVenda : 0,\n \"US$ \"\n )}\n \n )}\n \n
\n )}\n {posicao.categoria !== \"TesouroDireto\" &&\n posicao.categoria !== \"Bonds\" && (\n \n Lucro/prejuízo diário: \n {hideValues ? (\n \n ) : (\n \n )}{\" \"}\n
\n )}\n {posicao.hasPosition2019 && (\n \n posicao.lucroPrejuizoTotal === 0 || !posicao.lucroPrejuizoTotal\n ? setHelpModal(CustoMedioModal)\n : null\n }\n >\n Lucro/prejuízo total: {\" \"}\n {posicao.lucroPrejuizoTotal === 0 || !posicao.lucroPrejuizoTotal ? (\n \n ) : (\n \n )}\n
\n )}\n {!posicao.hasPosition2019 && (\n \n Lucro/prejuízo total: \n {posicao.custoMedio === 0 || !posicao.custoMedio ? (\n \"N/A\"\n ) : hideValues ? (\n \n ) : (\n \n )}\n
\n )}\n {details &&\n (details?.marketCap !== \"N/A\" ||\n details?.beta !== \"N/A\" ||\n details?.peRatio !== \"N/A\" ||\n details?.earningDate !== \"N/A\" ||\n details?.dividendYield !== \"N/A\" ||\n details?.targetPrice !== \"N/A\" ||\n details?.enterpriseValueRevenue !== \"N/A\" ||\n details?.enterpriseValueEbitda !== \"N/A\") && (\n <>\n \n \n
\n
\n Negociação\n \n
\n Captalização de mercado: \n \n {hideValues ? (\n \n ) : (\n <>\n {formatCurrency(\n Number(details?.marketCap || 0) / 1000000000\n )}{\" \"}\n B\n >\n )}\n \n
\n
\n Preço-alvo da ação: \n \n {hideValues ? (\n \n ) : typeof details?.targetPrice === \"string\" ? (\n details?.targetPrice\n ) : (\n formatCurrency(details?.targetPrice)\n )}\n \n
\n
\n Dividend yield (1YFW): \n \n {hideValues ? (\n \n ) : typeof details?.dividendYield === \"string\" ? (\n details?.dividendYield\n ) : (\n `${(details?.dividendYield * 100)?.toFixed(2)}%`\n )}\n \n
\n
\n Divulgação de resultados: \n \n {hideValues ? (\n \n ) : (\n details?.earningDate\n )}\n \n
\n
\n Avaliação\n \n {/*
\n Beta: \n \n {hideValues ? : typeof details?.beta !== 'string' ? details?.beta?.toFixed(2) : details?.beta}\n \n
*/}\n
\n EV/Receita (1YFW): \n \n {hideValues ? (\n \n ) : (\n `${\n typeof details?.enterpriseValueRevenue !== \"string\"\n ? `${details?.enterpriseValueRevenue?.toFixed(1)} x`\n : details?.enterpriseValueRevenue\n }`\n )}\n \n
\n
\n EV/EBITDA (1YFW): \n \n {hideValues ? (\n \n ) : (\n `${\n typeof details?.enterpriseValueEbitda !== \"string\"\n ? `${details?.enterpriseValueEbitda?.toFixed(1)} x`\n : details?.enterpriseValueEbitda\n }`\n )}\n \n
\n
\n P/E (1YFW): \n \n {hideValues ? (\n \n ) : (\n `${\n typeof details?.peRatio !== \"string\"\n ? `${details?.peRatio?.toFixed(1)}`\n : details?.peRatio\n } x`\n )}\n \n
\n
\n {details?.chart?.length && (\n
\n
\n \n \n \n \n \n \n \n {\n const date = new Date(v);\n return `${date.getMonth() + 1}/${date.getFullYear()}`;\n }}\n />\n \n \n (\n \n \n {moment(new Date(label)).format(\"DD/MM/YYYY\")}\n \n \n Cotação:{\" \"}\n {formatCurrency(Number(payload?.[0]?.value || 0))}\n \n
\n )}\n />\n \n \n \n
\n )}\n
\n >\n )}\n {loading && (\n \n \n
\n )}\n >\n );\n};\n\nexport const PosicaoCard = ({\n key,\n posicao,\n hideValues,\n setHelpModal,\n details,\n loading,\n cotacaoVenda,\n}: PosicaoCardProps) => (\n \n {posicao\n ? PosicaoBody(\n posicao,\n setHelpModal,\n hideValues,\n details,\n loading,\n cotacaoVenda\n )\n : null}\n \n);\n\nexport const ExpandIcon = ({ isActive }: { isActive?: boolean }) =>\n isActive ? (\n \n ) : (\n \n );\n","import clsx from \"clsx\";\nimport moment from \"moment\";\nimport { BiCheck } from \"react-icons/bi\";\nimport { GrUpdate } from \"react-icons/gr\";\nimport { BsDownload } from \"react-icons/bs\";\nimport { AiOutlineLock } from \"react-icons/ai\";\nimport { Fragment, useEffect, useMemo, useState } from \"react\";\nimport { InfoCircleOutlined, LoadingOutlined } from \"@ant-design/icons\";\nimport {\n Col,\n Collapse,\n Modal,\n Row,\n Typography,\n message,\n Select,\n Tooltip,\n Checkbox,\n} from \"antd\";\nimport {\n PieChart,\n Pie,\n Cell,\n Legend,\n ResponsiveContainer,\n Tooltip as RechartsTooltip,\n} from \"recharts\";\nimport Button from \"../../components/Button\";\nimport { DrawerModal } from \"../DrawerModal\";\nimport apiBolsa from \"../../services/apiBolsa\";\nimport AntButton from \"../../components/Button\";\nimport HandleTag from \"../../services/handleTag\";\nimport apiVeloPro from \"../../services/apiVeloPro\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport apiExterior from \"../../services/apiExterior\";\nimport { ReportTypeDrawer } from \"../ReportType/ReportType\";\nimport { Posicao, useCarteira } from \"../../contexts/CarteiraContext\";\nimport { chartColors, getProviderColor } from \"../../utils/walletCharts\";\nimport { BlurContent, Container, Content, IsentCardStyled } from \"./styles\";\nimport { ReactComponent as FiltroIcon } from \"../../assets/icons/filtro.svg\";\nimport { getNomeCorretora, serializePatrimonio } from \"../../utils/patrimonio\";\nimport {\n vendasIsentas,\n vendasTotaisExt,\n} from \"../../pages/HistoricBolsa/utils\";\nimport {\n ExpandIcon,\n HidenValue,\n LucroPrejuizo,\n PosicaoCard,\n} from \"./PosicaoCard\";\nimport {\n download,\n downloadPDF,\n formatCurrency,\n isMobile,\n numberToCurrency,\n} from \"../../utils\";\nimport { useNavigate } from \"react-router-dom\";\n\ntype ChartType = \"classe\" | \"posicoes\" | \"corretora\";\n\ninterface CarteiraProps {\n hideValues: boolean;\n}\n\nconst Months: any = {\n 0: \"Janeiro\",\n 1: \"Fevereiro\",\n 2: \"Março\",\n 3: \"Abril\",\n 4: \"Maio\",\n 5: \"Junho\",\n 6: \"Julho\",\n 7: \"Agosto\",\n 8: \"Setembro\",\n 9: \"Outubro\",\n 10: \"Novembro\",\n 11: \"Dezembro\",\n};\n\nexport const Carteira: React.FC = ({ hideValues }: any) => {\n const {\n user,\n showUserPlanModal,\n hasPlan: hasPremiunPlan,\n hasPermissionExterior,\n hasPermissionGeneral,\n } = useAuth();\n const hasPlan =\n (hasPremiunPlan || hasPermissionExterior || hasPermissionGeneral)\n && !user?.user?.userPlanInfoVelotax?.type.includes('BASIC');\n const { carteira, lastUpdate, loading, getCarteira, getPatrimonios, updateFilter, posicoesFilter, setPosicoesFilter } =\n useCarteira();\n\n const navigate = useNavigate();\n const [helpModal, setHelpModal] = useState();\n const [loadingReport, setLoadingReport] = useState(false);\n const [chartType, setChartType] = useState(\"classe\");\n const [classesAtivosChart, setClassesAtivosChart] = useState();\n const [corretorasAtivosChart, setCorretorasAtivosChart] = useState();\n const [ativosChart, setAtivosChart] = useState();\n const [vendasBolsa, setVendasBolsa] = useState(0);\n const [corretoras, setCorretoras] = useState([]);\n const [corretorasSemFiltro, setCorretorasSemFiltro] = useState([]);\n const [vendasExterior, setVendasExterior] = useState(0);\n const [checkboxCorretoras, setCheckboxCorretoras] = useState([]);\n const [selectedCorretoras, setSelectedCorretoras] = useState([]);\n const [reportDrawerWallet, setReportDrawerWallet] = useState(false);\n const [vendasRestantesBolsa, setVendasRestantesBolsa] = useState(0);\n const [filterCorretoraAtivo, setFilterCorretoraAtivo] = useState(\n []\n );\n const [vendasRestantesExterior, setVendasRestantesExterior] =\n useState(0);\n const [showCorretoraFilterModal, setShowCorretoraFilterModal] =\n useState(false);\n const [showPosicoesFilterModal, setShowPosicoesFilterModal] = useState(false);\n const [activeKeys, setActiveKeys] = useState([]);\n const [, setDetailedActiveKeys] = useState([]);\n const [detailedLoadingIndex, setDetailedLoadingIndex] = useState<\n number | undefined\n >(undefined);\n const [detailedPositions, setDetailedPositions] = useState([]);\n\n const checkAll = corretoras.length === checkboxCorretoras.length;\n const indeterminate =\n checkboxCorretoras.length > 0 &&\n checkboxCorretoras.length < corretoras.length;\n\n const onCloseHelpModal = () => setHelpModal(undefined);\n\n const largeChart =\n [\n ...(chartType === \"classe\"\n ? classesAtivosChart || []\n : chartType === \"corretora\"\n ? corretorasAtivosChart || []\n : ativosChart || []),\n ].length > 7;\n\n const carteiraCategorizada = useMemo(\n () =>\n (carteira?.posicoesSemFiltro || []).reduce(\n (acc, cur) => ({\n ...acc,\n [cur.categoria]: [...(acc[cur.categoria] || []), cur],\n }),\n {} as { [key: string]: Posicao[] }\n ) || {},\n [carteira.posicoesSemFiltro]\n );\n const carteiraCategorizadaFiltered = useMemo(\n () =>\n (carteira?.posicoes || []).reduce(\n (acc, cur) => ({\n ...acc,\n [cur.categoria]: [...(acc[cur.categoria] || []), cur],\n }),\n {} as { [key: string]: Posicao[] }\n ) || {},\n [carteira.posicoes]\n );\n\n const patrimonioTotalFiltrado = useMemo(\n () =>\n carteira?.posicoes?.reduce(\n (acc, cur) =>\n filterCorretoraAtivo.includes(cur.corretora)\n ? acc + cur.posicao\n : acc,\n 0\n ) || 0,\n [carteira?.posicoes, filterCorretoraAtivo]\n );\n\n const retornoFiltrado = useMemo(\n () =>\n carteira?.posicoes?.reduce(\n (acc, cur) =>\n filterCorretoraAtivo.includes(cur.corretora)\n ? {\n lucroPrejuizoDiario:\n acc.lucroPrejuizoDiario + cur.lucroPrejuizoDiario,\n lucroPrejuizoDiarioPercentualNumerator:\n acc.lucroPrejuizoDiarioPercentualNumerator +\n (cur.categoria === \"TesouroDireto\"\n ? 0\n : cur.resumo?.lucroPrejuizoDiarioPercentualNumerator),\n lucroPrejuizoDiarioPercentualDenominator:\n acc.lucroPrejuizoDiarioPercentualDenominator +\n (cur.categoria === \"TesouroDireto\"\n ? 0\n : cur.resumo?.lucroPrejuizoDiarioPercentualDenominator),\n lucroPrejuizoTotal:\n acc.lucroPrejuizoTotal + cur.lucroPrejuizoTotal,\n lucroPrejuizoTotalPercentualNumerator:\n acc.lucroPrejuizoTotalPercentualNumerator +\n cur.resumo?.lucroPrejuizoTotalPercentualNumerator,\n lucroPrejuizoTotalPercentualDenominator:\n acc.lucroPrejuizoTotalPercentualDenominator +\n cur.resumo?.lucroPrejuizoTotalPercentualDenominator,\n }\n : acc,\n {\n lucroPrejuizoDiario: 0,\n lucroPrejuizoDiarioPercentualNumerator: 0,\n lucroPrejuizoDiarioPercentualDenominator: 0,\n lucroPrejuizoTotal: 0,\n lucroPrejuizoTotalPercentualNumerator: 0,\n lucroPrejuizoTotalPercentualDenominator: 0,\n }\n ) || {\n lucroPrejuizoDiario: 0,\n lucroPrejuizoDiarioPercentualNumerator: 0,\n lucroPrejuizoDiarioPercentualDenominator: 0,\n lucroPrejuizoTotal: 0,\n lucroPrejuizoTotalPercentualNumerator: 0,\n lucroPrejuizoTotalPercentualDenominator: 0,\n },\n [carteira?.posicoes, filterCorretoraAtivo]\n );\n\n const onChangeActiveKeys = (keys: string | string[]) => {\n const isOpenEvent = keys.length > activeKeys.length;\n const lastOpened = keys[keys.length - 1];\n\n setActiveKeys(keys as string[]);\n let codigo =\n lastOpened?.replaceAll(\" \", \"\").split(\"-\")[3] ||\n lastOpened?.replaceAll(\" \", \"\").split(\"-\")[2];\n let market =\n lastOpened?.replaceAll(\" \", \"\").split(\"-\")[0] ||\n lastOpened?.replaceAll(\" \", \"\").split(\"-\")[1];\n let index = lastOpened?.replaceAll(\" \", \"\").split(\"-\")[1];\n if (\n lastOpened.includes(\"Imobiliário\") ||\n lastOpened.includes(\"Traded\") ||\n lastOpened.includes(\"Brazilian\")\n ) {\n codigo = lastOpened?.replaceAll(\" \", \"\").replace(\"-\", \"\").split(\"-\")[2];\n index = lastOpened?.replaceAll(\" \", \"\").replace(\"-\", \"\").split(\"-\")[1];\n }\n\n if (lastOpened && isOpenEvent) {\n setDetailedActiveKeys((detailedActiveKeys) => {\n if (\n !detailedActiveKeys.includes(lastOpened) &&\n keys.length > detailedActiveKeys.length\n ) {\n setDetailedLoadingIndex(Number(index));\n apiBolsa\n .get(`/warren/detalied/${codigo}`, {\n params: {\n exterior:\n market.toLowerCase() === \"exterior\" ? true : undefined,\n },\n })\n .then(({ data }: any) => {\n setDetailedPositions((positions) => [\n ...positions,\n {\n index,\n codigo: codigo,\n details: data,\n },\n ]);\n return [...detailedActiveKeys, lastOpened];\n })\n .finally(() => {\n setDetailedLoadingIndex(undefined);\n });\n }\n return detailedActiveKeys;\n });\n }\n };\n\n useEffect(() => {\n getCarteira();\n }, [getCarteira]);\n\n useEffect(() => {\n apiBolsa\n .get(\"/velotax/historic\", {\n params: {\n month: new Date().getMonth() + 1,\n },\n })\n .then((res) => {\n const isentas = vendasIsentas(res?.data);\n setVendasBolsa(isentas);\n setVendasRestantesBolsa(20000 - isentas);\n });\n }, []);\n\n useEffect(() => {\n apiExterior\n .get(\"/velotax/exterior/historic\", {\n params: {\n month: new Date().getMonth() + 1,\n },\n })\n .then((res) => {\n const isentas = vendasTotaisExt(res?.data)\n setVendasExterior(isentas);\n setVendasRestantesExterior(35000 - isentas);\n });\n }, []);\n\n useEffect(() => {\n const { classesAtivos, patrimonio, posicoes, corretoras, corretorasSemFiltro } =\n serializePatrimonio(carteira, selectedCorretoras);\n setClassesAtivosChart(classesAtivos);\n setCorretorasAtivosChart(patrimonio);\n setAtivosChart(posicoes);\n setCorretoras(corretoras);\n setCorretorasSemFiltro(corretorasSemFiltro)\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [loading, selectedCorretoras]);\n\n const downloadAction = async (type: string) => {\n try {\n setLoadingReport(true);\n if (type === \"pdf\") {\n const queryCorretoras =\n filterCorretoraAtivo?.length > 0\n ? `?corretoras=${JSON.stringify(filterCorretoraAtivo).replace(\n /\\]|\"|\\[/g,\n \"\"\n )}`\n : \"\";\n const res = await apiBolsa.get(\n `/warren/wallet/report${queryCorretoras}`\n );\n if (res.data.file && !isMobile()) {\n downloadPDF(res.data.file, \"Meus investimentos\");\n } else if (res.data.fileUrl && !isMobile()) {\n download(res.data.fileUrl);\n }\n message.success(\"O relatório foi enviado para o seu e-mail\");\n } else if (type === \"xlsx\") {\n if (filterCorretoraAtivo?.length > 0) {\n const res = await apiVeloPro.post(\n \"/calc/downloadRelatorioPatrimonioXlsx\",\n {\n username: user.user.cpf,\n corretoras: filterCorretoraAtivo,\n }\n );\n if (!isMobile()) download(res?.data?.fileUrl);\n message.success(\"O relatório foi enviado para o seu e-mail\");\n } else {\n const data = await getPatrimonios(user?.user?.cpf);\n if (!isMobile()) download(data.fileUrl);\n message.success(\"O relatório foi enviado para o seu e-mail\");\n }\n }\n } catch (err: any) {\n message.error(\n `Não foi possível baixar o relatório de investimentos. Tente novamente mais tarde`\n );\n } finally {\n setLoadingReport(false);\n }\n };\n\n const renderLegend = (value: string, entry: any) => {\n return {value} ;\n };\n\n const renderIsentCard = (categoria: string) => {\n if (categoria === \"Ações\" || categoria === \"Exterior\") {\n const vendas = categoria === \"Ações\" ? vendasBolsa : vendasExterior;\n const restante =\n categoria === \"Ações\" ? vendasRestantesBolsa : vendasRestantesExterior;\n const limit = categoria === \"Ações\" ? 20000 : 35000;\n const tipVendas = vendas\n ? `R$ ${String(\n vendas.toLocaleString(\"pt-br\", {\n minimumFractionDigits: 2,\n maximumFractionDigits: 2,\n })\n )}`\n : \"R$ 0,00\";\n const complete = vendas ? (vendas / limit) * 100 : 0;\n\n return (\n \n {restante <= 0 ? (\n \n O seu limite de vendas isentas foi atingido neste mês\n \n ) : (\n \n Você ainda possui{\" \"}\n \n {formatCurrency(restante)}{\" \"}\n {\" \"}\n em vendas isentas em{\" \"}\n \n {Months[new Date().getMonth()]} {new Date().getFullYear()}\n \n \n )}\n \n
target.parentElement!}\n >\n = 100 ? 100 : complete}%`,\n borderBottomRightRadius: complete >= 100 ? \"50px\" : 0,\n borderTopRightRadius: complete >= 100 ? \"50px\" : 0,\n }}\n >\n
\n {vendas > 5000 ? formatCurrency(vendas) : \"\"}\n
\n
\n \n
\n {/* */}\n \n );\n }\n };\n\n const renderCustomizedLabel = ({\n cx,\n cy,\n midAngle,\n innerRadius,\n outerRadius,\n percent,\n payload,\n }: any) => {\n const RADIAN = Math.PI / 180;\n const radius = innerRadius + (outerRadius - innerRadius) * 0.5;\n const x = cx + radius * Math.cos(-midAngle * RADIAN);\n const y = cy + radius * Math.sin(-midAngle * RADIAN);\n let fill = chartColors.find(\n (colors) => colors.background === payload.fill\n )?.label;\n if (payload.fill === \"#FCC102\") fill = \"#5a4709\";\n\n return (\n \n {percent * 100 > 5 && `${(percent * 100).toFixed(0)}%`}\n \n );\n };\n\n return (\n \n {/* \n Meus investimentos\n {hasPlan && (\n : }\n onClick={() => downloadAction()}\n />\n )}\n : }\n />\n */}\n {/* \n \n */}\n \n \n
}\n className=\"download-report-btn filter-btn\"\n onClick={() => setShowCorretoraFilterModal(true)}\n >\n Filtrar\n \n {hasPlan &&
target.parentElement!}\n >\n : }\n onClick={() => setReportDrawerWallet(true)}\n />\n }\n
\n \n \n Investimento total: \n \n {loading ? (\n \n ) : !hideValues ? (\n formatCurrency(\n filterCorretoraAtivo.length > 0\n ? patrimonioTotalFiltrado\n : carteira.patrimonioTotal\n )\n ) : (\n \n )}\n \n \n \n \n \n \n Retorno diário:{\" \"}\n {loading ? (\n \n ) : !hideValues ? (\n 0\n ? retornoFiltrado.lucroPrejuizoDiario\n : carteira.lucroPrejuizoDiario\n }\n p={\n filterCorretoraAtivo.length > 0\n ? retornoFiltrado.lucroPrejuizoDiarioPercentualNumerator /\n retornoFiltrado.lucroPrejuizoDiarioPercentualDenominator\n : carteira.lucroPrejuizoDiarioPercentual\n }\n />\n ) : (\n \n )}\n \n \n \n \n Retorno total:{\" \"}\n {loading ? (\n \n ) : !hideValues ? (\n 0\n ? retornoFiltrado.lucroPrejuizoTotal\n : carteira.lucroPrejuizoTotal\n }\n p={\n filterCorretoraAtivo.length > 0\n ? retornoFiltrado.lucroPrejuizoTotalPercentualNumerator /\n retornoFiltrado.lucroPrejuizoTotalPercentualDenominator\n : carteira.lucroPrejuizoTotalPercentual\n }\n />\n ) : (\n \n )}\n \n \n
\n \n
\n downloadAction(type)}\n onClose={() => setReportDrawerWallet(false)}\n />\n \n \n {loading ? (\n \n \n
\n ) : carteira?.posicoes?.length > 0 ? (\n <>\n {!loading ? (\n \n <>\n \n \n \n {(chartType === \"classe\"\n ? classesAtivosChart\n : chartType === \"posicoes\"\n ? ativosChart\n : corretorasAtivosChart\n )?.map((entry: any, index: number) => (\n | \n ))}\n \n {\n return `${numberToCurrency(Number(value))} (${String(\n payload.percentage\n ).replace(\".\", \",\")}%)`;\n }}\n />\n \n \n setChartType(value)}\n getPopupContainer={(target) => target.parentElement!}\n >\n \n Por classe de ativo\n \n \n Por corretora\n \n Por ativo \n \n
\n >\n \n ) : undefined}\n {!hasPlan ? (\n <>\n ExpandIcon({ isActive })}\n >\n {carteira.posicoes.slice(0, 1).map((posicao, index) => (\n \n {PosicaoCard({\n key: index.toString(),\n posicao,\n hideValues,\n setHelpModal,\n })}\n \n ))}\n \n \n ExpandIcon({ isActive })}\n >\n {Array.from(Array(3).keys()).map((index) => (\n \n {PosicaoCard({\n key: index.toString(),\n hideValues,\n setHelpModal,\n posicao: {\n lucroPrejuizoDiario: 0.01,\n },\n })}\n \n ))}\n \n \n Contrate o plano premium para ver a lista completa de ativos\n
}\n onClick={() => {\n // showUserPlanModal(true);\n navigate('/planos')\n HandleTag(\"54\");\n }}\n >\n PREMIUM\n \n
\n \n >\n ) : (\n ExpandIcon({ isActive })}\n >\n {(filterCorretoraAtivo.length > 0\n ? Object.keys(carteiraCategorizadaFiltered).filter(\n (categoria) =>\n carteiraCategorizadaFiltered[categoria].filter((item: any) =>\n filterCorretoraAtivo.includes(item?.corretora)\n ).length > 0\n )\n : Object.keys(carteiraCategorizadaFiltered)\n ).map((categoria) => (\n \n \n {categoria === \"TesouroDireto\"\n ? \"Tesouro Direto\"\n : categoria}\n \n \n \n >\n }\n >\n ExpandIcon({ isActive })}\n onChange={onChangeActiveKeys}\n >\n {categoria !== \"Exterior\" && renderIsentCard(categoria)}\n {(filterCorretoraAtivo.length > 0\n ? carteiraCategorizadaFiltered[categoria].filter((item: any) =>\n filterCorretoraAtivo.includes(item?.corretora)\n )\n : carteiraCategorizadaFiltered[categoria]\n )\n .sort(\n (a, b) =>\n ((b.categoria === \"TesouroDireto\"\n ? b.precoAtual\n : b.posicao) || 0) -\n ((a.categoria === \"TesouroDireto\"\n ? a.precoAtual\n : a.posicao) || 0)\n )\n .map((posicao, index) => (\n \n {PosicaoCard({\n key: `${categoria}-${index}-${posicao.codigo}`,\n posicao,\n hideValues,\n setHelpModal,\n cotacaoVenda:\n categoria === \"Exterior\" &&\n carteira.cotacaoVenda > 0\n ? carteira.cotacaoVenda\n : undefined,\n details: detailedPositions?.find(\n (el: any) =>\n el.codigo === posicao.codigo &&\n Number(el.index) === index\n )?.details,\n loading: detailedLoadingIndex === index,\n })}\n \n ))}\n \n \n ))}\n \n )}\n >\n ) : (\n \n Nenhuma posição encontrada\n \n )}\n\n \n : }\n onClick={() => getCarteira(true)}\n className=\"button-refresh-wallet\"\n />\n Atualizado em{\" \"}\n {moment(lastUpdate).utc().local().format(\"DD/MM/YYYY à[s] HH:mm\")}\n \n\n {hasPlan && carteira.posicoes?.length > 0 && (\n \n \n \n As posições são atualizadas em até quatro dias úteis após sua\n negociação (D+4)\n \n \n \n Bonds e/ou operações fora da bolsa de valores não são apresentadas\n na listagem acima.\n \n {/* \n \n Os preços dos ativos custodiados no exterior são convertidos para\n reais pela cotação do dólar de compra fixada pelo Banco Central do\n Brasil (PTAX) na data do último fechamento.\n \n \n O custo médio de compra de ativos no\n exterior é apresentado para fins ilustrativos de acompanhamento de\n carteira apenas. Para cálculo de imposto de renda, utilizar\n valores de aquisição considerando a origem dos recursos para\n cálculo de ganho de capital.\n */}\n \n )}\n \n \n {helpModal?.content}\n \n {\n setShowCorretoraFilterModal(false);\n }}\n bodyStyle={{paddingTop: '1rem'}}\n footer={\n }\n onClick={() => {\n setShowCorretoraFilterModal(false);\n setSelectedCorretoras(checkboxCorretoras);\n setFilterCorretoraAtivo(\n checkboxCorretoras.map((corretora: string) =>\n getNomeCorretora({ shortName: corretora })\n )\n );\n updateFilter(posicoesFilter || [])\n }}\n >\n Confirmar\n \n }\n >\n \n \n \n \n Corretoras \n \n \n {\n setPosicoesFilter([])\n setCheckboxCorretoras(e.target.checked ? corretorasSemFiltro : []);\n }}\n >\n Todas as corretoras\n \n {\n setPosicoesFilter([])\n setCheckboxCorretoras(e);\n }}\n >\n \n {corretorasSemFiltro.map((corretora, i) => (\n \n {corretora} \n \n ))}\n
\n \n \n
\n \n
\n \n
\n Filtrar por posições\n
\n
\n {\n setCheckboxCorretoras([])\n setPosicoesFilter(e)\n }}\n options={Object.keys(carteiraCategorizada).map((categoria) => ({\n label: categoria,\n options: carteiraCategorizada[categoria].map((c: any) => ({\n label: c.codigo,\n value: c.codigo,\n }))\n }))}\n />\n \n \n );\n};\n","import styled from \"styled-components\";\n\nexport const NotificationContainer = styled.div`\n h3 {\n border-bottom: 2px solid var(--gray-2);\n padding-bottom: 0.5rem;\n color: var(--velotax-font-color-light);\n }\n\n .divider {\n width: 100%;\n max-width: 80px;\n background-color: var(--ant-primary-color);\n height: 2px;\n margin-bottom: 1rem;\n }\n\n .illustration {\n display: flex;\n justify-content: center;\n align-items: center;\n flex-direction: column;\n position: relative;\n margin-top: 4rem;\n\n span {\n margin-right: 1rem;\n color: var(--gray-2);\n }\n\n h5 {\n font-weight: 500;\n color: var(--velotax-font-color-light);\n }\n\n button, span {\n display: flex;\n justify-content: center;\n align-items: center;\n color: black !important;\n }\n span {\n margin: 0 0 0 0.5rem;\n }\n button {\n max-width: 160px;\n height: 52px;\n margin-right: auto;\n margin-top: 1rem;\n }\n }\n\n .loading {\n display: flex; \n \n h5 {\n margin-left: 1rem;\n }\n }\n\n .card {\n background-color: var(--ant-primary-1);\n box-shadow: 0 2px 5px 0 rgba(0, 0, 0, 0.15);\n border-radius: 6px;\n margin-bottom: 1rem;\n padding: 1rem;\n position: relative;\n cursor: pointer;\n\n &:hover {\n transition: 0.3s;\n background-color: var(--ant-primary-2);\n }\n\n .loading {\n width: 100%;\n height: 100%;\n position: absolute;\n top: 0;\n left: 0;\n background-color: rgba(0, 0, 0, 0.2);\n display: flex;\n justify-content: center;\n align-items: center;\n }\n\n .header {\n display: flex;\n align-items: center;\n width: 100%;\n margin-bottom: 8px;\n\n h5 {\n flex-grow: 1;\n margin-bottom: 0;\n }\n\n svg {\n color: var(--ant-primary-color);\n font-size: 20px;\n margin-right: 4px;\n }\n\n span {\n color: var(--velotax-font-color-ghost);\n }\n\n .actionButton {\n color: var(--gray-2);\n margin: 0;\n transition: 0.3s;\n cursor: pointer;\n\n &:hover {\n color: var(--ant-primary-color);\n transition: 0.3s;\n }\n }\n }\n\n .body {\n span {\n margin: 0;\n color: var(--black);\n font-weight: 400;\n }\n }\n }\n\n .viewed {\n background-color: var(--velotax-ghost);\n cursor: default;\n h5,\n svg,\n span {\n color: var(--velotax-font-color-ghost) !important;\n }\n &:hover {\n background-color: var(--velotax-ghost);\n }\n }\n\n .section {\n display: flex;\n justify-content: space-between;\n align-items: center;\n margin-top: 2rem;\n\n h5 {\n text-transform: uppercase;\n font-size: 14px;\n color: var(--velotax-font-color-light);\n }\n\n button {\n color: var(--gray-2);\n transition: 0.3s;\n text-transform: uppercase;\n font-size: 13px;\n font-weight: 500;\n &:hover {\n color: var(--ant-primary-color);\n transition: 0.3s;\n }\n }\n }\n`;\n","export default __webpack_public_path__ + \"static/media/notification.d3262cf7.svg\";","import { useState } from \"react\";\nimport { Drawer, Typography, message } from \"antd\";\nimport { LoadingOutlined } from \"@ant-design/icons\";\nimport { AiOutlineArrowLeft } from \"react-icons/ai\";\nimport { HiOutlineCurrencyDollar } from \"react-icons/hi\";\nimport api from \"../../services/apiBolsa\";\nimport { NotificationContainer } from \"./styles\";\nimport { isMobile } from \"../../utils\";\nimport notificationImage from \"../../assets/notification.svg\";\nimport { useNotification } from \"../../contexts/NotificationContext\";\n\ninterface IProps {\n open: boolean;\n onClose:\n | ((\n e: React.MouseEvent | React.KeyboardEvent\n ) => void)\n | undefined;\n}\n\nexport const NotificationDrawer = ({ open, onClose }: IProps) => {\n const { notification, setNotification, loading } = useNotification();\n const [loadingIndex, setLoadingIndex] = useState(2);\n const lidas = (notification || []).filter((n) => n.viewed);\n const naoLidas = (notification || []).filter((n) => !n.viewed);\n\n const removeNotification = (index?: number) => {\n const notifications = notification.map((n, i: number) => {\n if (index === undefined || index === i)\n return { ...n, viewed: true, viewedDate: new Date() };\n return n;\n });\n\n setLoadingIndex(index === undefined ? \"all\" : index);\n api\n .patch(`/warren/notification`, {\n notifications,\n })\n .then(() => {\n setNotification(notifications);\n setLoadingIndex(null);\n })\n .catch(() => {\n message.error(\"Não foi possível remover notificação\");\n });\n };\n\n return (\n }\n title={\n \n Voltar\n
\n }\n {...(isMobile() ? { width: \"100%\" } : {})}\n >\n \n \n\n {(loading || !notification?.length) && (\n \n
\n {loading ? (\n
\n \n \n Buscando notificações\n \n
\n ) : !notification?.length ? (\n
\n Você não possui novas notificações.\n \n ) : (\n <>>\n )}\n
\n )}\n\n {}\n {!!naoLidas?.length && !loading && (\n \n Não lidas \n removeNotification()}>\n Marcar todas como lida\n \n
\n )}\n {!loading &&\n (notification || [])?.map((item, i) => {\n if (item.viewed) return <>>;\n return (\n \n
{\n if (!(loadingIndex === i || loadingIndex === \"all\"))\n removeNotification(i);\n }}\n >\n {(loadingIndex === i || loadingIndex === \"all\") && (\n
\n \n
\n )}\n
\n \n \n Rendimentos a receber\n \n {item?.date && (\n \n {new Date(item?.date)?.toLocaleDateString(\"pt-BR\", {\n year: \"numeric\",\n month: \"2-digit\",\n day: \"2-digit\",\n })}\n \n )}\n
\n
\n {item.message} \n
\n
\n
\n );\n })}\n\n {/* Notificações Lidas */}\n {!!lidas?.length && !loading && (\n \n Lidas \n
\n )}\n {!loading &&\n (notification || [])?.map((item, i) => {\n if (!item.viewed) return <>>;\n return (\n \n
\n
\n \n \n Rendimentos a receber\n \n {item?.date && (\n \n {new Date(item?.date)?.toLocaleDateString(\"pt-BR\", {\n year: \"numeric\",\n month: \"2-digit\",\n day: \"2-digit\",\n })}\n \n )}\n
\n
\n {item.message} \n
\n
\n
\n );\n })}\n \n \n );\n};\n","import styled from \"styled-components\";\n\nexport const AvisosContainer = styled.div`\n display: flex;\n justify-content: center;\n align-items: center;\n\n .client-integration-card {\n padding: 24px;\n border-radius: 8px;\n background-color: var(--ant-primary-1);\n border: 1px solid var(--ant-primary-color);\n display: flex;\n align-items: start;\n margin-bottom: 1rem;\n &:last-of-type {\n /* margin-bottom: 48px; */\n }\n & > svg {\n min-width: 32px;\n min-height: 32px;\n color: var(--ant-primary-color);\n stroke: var(--ant-primary-color);\n &.fill {\n fill: var(--ant-primary-color);\n }\n }\n\n .client-integration-card-content {\n width: 100%;\n margin-left: 16px;\n h3 {\n color: #000;\n font-size: 16px;\n font-weight: 600;\n line-height: 24px;\n margin-bottom: 8px;\n }\n div.ant-typography {\n color: #000;\n font-size: 14px;\n font-weight: 300;\n line-height: 20px;\n width: calc(100% - 48px);\n }\n .ant-btn {\n padding: 0;\n border: none;\n height: 40px;\n font-size: 14px;\n font-weight: 500;\n margin: -12px 0;\n }\n button {\n display: flex;\n align-items: center;\n\n svg {\n margin-left: 4px;\n }\n }\n }\n }\n\n div:first-of-type {\n width: 100%;\n }\n\n .title {\n h3 {\n font-weight: 600;\n }\n }\n .divider {\n width: 100%;\n background-color: var(--ant-primary-color);\n height: 2px;\n margin-bottom: 2rem;\n }\n\n .plan-boxes {\n flex-wrap: nowrap;\n margin-top: 1rem !important;\n margin-bottom: 1rem !important;\n }\n\n .ant-divider {\n margin: 8px 0;\n }\n\n .ant-btn {\n height: 56px;\n\n span {\n font-weight: 500;\n svg {\n display: none;\n }\n }\n }\n .total-contratado {\n display: none;\n }\n .dismiss {\n cursor: pointer;\n transition: 0.3s;\n &:hover {\n color: var(--ant-primary-7);\n transition: 0.3s;\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n .ant-card-body {\n padding: 16px;\n }\n }\n`;\n","import { Drawer, DrawerProps, Modal, ModalProps } from \"antd\";\nimport { isMobile } from \"../../../utils\";\n\nexport interface DrawerModalProps\n extends Omit,\n Omit {\n visible?: boolean;\n onCancel?: (e: any) => void;\n}\n\nexport const DrawerModal: React.FC = ({\n height,\n visible,\n onCancel,\n children,\n className,\n getContainer,\n ...props\n}) => {\n return isMobile() ? (\n \n {children}\n \n ) : (\n \n {children}\n \n );\n};\n","import styled from \"styled-components\";\n\nexport const B3Link = styled.a`\n margin: auto;\n font-size: 10px;\n text-align: center;\n white-space: normal;\n display: inline-block;\n word-break: break-all;\n color: rgb(1, 103, 153);\n\n &:hover {\n text-decoration: underline;\n color: rgb(1, 103, 153);\n }\n`;\n\nexport const CopyLink = styled.button`\n color: #0566cd;\n font-weight: 500;\n text-decoration: underline;\n\n &:hover {\n text-decoration: underline;\n color: #06417f;\n }\n`;\n\nexport const ShareItem = styled.div`\n display: flex;\n flex-direction: column;\n align-items: center;\n justify-content: center;\n margin-right: 1rem;\n cursor: pointer;\n\n &:hover {\n text-decoration: underline;\n color: #06417f;\n }\n\n img {\n width: 40px;\n height: 40px;\n }\n\n p {\n font-size: 11px;\n margin: 0.5rem 0;\n }\n`;\n\nexport const Instructions = styled.p`\n font-size: 13px;\n margin-bottom: 4px;\n`;\n","export default __webpack_public_path__ + \"static/media/Integracao_XP_B3.635b7422.gif\";","export default __webpack_public_path__ + \"static/media/integracao_warren_exterior.abe871c7.gif\";","import { Typography, Image, message } from \"antd\";\nimport { DrawerModal } from \"../DrawerModal\";\nimport { B3Link, CopyLink, Instructions } from \"./styles\";\nimport b3IntegrationExample from \"../../../assets/Integracao_XP_B3.gif\";\nimport warrenIntegrationExample from \"../../../assets/integracao_warren_exterior.gif\";\n\nexport type TIntegrationModal = \"B3\" | \"Warren Internacional\" | null;\ninterface IProps {\n integrationModal: TIntegrationModal;\n setIntegrationModal: React.Dispatch>;\n}\n\nconst b3Link =\n \"https://b3investidor.b2clogin.com/b3Investidor.onmicrosoft.com/oauth2/v2.0/authorize?p=B2C_1A_FINTECH&client_id=069ecfdf-7471-48eb-be82-d50799c99e5c&nonce=defaultNonce&redirect_uri=https%3A%2F%2Fwww.investidor.b3.com.br&scope=openid&response_type=code&prompt=login\";\n// const b3Encoded =\n// \"https%3A%2F%2Fb3investidor.b2clogin.com%2Fb3Investidor.onmicrosoft.com%2Foauth2%2Fv2.0%2Fauthorize%3Fp%3DB2C_1A_FINTECH%26client_id%3D069ecfdf-7471-48eb-be82-d50799c99e5c%26nonce%3DdefaultNonce%26redirect_uri%3Dhttps%253A%252F%252Fwww.investidor.b3.com.br%26scope%3Dopenid%26response_type%3Dcode%26prompt%3Dlogin\";\n\nexport const IntegrationModal = ({\n integrationModal,\n setIntegrationModal,\n}: IProps) => {\n return (\n {\n setIntegrationModal(null);\n }}\n >\n {integrationModal === \"B3\" ? (\n \n Permita a conexão entre Velotax e B3 para leitura das operações\n realizadas na bolsa de valores.\n \n ) : (\n \n Permita a conexão entre Velotax e Warren Internacional para leitura das\n operações realizadas.\n \n )}\n \n \n
\n\n {integrationModal === \"B3\" ? (\n <>\n \n {b3Link}\n \n\n \n {\n navigator.clipboard.writeText(b3Link);\n message.success(\"Link copiado\");\n }}\n >\n Copiar link\n \n
\n >\n ) : (\n <>\n \n Instruções\n \n \n \n Passo 1 . Acesse a aba de{\" \"}\n Investimento Global dentro do aplicativo da Warren\n \n \n Passo 2 . Deslize a tela para baixo e clique em{\" \"}\n Imposto de Renda e DARF \n \n \n Passo 3 . Clique em{\" \"}\n Calcular Imposto de Renda e aceite os termos\n \n
\n >\n )}\n \n );\n};\n","// Config\n// Components\nimport moment from \"moment\";\nimport { AvisosContainer } from \"./styles\";\nimport { useEffect, useState } from \"react\";\nimport { Button, Spin, Typography, message } from \"antd\";\nimport { FiArrowRight } from \"react-icons/fi\";\nimport { AiOutlineLink } from \"react-icons/ai\";\nimport { BiX } from \"react-icons/bi\";\nimport api from \"../../../services/api\";\nimport { useAuth } from \"../../../contexts/AuthContext\";\nimport { IntegrationModal } from \"../IntegrationModal\";\n// Assets\n\n// Types\ninterface IProps {\n}\nexport type TIntegrationModal = \"B3\" | \"Warren Internacional\" | null;\n\nexport const Avisos = ({\n}: IProps) => {\n const { user } = useAuth();\n const [dismissBolsa, setDismissBolsa] = useState(false);\n const [dismissExterior, setDismissExterior] = useState(false);\n const [loading, setLoading] = useState(false);\n const [integrationModal, setIntegrationModal] = useState<\"B3\" | \"Warren Internacional\" | null>(null); \n const getCloseBanner = async () => {\n try {\n setLoading(true);\n const { data } = await api.post(\"user/xp/close-banner\", { cpf: user?.user?.cpf })\n setDismissBolsa(data.closeBannerBolsaXp)\n setDismissExterior(data.closeBannerExteriorXp)\n } catch (error) {\n message.error(\"Algo deu errado, tente mais tarde!\");\n } finally {\n setLoading(false);\n }\n }\n\n const handleDismissBolsa = async () => {\n try {\n setLoading(true);\n setDismissBolsa(true);\n const { data } = await api.put(\"user/xp/close-banner\", { cpf: user?.user?.cpf, closeBannerBolsaXp: true })\n setDismissBolsa(data.closeBannerBolsaXp)\n } catch (error) {\n message.error(\"Algo deu errado, tente mais tarde!\");\n } finally {\n setLoading(false);\n }\n }\n\n const handleDismissExterior = async () => {\n try {\n setLoading(true);\n setDismissExterior(true);\n const { data } = await api.put(\"user/xp/close-banner\", { cpf: user?.user?.cpf, closeBannerExteriorXp: true })\n setDismissExterior(data.closeBannerExteriorXp)\n } catch (error) {\n message.error(\"Algo deu errado, tente mais tarde!\");\n } finally {\n setLoading(false);\n }\n }\n\n useEffect(() => {\n getCloseBanner()\n }, []);\n \n return (\n \n \n
\n { user.user.isIntegratedBolsa || !dismissBolsa && (\n \n
\n
\n \n Integração com bolsa pendente\n \n \n Conceda a permissão na B3 para que seja\n possível acessar suas operações da bolsa de valores de forma\n automática.\n \n { setIntegrationModal(\"B3\") }}\n >\n Saiba mais\n \n \n
\n
{ handleDismissBolsa() }}\n />\n \n )}\n { user.user.isIntegratedExterior || !dismissExterior && (\n \n
\n
\n \n Conecte com a Warren Internacional\n \n \n Conceda a permissão na Warren Internacional\n para que seja possível acessar suas operações no exterior de\n forma automática.\n \n {\n console.log('caiu aqui');\n setIntegrationModal(\"Warren Internacional\");\n }}\n >\n Saiba mais\n \n \n
\n
{ handleDismissExterior() }}\n />\n \n )}\n \n
\n \n \n );\n};\n","import { useState } from \"react\";\nimport { BiBell } from \"react-icons/bi\";\nimport { Badge, Tabs, Typography } from \"antd\";\nimport { BsEye, BsEyeSlash } from \"react-icons/bs\";\nimport { Container, Content } from \"./styles\";\nimport AntButton from \"../../components/Button\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport { Proventos } from \"../../components/Proventos\";\nimport { Carteira } from \"../../components/MeusInvestimentos/index\";\nimport { useNotification } from \"../../contexts/NotificationContext\";\nimport { NotificationDrawer } from \"../../components/Notification/Notification\";\nimport { Avisos } from \"../../components/MeusInvestimentos/Avisos\";\nimport { useBroker } from \"../../contexts/BrokerContext\";\nimport { useCarteira } from \"../../contexts/CarteiraContext\";\n\ninterface CarteiraProps { }\n\nexport const Wallet: React.FC = () => {\n const { notification } = useNotification();\n const { integration } = useBroker();\n const {\n user,\n hasPlan: hasPremiunPlan,\n hasPermissionExterior,\n hasPermissionGeneral,\n } = useAuth();\n const hasPlan =\n (hasPremiunPlan || hasPermissionExterior || hasPermissionGeneral) && !user?.user?.userPlanInfoVelotax?.type.includes('BASIC');\n\n const { carteira } = useCarteira();\n\n const [notificationDrawerOpen, setNotificationDrawerOpen] = useState(false);\n const [hideValues, setHideValues] = useState(false);\n\n return (\n \n \n
\n Meus investimentos\n : }\n onClick={() => setHideValues((value) => !value)}\n />\n \n {hasPlan && (\n
\n
!n.viewed)?.length || 0}\n >\n }\n onClick={() => {\n setNotificationDrawerOpen(true);\n }}\n />\n \n
\n )}\n
\n \n {carteira.erroB3\n ? No momento nossa integração com os sistemas da B3 está instável. Tente novamente mais tarde. \n : <>\n \n {(!integration.integrated && !user.user.isIntegratedExterior) ? (<>>) : (\n \n \n \n \n \n \n \n \n )}\n >\n }\n \n setNotificationDrawerOpen(false)}\n />\n \n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n .page-title.conta {\n display: flex;\n align-items: baseline;\n justify-content: space-between;\n button {\n display: flex;\n align-items: center;\n height: 48px;\n padding: 0 24px;\n background-color: var(--velotax-background-color);\n svg {\n fill: var(--ant-primary-color);\n }\n }\n }\n p {\n width: 700px;\n margin: 0 auto 0.3rem;\n font-size: 1rem;\n line-height: 1.5rem;\n color: var(--velotax-font-color-light);\n\n span {\n cursor: pointer;\n font-weight: 500;\n color: var(--ant-primary-color);\n }\n }\n\n h4 {\n color: var(--velotax-font-color-light);\n }\n\n .ant-descriptions:last-of-type {\n .ant-descriptions-item {\n padding-bottom: 0;\n }\n }\n\n .ant-descriptions {\n h4,\n .ant-descriptions-item-content {\n color: var(--velotax-font-color-dark);\n }\n .ant-descriptions-header {\n margin-bottom: 0px;\n }\n }\n\n .excluir-conta-container {\n width: 700px;\n margin: 16px auto 0 auto;\n }\n\n .parametrizar-custos-container {\n width: 700px;\n margin: 0px auto;\n button { \n color: var(--ant-primary-color);\n background-color: var(--velotax-background-color);\n padding: 0.5rem 1rem;\n transition: .3s;\n \n &:hover{\n color: var(--ant-primary-color-hover);\n transition: .3s;\n }\n\n svg {\n margin-right: 0.5rem;\n transform: translateY(-1px);\n }\n }\n }\n\n .cancelar-plano-container {\n width: 700px;\n margin: -32px auto 32px;\n }\n\n .content-button {\n display: flex;\n margin: 16px auto 0;\n align-items: center;\n justify-content: right;\n }\n\n .velotax-content.conta {\n background-color: var(--white);\n }\n\n @media only screen and (max-device-width: 812px) {\n .page-title.conta {\n margin-bottom: 80px;\n button {\n left: 0;\n width: 100%;\n bottom: -64px;\n position: absolute;\n }\n }\n p {\n margin: 0 24px 24px;\n width: calc(100% - 48px);\n }\n .excluir-conta-container {\n width: 100%;\n padding: 0 8px;\n }\n .cancelar-plano-container {\n width: 100%;\n padding: 0 8px;\n margin: -12px auto 24px;\n }\n }\n`;\n","import { Button } from \"@mui/material\";\nimport { LoadingOutlined } from \"@ant-design/icons\";\nimport { useEffect, useState, useMemo } from \"react\";\nimport { AiFillWarning, AiOutlineUnlock } from \"react-icons/ai\";\nimport {\n Col,\n Descriptions,\n Form,\n Input,\n message,\n Modal,\n Row,\n Typography,\n} from \"antd\";\nimport { formatDate } from \"../../../utils\";\nimport HandleTag from \"../../../services/handleTag\";\nimport { UserPlans } from \"../../../constants/plans\";\nimport { useAuth } from \"../../../contexts/AuthContext\";\nimport { apiPayment } from \"../../../services/apiPayment\";\nimport { PaymentModal } from \"../../../components/PaymentModal\";\nimport apiVeloPro from \"../../../services/apiVeloPro\";\nimport { useNavigate } from \"react-router-dom\";\n\nconst KEY_PASS = \"DELETAR\";\n\nexport const MeuPlano: React.FC = () => {\n const navigate = useNavigate();\n const { user, getUserInfo, showUserPlanModal } = useAuth();\n const [data, setData] = useState();\n const [loading, setLoading] = useState(false);\n const [cancelModal, showCancelModal] = useState(false);\n const pendingCancel =\n data?.signature?.useDate > 0 &&\n (!data?.signature?.status || data?.signature?.status === \"canceled\");\n\n const [changeCardModal, showChangeCardModal] = useState(false);\n\n const [formKey] = Form.useForm();\n const [keyPass, setKeyPass] = useState(\"\");\n\n const planBasicInfo = user?.user?.userPlanInfoVelotax;\n const [isVelopro, setVelopro] = useState(false);\n const [hasCreditCardInfo, setCreditCardInfo] = useState(true);\n\n const checkPlanVelopro = () => {\n apiVeloPro\n .get('/calc/hasPlanActive/')\n .then(({ data }) => {\n if (data?.isPlanVelopro) setVelopro(true);\n else setVelopro(false);\n })\n .catch(() =>\n message.error(\"Ocorreu um erro, tente novamente mais tarde.\")\n );\n return;\n };\n\n useEffect(() => {\n checkPlanVelopro();\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [isVelopro]);\n\n const currentPlan = useMemo(() => {\n const plansTypes: Record = {\n VELOTAX_MAIN_BASIC: \"Velotax Basic Anual\",\n VELOTAX_MAIN_BASIC_MONTH: \"Velotax Basic Mensal\",\n VELOTAX_MAIN_PRO: \"Velotax Premium Anual\",\n VELOTAX_MAIN_PRO_MONTH: \"Velotax Premium Mensal\",\n VELOTAX_MAIN_CONCIERGE: \"Velotax Concierge Anual\",\n VELOTAX_MAIN_CONCIERGE_MONTH: \"Velotax Concierge Mensal\",\n WARREN_BASIC: \"WARREN Basic Anual\",\n WARREN_BASIC_MONTH: \"WARREN Basic Mensal\",\n WARREN_PRO: \"WARREN Premium Anual\",\n WARREN_PRO_MONTH: \"WARREN Premium Mensal\",\n WARREN_CONCIERGE: \"WARREN Concierge Anual\",\n WARREN_CONCIERGE_MONTH: \"WARREN Concierge Mensal\",\n };\n if (\n planBasicInfo.type === \"VELOTAX_MAIN_PRO\" ||\n planBasicInfo.type === \"VELOTAX_MAIN_BASIC\" ||\n planBasicInfo.type === \"VELOTAX_MAIN_PRO_MONTH\"\n ) {\n setCreditCardInfo(false);\n }\n return (\n (planBasicInfo?.active && plansTypes[planBasicInfo.type]) || \"Gratuito\"\n );\n }, [planBasicInfo]);\n\n const showChangeCardButton = useMemo(\n () =>\n currentPlan !== \"Gratuito\" &&\n !pendingCancel &&\n data?.signature?.paymentForm !== \"pix\" &&\n !data?.signature?.fromAccountant,\n [\n currentPlan,\n pendingCancel,\n data?.signature?.fromAccountant,\n data?.signature?.paymentForm,\n ]\n );\n\n const getData = () => {\n setLoading(true);\n apiPayment\n .get(\"/user-plan/get-plan-details\")\n .then((ret) => {\n setLoading(false);\n setData(ret.data);\n })\n .catch(() => {\n message.error(\n \"Houve um erro ao buscar os detalhes contate nosso suporte\"\n );\n });\n };\n\n const cancelPlan = () => {\n HandleTag(\"60\");\n if (keyPass.toLowerCase() !== KEY_PASS.toLowerCase()) {\n message.error(`Digite ${KEY_PASS} corretamente!`);\n return;\n }\n\n if (isVelopro) {\n apiVeloPro\n .put(`/calc/cancelRecuring/${user.user.userId}`)\n .then(() => {\n message.success(\"Plano cancelado\");\n showCancelModal(false);\n getUserInfo();\n getData();\n })\n .catch(() =>\n message.error(\n \"Ocorreu um erro ao cancelar a recorrência, tente novamente mais tarde.\"\n )\n );\n return;\n }\n\n apiPayment\n .post(\"/user-plan/cancel_signature\")\n .then((ret) => {\n message.success(\"Plano cancelado\");\n showCancelModal(false);\n getUserInfo();\n getData();\n })\n .catch(() => {\n message.error(\"Ocorreu um erro. Contate nosso suporte\");\n });\n };\n\n useEffect(() => {\n getData();\n }, []);\n\n const payForm: Record = {\n pix: \"Pix\",\n credit_card: \"Cartão de Crédito\",\n };\n\n const planStatus = useMemo(\n () =>\n currentPlan === \"Gratuito\"\n ? \"Ativo\"\n : planBasicInfo?.active\n ? \"Ativo\"\n : \"Inativo\",\n [planBasicInfo?.active, currentPlan]\n );\n\n const isActive = useMemo(() => planStatus === \"Ativo\", [planStatus]);\n\n const onDeleteCancel = () => {\n showCancelModal(false);\n formKey.resetFields();\n setKeyPass(\"\");\n };\n\n const contentStyle = {\n fontSize: 15,\n marginBottom: 0,\n };\n\n return (\n <>\n Dados do plano \n {loading ? (\n \n {\" \"}\n
\n ) : (\n <>\n \n
{\"Plano Atual\"}}\n >\n {currentPlan} \n \n
Status do Plano}\n >\n \n \n {planStatus}\n
\n {/* {pendingCancel && (\n \n \n \n )} */}\n \n \n {data?.signature?.next_billing_at &&\n planBasicInfo?.active &&\n !pendingCancel && (\n
{\"Próxima Cobrança\"}}\n >\n \n {formatDate(data?.signature?.next_billing_at)}\n \n \n )}\n {data?.signature?.start_at && planBasicInfo?.active && (\n
{\"Data da Assinatura\"}}\n >\n {data?.signature?.start_at && (\n \n {formatDate(data?.signature?.start_at)}\n \n )}\n \n )}\n {data?.signature?.dueDate && planBasicInfo?.active && (\n
{\"Vencimento da Assinatura\"}}\n >\n {data?.signature?.dueDate && (\n \n {formatDate(data?.signature?.dueDate)}\n \n )}\n \n )}\n {data?.signature?.paymentForm &&\n planBasicInfo?.active &&\n hasCreditCardInfo && (\n
Forma de pagamento}\n >\n {data?.signature?.paymentForm && (\n \n {<>{payForm[data?.signature?.paymentForm]}>}\n \n )}\n \n )}\n\n {data?.signature?.paymentForm !== \"pix\" &&\n planBasicInfo?.active &&\n hasCreditCardInfo &&\n data?.signature?.card?.lastFourDigits && (\n
{\"Cartão de cobrança\"}}\n >\n \n {\n <>\n **** **** **** {data?.signature?.card?.lastFourDigits}\n >\n }\n \n \n )}\n\n {/* {currentPlan === \"Gratuito\" && (\n
\n {\n showUserPlanModal(true);\n }}\n size=\"medium\"\n color=\"secondary\"\n variant=\"contained\"\n style={{\n display: \"flex\",\n justifyContent: \"right\",\n fontSize: 12,\n }}\n >\n CONTRATAR PLANO\n \n
\n )} */}\n\n {currentPlan === \"Gratuito\" && (\n
\n }\n onClick={() => {\n // showUserPlanModal(true);\n navigate('/planos')\n }}\n >\n CONTRATAR PLANO PREMIUM\n \n
\n )}\n\n {showChangeCardButton && (\n
\n \n {\n showChangeCardModal(true);\n }}\n >\n Alterar cartão de cobrança\n \n \n
\n )}\n
\n {((currentPlan !== \"Gratuito\" &&\n !pendingCancel &&\n data?.signature?.paymentForm !== \"pix\") ||\n isVelopro) && (\n \n \n {\n HandleTag(\"59\");\n showCancelModal(true);\n }}\n style={{\n opacity: 0.8,\n color: \"var(--ant-error-color)\",\n }}\n >\n Cancelar Recorrência\n \n \n
\n )}\n >\n )}\n\n {changeCardModal && (\n {\n showChangeCardModal(false);\n getData();\n }}\n changeCardAction\n onSuccessGeneral={() => {\n showChangeCardModal(false);\n getData();\n }}\n planCurrent={UserPlans.find(\n (plan) => plan?.type === user?.user?.userPlanInfoVelotax?.type\n )}\n hidePixButton\n blockListCard={[data?.signature?.card?.id]}\n show={changeCardModal}\n onCancel={() => showChangeCardModal(false)}\n paymentData={{\n type: planBasicInfo.type,\n isPlan: true,\n }}\n callDarf={(value: boolean) => null}\n customInstallments={12}\n />\n )}\n\n \n Deseja realmente cancelar o plano ?
\n \n \n \n \n \n \n Deseja realmente cancelar seu plano?\n \n \n Atenção: todos os seus dados de impostos,\n inclusive a memória dos cálculos e o histórico dos DARFs pagos,\n serão excluídos definitivamente da plataforma. Você poderá ser\n obrigado pela Receita Federal a apresentar tais informações, caso\n seja questionado.\n \n \n \n \n Digite {KEY_PASS} para prosseguir:\n \n \n \n \n \n \n \n \n
\n \n >\n );\n};\n","import ReactInputMask from \"react-input-mask\";\nimport { FaUserShield } from \"react-icons/fa\";\nimport { useNavigate } from \"react-router-dom\";\nimport { AiFillWarning } from \"react-icons/ai\";\nimport { Button, TextField } from \"@mui/material\";\nimport React, { useCallback, useEffect, useState } from \"react\";\nimport { CheckOutlined, LoadingOutlined } from \"@ant-design/icons\";\nimport { Col, Form, message, Row, Modal, Input, Typography, Select } from \"antd\";\nimport api from \"../../services/api\";\nimport apiBolsa from '../../services/apiBolsa'\nimport { Container } from \"./styles\";\nimport { MeuPlano } from \"./MeuPlano\";\nimport { cpfInputMask } from \"../../utils\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport {\n validationCpf,\n validationDateNotRequired,\n validationEmail,\n validationFieldRequired,\n validationFullName,\n validationPassword,\n validationPhone,\n} from \"../../utils/formValidations\";\nimport { BsFillGearFill, BsInfoCircleFill } from \"react-icons/bs\";\nimport { useB3Integration } from \"../../contexts/B3IntegrationContext\";\n\nconst KEY_PASS = \"DELETAR\";\n\nconst MaskCPF = React.forwardRef((props, ref) => (\n \n));\n\nconst MaskBirth = React.forwardRef((props, ref) => (\n \n));\n\nconst MaskPhone = React.forwardRef((props, ref) => (\n \n));\n\nexport const MinhaConta: React.FC = () => {\n const { user, setUser, signOut } = useAuth();\n const {\n handleIntegrate,\n loadingIntegration,\n } = useB3Integration();\n const [form] = Form.useForm();\n const [warningDeleteModal, showWarningDeleteModal] = useState(false);\n const navigate = useNavigate();\n const [formKey] = Form.useForm();\n const [loading, setLoading] = useState(loadingIntegration);\n const [showPassword, setShowPassword] = useState(false);\n const [keyPass, setKeyPass] = useState(\"\");\n const [fieldsError, setFieldsError] = useState<{ [key: string]: string[] }>(\n {}\n );\n const [modalParametrizacaoOpen, setModalParametrizacaoOpen] = useState(false);\n const [isMesa, setIsMesa] = useState(false);\n\n useEffect(() => {\n if (user?.user) {\n form.setFieldsValue({\n cpf: user.user.cpf,\n name: user.user.name,\n email: user.user.email,\n birthdate: user.user.birthdate ?? \"\",\n cellphone: user.user.cellphone ?? \"\",\n });\n }\n }, [user.user, form]);\n\n const getMesaOperacoes = useCallback(() => {\n apiBolsa.get('/warren/mesaOperacoes')\n .then((res) => {\n setIsMesa(res?.data?.isMesa || false)\n })\n .catch((err) => {\n message.error('Não foi possível consultar informações de mesa de operações')\n })\n }, [])\n\n useEffect(() => {\n if(modalParametrizacaoOpen){\n getMesaOperacoes()\n }\n }, [getMesaOperacoes, modalParametrizacaoOpen]);\n\n const setMesaOperacoes = () => {\n apiBolsa.post('/warren/mesaOperacoes', { isMesa })\n .then((res) => {\n if(res?.status === 202) {\n message.warning(res?.data?.message)\n } else {\n message.success('Suas informações foram salvas')\n getMesaOperacoes()\n handleIntegrate({onlyPrepareBolsa: true})\n }\n })\n .catch(() => {\n message.error('Não foi possível atualizar suas informações de mesa de operações')\n })\n }\n\n const onFinish = (data: any) => {\n if (loading) {\n return;\n }\n setLoading(true);\n api\n .put(\"/configuracoes/minhaConta\", {\n email: data.email,\n birthdate: data.birthdate,\n cellphone: data.cellphone,\n name: data.name,\n ...(showPassword\n ? {\n pass: data.pass,\n currentPass: data.currentPass,\n }\n : {}),\n })\n .then(() => {\n message.config({\n top: 50,\n });\n message.success(\"Dados atualizados com sucesso\");\n setLoading(false);\n setUser({\n ...user,\n user: {\n ...user.user,\n email: data.email,\n birthdate: data.birthdate,\n cellphone: data.cellphone,\n name: data.name,\n },\n });\n })\n .catch((err) => {\n message.error(\n err.response?.data?.message ||\n \"Erro ao atualizar dados. Tente novamente mais tarde!\"\n );\n setLoading(false);\n });\n };\n\n const removeAccount = () => {\n showWarningDeleteModal(true);\n };\n\n const onFieldsChange = () => {\n setFieldsError(\n form\n .getFieldsError()\n .reduce((acc, cur) => ({ ...acc, [cur.name?.[0]]: cur.errors }), {})\n );\n };\n\n const onCancel = () => {\n if (loading) {\n return;\n }\n navigate(\"/carteira\");\n };\n\n const handlePassword = () => {\n setShowPassword((show) => !show);\n };\n\n const sendDeleteReq = () => {\n if (keyPass.toLowerCase() !== KEY_PASS.toLowerCase()) {\n message.error(`Digite ${KEY_PASS} corretamente!`);\n return;\n }\n api\n .post(\"/user/remove-user\", {\n from: \"WARREN\",\n pass: keyPass,\n cpf: user.user.cpf,\n })\n .then(() => {\n message.success(\"Conta excluída com sucesso\");\n signOut();\n })\n .catch((err) => {\n message.error(err?.response?.data?.message);\n });\n };\n\n const onDeleteCancel = () => {\n showWarningDeleteModal(false);\n formKey.resetFields();\n setKeyPass(\"\");\n };\n\n return (\n \n \n \n Dados do usuário\n navigate(\"/gestaoacesso\")}\n startIcon={ }\n >\n Gestão de acessos\n \n \n \n
\n 0}\n />\n \n\n \n \n \n\n \n 0}\n InputProps={{ inputComponent: MaskBirth as any }}\n InputLabelProps={{ shrink: true }}\n />\n \n\n \n \n \n\n \n 0}\n InputProps={{ inputComponent: MaskPhone as any }}\n InputLabelProps={{ shrink: true }}\n />\n \n\n \n {showPassword ? \"Não alterar senha\" : \"ALTERAR SENHA\"}\n \n\n {showPassword && (\n \n \n \n 0}\n />\n \n \n \n \n 0}\n />\n \n \n
\n )}\n\n \n \n\n \n \n \n Cancelar\n \n \n \n }\n startIcon={loading && }\n >\n Salvar\n \n \n
\n \n \n \n \n \n EXCLUIR CONTA\n \n \n
\n {/* \n \n setModalParametrizacaoOpen(true)}>\n \n Parametrizar custos de corretagem\n \n \n
*/}\n setModalParametrizacaoOpen(false)}\n bodyStyle={{ paddingTop: '1rem' }}\n visible={modalParametrizacaoOpen}\n className=\"parametrizacao-modal\"\n >\n \n \n \n \n Funcionalidade exclusiva para investidores que realizam suas operações via Mesa de Operações (assessor).\n \n \n
\n \n \n \n Selecione a parametrização de custos de corretagem:\n \n\n \n
\n \n \n {setIsMesa(e === 'Mesa de operações' ? true : false)}}\n options={[\n { label: 'Não considerar', value: 'Não considerar' },\n { label: 'Mesa de operações', value: 'Mesa de operações' }\n ]}\n defaultValue={isMesa ? 'Mesa de operações' : 'Não considerar'} />\n \n
\n \n \n }\n startIcon={loading && }\n onClick={() => {\n setMesaOperacoes()\n setModalParametrizacaoOpen(false)\n }}\n >\n Salvar\n \n \n
\n \n \n \n \n \n \n \n \n Deseja realmente excluir sua conta?\n \n \n Lembre-se de que todos os seus dados também serão excluídos. Caso\n possua um plano ativo, você também perderá todas as\n funcionalidades.\n \n \n \n \n Digite {KEY_PASS} para prosseguir:\n \n \n \n \n \n \n \n \n
\n \n \n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 128px;\n h1 {\n font-weight: 400;\n font-size: 1.5rem;\n line-height: 2rem;\n color: var(--velotax-font-color-light);\n width: 700px;\n margin: 0 auto 16px;\n padding-bottom: 16px;\n position: relative;\n\n &::after {\n content: \"\";\n left: 0;\n bottom: 0;\n width: 100px;\n height: 2px;\n position: absolute;\n background-color: var(--ant-primary-color);\n }\n }\n p {\n font-size: 1rem;\n line-height: 1.5rem;\n color: var(--velotax-font-color-light);\n width: 700px;\n margin: 0 auto 32px;\n }\n @media only screen and (max-device-width: 812px) {\n min-height: 100%;\n padding: 24px 0 0;\n h1 {\n width: calc(100% - 48px);\n margin: 0 24px 16px;\n }\n p {\n width: calc(100% - 48px);\n padding: 0 0 24px;\n margin: 0 24px;\n }\n }\n`;\n\nexport const Content = styled.div`\n padding: 24px;\n margin: 0 auto;\n width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n\n h3 {\n color: var(--velotax-font-color);\n }\n\n span.link {\n cursor: pointer;\n color: var(--ant-primary-color);\n }\n\n .ant-typography:not(h3) {\n text-align: justify;\n }\n\n a.ant-typography,\n .ant-typography a {\n color: var(--ant-primary-color);\n }\n\n .ant-typography i strong {\n font-size: 13px;\n letter-spacing: 0.625px;\n word-spacing: 0.625px;\n }\n\n iframe {\n max-width: 100%;\n }\n\n .ant-list {\n margin-bottom: 1em;\n }\n\n .ant-collapse,\n .ant-collapse.ant-collapse-borderless {\n border-radius: 8px;\n background-color: transparent;\n border: 1px solid var(--velotax-font-color-ghost);\n .ant-collapse-item {\n .ant-collapse-header {\n padding: 16px;\n font-size: 1.125rem;\n color: var(--velotax-font-color-light);\n .ant-collapse-arrow {\n font-size: 0.875rem;\n }\n }\n .ant-collapse-content {\n border-bottom: 1px solid;\n color: var(--velotax-font-color-light);\n background-color: var(--velotax-background-color);\n border-color: var(--velotax-font-color-ghost);\n }\n }\n\n .ant-collapse-item:last-child {\n .ant-collapse-content {\n border-bottom: none;\n border-radius: 0 0 8px 8px;\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n width: 100%;\n border-radius: 0;\n\n .ant-collapse,\n .ant-collapse.ant-collapse-borderless {\n border-radius: 0;\n border: none;\n }\n }\n`;\n","import { Divider, Typography } from \"antd\";\n\nexport const CompaniesExemptList = {\n title: \"Lista de empresas isentas\",\n content: (\n
\n \n Lista de ações que, se vendidas com lucro até o dia 31/12/2023, \n são isentas de Impostos de Renda com base na lei 13.043, de 13 de novembro de 2014:\n \n \n
\n - NUTRIPLANT (NUTR3) \n - PETRORIO (PRIO3)* \n - GENERAL SHOPPING (GSHP3) \n - POMIFRUTAS (FRTA3) \n - CR2 EMPREENDIMENTOS IMOBILIÁRIOS (CRDE3) \n - SINQIA (SQIA3)* \n - BRASIL AGRO (AGRO3) \n - PRINER (PRNR3)\n \n
\n
\n * Válido para ações compradas antes do follow-on. A Sinqia e a Petrorio \n fizeram uma oferta de ações em setembro de 2019 e fevereiro de 2021, \n respectivamente. Caso você tenha comprado essas ações antes das datas das \n respectivas ofertas, você tem direito a isenção de imposto de renda sobre seus ganhos.\n \n
)\n};","import { useState } from \"react\";\nimport { List, Typography, Modal, Collapse } from \"antd\";\nimport { Container, Content } from \"./styles\";\nimport { CompaniesExemptList } from \"../../constants/regulation\";\n\ninterface RegulationProps {}\n\nexport const Regulation: React.FC = () => {\n const [showListModal, setShowListModal] = useState(false);\n\n const handleListModal = () => {\n setShowListModal((show) => !show);\n };\n return (\n \n Regras de IR utilizadas pelo Velotax \n \n Entenda quais são as normativas da Receita Federal que utilizamos para\n calcular automaticamente o seu imposto de renda\n
\n \n \n \n \n Entendendo o que é renda variável\n \n \n O mercado de renda variável compreende ativos cuja remuneração ou\n retorno de capital não pode ser dimensionado no momento da\n aplicação. São eles as ações, quotas ou quinhões de capital, o\n ouro, ativo financeiro, e os contratos negociados nas bolsas de\n valores, de mercadorias, de futuros e assemelhadas.\n \n \n (Instrução Normativa RFB nº 1.585, de 31 de agosto de 2015, arts.\n 56, 58 e 60 a 62)\n \n \n Qual imposto de renda incide sobre operações de renda variável\n \n \n O ganhos líquidos são constituídos pela diferença positiva entre o\n valor de venda do ativo e o seu custo de aquisição. Os ganhos\n líquidos auferidos em operações realizadas em bolsas de valores,\n de mercadorias, de futuros, e assemelhadas, inclusive day trade,\n serão tributados às seguintes alíquotas:\n \n \n \n a) 15%, nas operações realizadas nos mercados à vista, a termo,\n de opções e de futuros;\n \n b) 20%, no caso de operação day trade; \n \n c) 20%, no caso de operação com fundos imobiliários (FIIs).\n \n
\n \n (Lei nº 11.033, de 21 de dezembro de 2004, art. 2º, caput, incisos\n I e II, e §§ 1º e 2º; e Instrução Normativa RFB nº 1.585, de 31 de\n agosto de 2015, arts. 57, 63 e 65)\n \n \n Quando o imposto deve ser pago\n \n \n O imposto sobre a renda deve ser pago até o último dia útil do mês\n subsequente àquele em que os ganhos houverem sido apurados.\n \n O código a ser utilizado no Documento de Arrecadação das\n Receitas Federais (DARF) para pagamento desse tributo é 6015.\n \n \n (Instrução Normativa RFB nº 1.585, de 31 de agosto de 2015, art.\n 56, § 5º)\n \n \n Operações isentas de imposto de renda\n \n \n A isenção de imposto de renda aplica-se para transações realizadas\n em mercados comuns e que se encaixem nas situações abaixo:\n \n \n \n a) com ações, no mercado à vista de bolsas de valores ou mercado\n de balcão, se o total das alienações desse ativo, realizadas no\n mês, não exceder a R$ 20.000,00 (vinte mil reais);\n \n \n b) com ações de pequenas e médias empresas a que se refere o\n art. 16 da Lei nº 13.043, de 13 de novembro de 2014;\n \n \n Clique aqui para ver a lista de empresas isentas\n \n \n \n c) com ouro, ativo financeiro, se o total das alienações desse\n ativo, realizadas no mês, não exceder a R$ 20.000,00 (vinte mil\n reais);\n \n
\n \n A isenção acima não se aplica, entre outras, às operações de day\n trade, às negociações de cotas de fundos de investimentos\n imobiliários (FIIs) e fundos de investimento em índice de ações,\n aos resgates de cotas de fundos ou clubes de investimento em ações\n e à alienação de ações efetivada em operações de exercício de\n opções e no vencimento ou liquidação antecipada de contratos a\n termo.\n \n \n (Lei nº 11.033, de 21 de dezembro de 2004, art.3º, inciso I; Lei\n nº 13.043, de 13 de novembro de 2014, art. 16; e Instrução\n Normativa RFB nº 1.585 , de 31 de agosto de 2015, art. 59, incisos\n I e II e § 2º)\n \n \n O que é operação day trade\n \n \n Considera-se day trade a operação ou a conjugação de operações\n iniciadas e encerradas em um mesmo dia, com o mesmo ativo, {\" \"}\n em uma mesma instituição intermediadora (isto é, corretora) \n , em que a quantidade negociada tenha sido liquidada, total ou\n parcialmente. Na apuração do resultado da operação day trade são\n considerados, pela ordem, o primeiro negócio de compra com o\n primeiro de venda ou o primeiro negócio de venda com o primeiro de\n compra, sucessivamente.\n \n \n (Lei nº 9.959, de 27 de janeiro de 2000, art. 8º; Lei nº 12.350,\n de 20 de dezembro de 2010, art. 45; e Instrução Normativa RFB nº\n 1.585, de 31 de agosto de 2015, art. 65)\n \n \n Compensação de perdas passadas\n \n \n As perdas incorridas nas operações de renda variável nos mercados\n à vista, de opções, futuros, a termo e assemelhados, poderão ser\n compensadas com os ganhos líquidos auferidos, no próprio mês ou\n nos meses subsequentes, em outras operações realizadas em qualquer\n das modalidades operacionais previstas nesses mercados.\n \n \n Atenção: as perdas incorridas em operações\n comuns, somente poderão ser compensadas com ganhos líquidos\n auferidos em operações da mesma espécie (comuns), realizadas no\n mês ou meses subsequentes. Do mesmo modo, as perdas incorridas em\n operações day trade somente são compensáveis com os ganhos\n líquidos auferidos nessas operações, e perdas incorridas em FIIs\n são compensáveis com os ganhos líquidos auferidos em FIIs.\n \n \n (Regulamento do Imposto sobre a Renda – RIR/2018, art. 841, §§ 1º\n e 2º, aprovado pelo Decreto nº 9.580, de 22 de novembro de 2018; e\n Instrução Normativa RFB nº 1.585, de 31 de agosto de 2015, art.\n 64)\n \n \n Dedução de despesas operacionais\n \n \n As despesas efetivamente pagas destacadas na nota de corretagem ou\n no extrato da conta-corrente para a realização de operações de\n compra ou venda (corretagens, emolumentos etc.) podem ser\n consideradas na apuração do ganho líquido, sendo acrescidas ao\n preço de compra e deduzidas do preço de venda dos ativos ou\n contratos negociados.\n \n \n (Lei nº 8.383, de 30 de dezembro de 1991, art. 27; e Regulamento\n do Imposto sobre a Renda - RIR/2018, art. 841, caput e § 2º,\n aprovado pelo Decreto nº 9.580, de 22 de novembro de 2018; e\n Instrução Normativa RFB nº 1.585, de 31 de agosto de 2015, art.\n 56, § 3º)\n \n \n O que é o Portal da B3\n \n \n O Portal da B3 armazena os saldos e os históricos de transação de\n forma consolidada para os investidores da bolsa de valores.\n \n \n Caso você nunca tenha se cadastrado no Portal da B3, faça o seu{\" \"}\n \n cadastro aqui\n \n . Só precisa ter o seu CPF e a data de nascimento em mãos.\n \n {/*Remover os vídeos do passo a posso pois são baseados na calculadora antiga*/}\n {/* \n Ajustando manualmente o Velotax\n \n \n Apesar do sistema do Velotax ser o mais automático possível, algumas\n informações não estão disponíveis na B3. Portanto os investidores\n devem ter atenção para ajustar manualmente na{\" \"}\n Calculadora os pontos\n abaixo.\n */}\n {/* \n \n \n VIDEO \n \n \n VIDEO \n \n \n VIDEO \n \n \n VIDEO \n \n \n VIDEO \n \n \n VIDEO \n \n \n VIDEO \n \n \n VIDEO \n \n \n */}\n \n Ainda tem dúvidas? Acesse:\n \n \n Central de Ajuda\n \n , onde nosso time de suporte responde suas dúvidas\n \n\n \n \n Entendendo o que são criptoativos\n \n \n Os criptoativos são a representação digital de valor denominada em\n sua própria unidade de conta, cujo preço pode ser expresso em\n moeda soberana local ou estrangeira, transacionado eletronicamente\n com a utilização de criptografia e de tecnologias de registros\n distribuídos, que pode ser utilizado como forma de investimento,\n instrumento de transferência de valores ou acesso a serviços, e\n que não constitui moeda de curso legal.\n \n \n Alguns exemplos de criptoativos são Bitcoins, Altcoins (Ether,\n Ripple, Litecoin, entre outros) e Payment Tokens (Chiliz, Binance\n Coin, Chainlink, entre outros). Eles podem ser equiparados a\n ativos financeiros sujeitos a ganho de capital e devem ser\n declarados pelo seu valor de aquisição.\n \n \n (Instrução Normativa RFB nº 1.888, de 3 de maio de 2019)\n \n \n Qual imposto de renda incide sobre ganhos com criptoativos\n \n \n Os ganhos obtidos com a alienação de criptoativos cujo total\n alienado no mês seja superior a R$ 35.000,00 são tributados, a\n título de ganho de capital, segundo alíquotas progressivas\n estabelecidas em função do lucro:\n \n \n \n a) 15% sobre a parcela dos ganhos que não ultrapassar R$\n 5.000.000,00;\n \n \n b) 17,5% sobre a parcela dos ganhos que exceder R$ 5.000.000,00\n e não ultrapassar R$ 10.000.000,00;\n \n \n c) 20% sobre a parcela dos ganhos que exceder R$ 10.000.000,00 e\n não ultrapassar R$ 30.000.000,00; e\n \n \n d) 22,5% sobre a parcela dos ganhos que ultrapassar R$\n 30.000.000,00.\n \n
\n \n (Lei nº 5.172, de 25 de outubro de 1966 - Código Tributário\n Nacional - CTN, art. 118; Lei nº 8.981, de 20 de janeiro de 1995,\n art. 21; Instrução Normativa SRF nº 84, de 11 de outubro de 2001;\n Instrução Normativa SRF nº 599, de 28 de dezembro de 2005; e Ato\n Declaratório Interpretativo RFB nº 3, de 27 de abril de 2016)\n \n \n Quando o imposto deve ser pago\n \n \n O recolhimento do imposto sobre a renda deve ser feito até o\n último dia útil do mês seguinte ao da transação de venda.\n \n \n O código a ser utilizado no Documento de Arrecadação das Receitas\n Federais (DARF) para pagamento desse tributo é 4600.\n \n \n (Lei nº 5.172, de 25 de outubro de 1966 - Código Tributário\n Nacional - CTN, art. 118; Lei nº 8.981, de 20 de janeiro de 1995,\n art. 21; Instrução Normativa SRF nº 84, de 11 de outubro de 2001;\n Instrução Normativa SRF nº 599, de 28 de dezembro de 2005; e Ato\n Declaratório Interpretativo RFB nº 3, de 27 de abril de 2016)\n \n \n Operações isentas de imposto de renda\n \n \n A isenção relativa às alienações de até R$ 35.000,00 mensais deve\n observar o conjunto de criptoativos ou moedas virtuais alienados\n no Brasil ou no exterior, independente de seu nome (bitcoin,\n ethereum, litecoin, tether ...). Caso o total alienado no mês\n ultrapasse esse valor, o ganho de capital relativo a todas as\n alienações estará sujeito à tributação.\n \n \n (Lei nº 5.172, de 25 de outubro de 1966 - Código Tributário\n Nacional - CTN, art. 118; Lei nº 8.981, de 20 de janeiro de 1995,\n art. 21; Instrução Normativa SRF nº 84, de 11 de outubro de 2001;\n Instrução Normativa SRF nº 599, de 28 de dezembro de 2005; e Ato\n Declaratório Interpretativo RFB nº 3, de 27 de abril de 2016)\n \n \n Como converter de dólares (US$) para reais (R$)\n \n \n Para a conversão de valores em reais, o valor expresso em moeda\n estrangeira deve ser convertido em dólar dos Estados Unidos da\n América e convertido em moeda nacional pela cotação do dólar dos\n Estados Unidos da América fixada, para venda, pelo Banco Central\n do Brasil (BCB) para a data da operação ou saldo, extraída do\n boletim de fechamento PTAX divulgado pelo BCB.\n \n \n (Instrução Normativa RFB nº 1.888, de 3 de maio de 2019)\n \n \n Compensação de perdas passadas\n \n \n Como os ganhos em criptoativos estão sujeitos as regulações de\n ganho de capital, as perdas passadas em criptoativos não podem ser\n compensadas dos ganhos futuros. As operações em criptoativos são\n tratadas de maneira singular, não sendo possível compensar os\n prejuízos de algumas operações com lucros de outras.\n \n \n Mesmo para operações no mesmo mês, de forma conservadora, valem as\n disposições acima: informar cada operação de venda, sem compensar\n lucros com perdas.\n \n \n (Instrução Normativa SRF nº 84, de 11 de outubro de 2001, art. 2º,\n parágrafo único)\n \n \n Dedução de despesas operacionais\n \n \n As despesas com corretagem na venda de criptoativos podem ser\n abatidas do ganho de capital a ser tributado.\n \n \n Ainda tem dúvidas? Acesse:\n \n -{\" \"}\n \n Central de Ajuda\n \n , onde nosso time de suporte responde suas dúvidas\n \n\n \n \n Entendendo quais ativos no exterior estão sujeitos a tributação\n \n \n A alienação de bens ou direitos e liquidação ou resgate de\n aplicações financeiras, de propriedade de pessoa física,\n adquiridos em moeda estrangeira estarão sujeitos as regras de\n tributação sobre ganho de capital.\n \n \n Caso você tenha realizado operações que importem na alienação\n (venda) de bens ou direitos adquiridos em moeda estrangeira, como\n ações e outros ativos financeiros em bolsa de valores, de\n mercadorias, de futuros ou assemelhadas, você deverá efetuar o\n cálculo de ganho de capital tributável e o recolhimento do imposto\n devido segundo as alíquotas progressivas estabelecidas em função\n do lucro.\n \n \n (Lei nº 13.254, de 13 de janeiro de 2016; Lei nº 13.428, de 30 de\n março de 2017; Instrução Normativa SRF nº 118, de 27 de dezembro\n de 2000; Instrução Normativa SRF nº 84, de 11 de outubro de 2001,\n art. 3º; Instrução Normativa RFB nº 1.627, de 11 de março de 2016;\n e Instrução Normativa RFB nº 1.704, de 31 de março de 2017)\n \n \n Qual imposto de renda incide sobre ganhos na venda de ativos no\n exterior\n \n \n Os ganhos obtidos com a alienação de investimentos no exterior\n cujo total alienado no mês seja superior a R$ 35.000,00 são\n tributados, a título de ganho de capital, segundo alíquotas\n progressivas estabelecidas em função do lucro:\n \n \n \n a) 15% sobre a parcela dos ganhos que não ultrapassar R$\n 5.000.000,00;\n \n \n b) 17,5% sobre a parcela dos ganhos que exceder R$ 5.000.000,00\n e não ultrapassar R$ 10.000.000,00;\n \n \n c) 20% sobre a parcela dos ganhos que exceder R$ 10.000.000,00 e\n não ultrapassar R$ 30.000.000,00;\n \n \n d) 22,5% sobre a parcela dos ganhos que ultrapassar R$\n 30.000.000,00.\n \n
\n \n (Lei nº 8.981, de 20 de janeiro de 1995, art. 21; Lei nº 13.259,\n de 16 de março de 2016; e Ato Declaratório Interpretativo RFB nº\n 3, de 27 de abril de 2016)\n \n \n Quando o imposto deve ser pago\n \n \n O recolhimento do imposto sobre a renda deve ser feito até o\n último dia útil do mês seguinte ao da transação de venda.\n \n \n O código a ser utilizado no Documento de Arrecadação das Receitas\n Federais (DARF) para pagamento desse tributo é 8523.\n \n \n (Lei nº 5.172, de 25 de outubro de 1966 - Código Tributário\n Nacional - CTN, art. 118; Lei nº 8.981, de 20 de janeiro de 1995,\n art. 21; Instrução Normativa SRF nº 84, de 11 de outubro de 2001;\n Instrução Normativa SRF nº 599, de 28 de dezembro de 2005; e Ato\n Declaratório Interpretativo RFB nº 3, de 27 de abril de 2016)\n \n \n Operações isentas de imposto de renda\n \n \n A isenção relativa às alienações de até R$ 35.000,00 mensais deve\n observar o conjunto de instrumentos financeiros negociados em\n bolsa de valores no exterior, como ETFs (Exchange Traded Funds),\n REITs (Real Estate Investment Trust), ADRs (American Depositary\n Receipt) e Stocks (ações). Caso o total alienado no mês ultrapasse\n esse valor, o ganho de capital relativo a todas as alienações\n estará sujeito à tributação.\n \n \n (Lei nº 9.250, de 26 de dezembro de 1995, art. 22, I e II; Lei nº\n 10.406, de 10 de janeiro de 2002, art. 1.725; Instrução Normativa\n SRF nº 599, de 28 de dezembro de 2005, art. 1º; Solução de\n Consulta Cosit nº 320, de 20 de junho de 2017; e Solução de\n Consulta Cosit nº 264, de 24 de junho de 2019)\n \n \n Como funciona o cálculo do imposto de renda no exterior\n \n \n Na hipótese de bens ou direitos adquiridos e aplicações\n financeiras realizadas em moeda estrangeira com rendimentos\n auferidos originariamente em reais, o ganho de capital corresponde\n à diferença positiva, em reais, entre o valor de venda e o custo\n de aquisição da aplicação financeira.\n \n \n Neste caso, o valor de venda deve ser convertido para reais pela\n cotação do dólar fixada para compra, pelo Banco Central do Brasil,\n para a data do recebimento. Já o custo de aquisição deve ser\n convertido em reais pela cotação do dólar fixada para venda para a\n data do pagamento.\n \n \n Caso os rendimentos auferidos utilizados na aquisição sejam\n originariamente em moeda estrangeira, o ganho de capital\n corresponde à diferença positiva, em dólares dos Estados Unidos da\n América, entre o valor de venda e o custo de aquisição da\n aplicação, convertida em moeda nacional mediante a utilização da\n cotação do dólar fixada, para compra, pelo Banco Central do\n Brasil, para a data do recebimento\n \n \n (Medida Provisória nº 2.158-35, de 24 de agosto de 2001, art. 24;\n Lei nº 8.981, de 20 de janeiro de 1995, art. 21; Instrução\n Normativa SRF nº 208, de 27 de setembro de 2002; Instrução\n Normativa SRF nº 118, de 27 de dezembro de 2000; Solução de\n Consulta Interna Cosit nº 5, de 15 de fevereiro de 2013; e Solução\n de Consulta Cosit nº 33, de 26 de fevereiro de 2015)\n \n \n Como converter de dólares (US$) para reais (R$)\n \n \n Para a conversão de valores em reais, o valor expresso em moeda\n estrangeira deve ser convertido em dólar dos Estados Unidos da\n América e convertido em moeda nacional pela cotação do dólar dos\n Estados Unidos da América fixada, para compra ou venda, pelo Banco\n Central do Brasil (BCB) para a data da operação ou saldo, extraída\n do boletim de fechamento PTAX divulgado pelo BCB.\n \n \n (Instrução Normativa RFB nº 1.888, de 3 de maio de 2019)\n \n \n Compensação de perdas passadas\n \n \n Como os ganhos na venda de ativos no exterior estão sujeitos as\n regulações de ganho de capital, as perdas passadas em ativos no\n exterior não podem ser compensadas dos ganhos futuros. As\n operações com ativos no exterior são tratadas de maneira singular,\n não sendo possível compensar os prejuízos de algumas operações com\n lucros de outras.\n \n \n Mesmo para operações no mesmo mês, de forma conservadora, valem as\n disposições acima: informar cada operação de venda, sem compensar\n lucros com perdas.\n \n \n (Instrução Normativa SRF nº 84, de 11 de outubro de 2001, art. 2º,\n parágrafo único)\n \n \n Dedução de despesas operacionais\n \n \n As despesas com corretagem na venda de ativos no exterior podem\n ser abatidas do ganho de capital a ser tributado.\n \n \n Ainda tem dúvidas? Acesse:\n \n -{\" \"}\n \n Central de Ajuda\n \n , onde nosso time de suporte responde suas dúvidas\n \n\n {/* \n \n Imposto de Renda sobre Rendimentos\n \n \n Os rendimentos recebidos de outras pessoas físicas ou de fontes no\n exterior que ultrapassem o valor isento de R$1.903,98 por mês\n estarão sujeitos ao recolhimento mensal do imposto de renda.\n \n \n Nessa categoria, os principais rendimentos tributados incluem,\n entre outros:\n \n \n \n a) Autônomos, incluindo trabalhos sem vínculo empregatício;\n \n \n b) Aluguéis, incluindo a locação e sublocação de bens móveis e\n imóveis;\n \n \n c) Pensão, incluindo pensão alimentícia e alimentos\n provisionais;\n \n \n d) Exterior, incluindo quaisquer valores recebidos de fontes do\n exterior.\n \n
\n \n (Regulamento do Imposto sobre a Renda - RIR/2018, arts. 118 a 121\n e art. 123, aprovado pelo Decreto nº 9.580, de 22 de novembro de\n 2018; Instrução Normativa RFB nº 1.500, de 29 de outubro de 2014,\n arts. 24, § 4º, 53 e 54; Instrução Normativa RFB nº 1.531, de 19\n de dezembro de 2014)\n \n Autônomos \n \n Inclui os profissionais não-assalariados que recebem de outras\n pessoas físicas, e que possuem rendimentos advindos da prestação\n de serviços de representante comercial autônomo, intermediação na\n realização de negócios por conta de terceiros, prestação de\n serviços de transporte de passageiros, prestação de serviços de\n transporte de cargas, entre outros trabalhos sem vínculo\n empregatício.\n \n Aluguéis \n \n Inclui proprietários de imóveis que possuem rendimentos de\n aluguéis, considerando que:\n \n \n \n (a) Apenas o valor líquido do aluguel é tributado. As despesas\n de IPTU, condomínio, cobrança/recebimento e aluguel no caso de\n imóvel sublocado podem ser descontadas do rendimento;\n \n \n (b) Se você tiver mais de um imóvel e receber mais de um aluguel\n por mês, será necessário considerar a quantia total dos aluguéis\n recebidos.\n \n
\n \n A data do pagamento, para efeito de incidência do imposto, será\n considerada aquela em que o locatário pagar o aluguel ao\n proprietário do bem ou à administradora.\n \n Pensão Alimentícia \n \n Inclui os alimentandos que recebem pensão alimentícia. A pensão\n alimentícia pode ser recebida não apenas pelos filhos, mas por\n qualquer pessoa que tenha recebido esse benefício por decisão\n judicial, acordo homologado judicialmente ou por escritura\n pública. O contribuinte é o beneficiário da pensão, ainda que\n tenha sido paga a seu representante legal.\n \n Exterior \n \n Inclui pessoas físicas com rendimentos ou quaisquer outros valores\n recebidos de fontes do exterior. Neste caso, é importante observar\n a possível existência de{\" \"}\n \n acordos, convenções e tratados internacionais\n {\" \"}\n firmados entre o Brasil e o país de origem dos rendimentos para\n evitar a bitributação.\n \n \n Qual imposto de renda incide\n \n \n O imposto sobre rendimentos é calculado sobre o total recebido no\n mês, mediante a aplicação da seguinte tabela progressiva mensal:\n \n \n \n a) Até R$ 1.903,98 não há impostos a pagar;\n \n \n b) De R$ 1.903,99 até R$ 2.826,65, a alíquota é de 7,5% sobre os\n rendimentos, com parcela a deduzir de R$ 142,80;\n \n \n c) De R$ 2.826,66 até R$ 3.751,05, a alíquota é de 15% sobre os\n rendimentos, com parcela a deduzir de R$ 354,80;\n \n \n d) De R$ 3.751,06 até R$ 4.664,68, a alíquota é de 22,5% sobre\n os rendimentos, com parcela a deduzir de R$ 636,13;\n \n \n e) Acima de R$ 4.664,68, a alíquota é de 27,5% sobre os\n rendimentos, com parcela a deduzir de R$ 869,36.\n \n
\n \n (Lei nº 11.482, de 31 de maio de 2007, arts. 1º e 3º; Lei nº\n 12.024, de 27 de agosto de 2009, art. 3º; Regulamento do Imposto\n sobre a Renda - RIR/2018, art. 121, aprovado pelo Decreto nº\n 9.580, de 22 de novembro de 2018; Instrução Normativa RFB nº\n 1.500, de 29 de outubro de 2014, arts. 53 a 57 e 65; Solução de\n Consulta Interna Cosit nº 3, de 8 de fevereiro de 2012; e Solução\n de Consulta Interna Cosit nº 6, de 25 de maio de 2015)\n \n \n Quando o imposto deve ser pago\n \n \n O recolhimento do imposto de renda deve ser feito até o último dia\n útil do mês seguinte ao recebimento dos rendimentos.\n \n \n O código a ser utilizado no Documento de Arrecadação das Receitas\n Federais (DARF) para pagamento desse tributo é 0190.\n \n \n (Lei nº 8.383, de 30 de dezembro de 1991, art. 7º; e Instrução\n Normativa RFB nº 1.500, de 29 de outubro de 2014, arts. 53 a 57 e\n 67)\n \n \n VIDEO \n \n \n Deduções no imposto de renda\n \n \n São admitidas as seguintes deduções na base de cálculo do imposto\n sobre rendimentos:\n \n \n \n a) Contribuição previdenciária oficial, o INSS;\n \n \n b) Dependentes, sendo R$ 189,59 por dependente a partir do ano\n 2021;\n \n \n c) Pensão alimentícia, quando paga em cumprimento de decisão\n judicial, inclusive os alimentos provisionais, de acordo\n homologado judicialmente ou de escritura pública; e\n \n \n d) Despesas relacionadas à atividade do profissional autônomo\n \n
\n \n No caso de autônomos, somente as seguintes categorias de despesas\n podem ser deduzidas da base tributária:\n \n \n \n i) Despesas trabalhistas;\n \n \n ii) Serviços de terceiros;\n \n \n iii) Despesas com custeio e funcionamento;\n \n \n iv) Propaganda, assinatura de jornais e revistas;\n \n v) Roupas especiais; \n \n vi) Contribuições a sindicatos, associações ou conselhos;\n \n \n vii) Despesas com congressos e seminários;\n \n \n viii) Honorários profissionais;\n \n \n ix) Imóvel residencial ou profissional;\n \n \n x) Manutenção e conservação;\n \n \n xi) Correios e transporte;\n \n \n xii) Instalações, máquinas e equipamentos;\n \n \n xii) Despesas Comuns como rateio sob forma de Condomínio;\n \n \n xiv) Outras despesas necessárias à atividade específica e\n individual de cada contribuinte não contempladas anteriormente.\n \n
\n \n (Lei nº 7.290, de 19 de dezembro de 1984; Lei nº 7.713, de 22 de\n dezembro de 1988, art. 9º; Regulamento do Imposto sobre a Renda -\n RIR/2018, arts. 39 e 158, § 1º, aprovado pelo Decreto nº 9.580, de\n 22 de novembro de 2018; e Parecer Normativo CST nº 122, de 8 de\n junho de 1974)\n \n \n Utilizando o Velotax para IR em Rendimentos\n \n \n A{\" \"}\n navigate(`/${currentBroker.path}/autonomos`)}\n >\n Calculadora\n {\" \"}\n do Velotax para cálculo e pagamento de IR sobre rendimentos de\n pessoas físicas ou fontes no exterior permite a inclusão manual de\n operações, e realiza os cálculos automaticamente seguindo as\n regras da Receita Federal.\n \n \n Após inserir os seus rendimentos do mês, o Velotax calculará os\n seus impostos devidos de acordo com as regras acima e emitirá o\n seu DARF para pagamento.\n \n \n Caso você esteja pagando em atraso, o Velotax também incluirá\n automaticamente os juros e multa devidos no seu DARF.\n \n \n Ainda tem dúvidas? Acesse:\n \n \n -{\" \"}\n \n Fórum\n {\" \"}\n , onde nossa comunidade de usuários responde suas dúvidas\n \n */}\n \n \n\n \n {CompaniesExemptList?.content}\n \n \n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 128px;\n &.view-edit {\n padding: 0;\n }\n &.view {\n padding: 0;\n }\n\n h1 {\n width: 700px;\n margin: 0 auto 1.5rem;\n font-weight: 400;\n font-size: 2rem;\n position: relative;\n line-height: 2.5rem;\n padding-bottom: 1rem;\n color: var(--velotax-font-color-light);\n\n :after {\n content: \"\";\n position: absolute;\n left: 0;\n bottom: 0;\n height: 2px;\n width: 100px;\n background-color: var(--ant-primary-color);\n }\n }\n\n .juridic-messages-container {\n opacity: 0.8;\n margin: 0 auto;\n padding: 24px 0 0;\n max-width: 700px;\n\n .message {\n display: flex;\n flex-direction: row;\n align-items: center;\n margin-bottom: 1rem;\n }\n\n p {\n font-size: 0.85rem;\n font-weight: 400;\n text-align: justify;\n line-height: 1.5rem;\n color: var(--velotax-font-color-light);\n margin-bottom: 0;\n\n i {\n font-weight: bold;\n color: var(--velotax-font-color-light);\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 0 0 64px;\n\n h1 {\n max-width: calc(100% - 48px);\n margin: 0 24px 1.5rem;\n padding: 24px 0 1rem;\n }\n\n .juridic-messages-container {\n margin: 0;\n width: 100%;\n p {\n margin-left: 0;\n }\n }\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n &.second.view,\n &.second.view-edit {\n padding: 0 32px 32px;\n }\n &.second {\n padding: 32px;\n }\n margin: 0 auto;\n max-width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n h3 {\n display: flex;\n align-items: center;\n font-size: 20px;\n column-gap: 16px;\n color: var(--velotax-font-color-light);\n margin-bottom: 0;\n user-select: none;\n }\n h3.date-title {\n font-size: 24px;\n font-weight: 700;\n }\n .month-chevron {\n width: 32px;\n height: 32px;\n cursor: pointer;\n &.disabled {\n opacity: 0.15;\n cursor: default;\n }\n }\n .desc-label {\n display: flex;\n align-items: center;\n svg {\n fill: var(--ant-primary-color);\n }\n }\n .desc-content {\n display: flex;\n align-items: flex-start;\n justify-content: center;\n column-gap: 8px;\n svg {\n fill: var(--ant-primary-color);\n }\n span {\n align-self: center;\n }\n button {\n min-width: 32px;\n }\n }\n .ml-40 {\n margin-left: 40px;\n }\n .add {\n float: right;\n min-width: 128px;\n margin-bottom: 16px;\n }\n .ant-descriptions-view {\n border-radius: 4px !important;\n }\n .ant-descriptions-item-label {\n width: 50%;\n color: var(--velotax-font-color-light) !important;\n }\n .ant-descriptions-item-content {\n text-align: center;\n background-color: var(--velotax-background-color-ghost) !important;\n span {\n color: var(--velotax-font-color-light) !important;\n }\n }\n .ant-list.ant-list-split {\n clear: both;\n padding: 0 16px;\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-ghost);\n border-radius: 4px;\n .ant-list-item-action > li {\n padding: 0;\n }\n }\n .ant-list-item-action > li,\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color-light);\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n .list-description {\n display: flex;\n flex-direction: column;\n }\n .min-darf-price {\n margin-top: 24px;\n display: flex;\n align-items: center;\n column-gap: 8px;\n }\n .text-center {\n display: block;\n text-align: center;\n }\n .ant-collapse-header {\n padding: 12px 0 !important;\n }\n .ant-collapse-content-box {\n padding: 16px 0 !important;\n }\n .ant-collapse\n > .ant-collapse-item\n > .ant-collapse-header\n .ant-collapse-arrow\n svg {\n transform: rotate(-90deg);\n }\n .ant-collapse-item-disabled {\n & > .ant-collapse-header {\n cursor: default;\n .ant-collapse-arrow {\n opacity: 0;\n }\n }\n }\n .ant-collapse-header-text {\n width: 100%;\n }\n .total-tax-header {\n width: calc(100% - 48px);\n display: flex;\n align-items: center;\n justify-content: space-between;\n &.original {\n h3:last-of-type {\n width: 40%;\n }\n }\n h3 {\n margin: 0 0 8px;\n }\n }\n\n .no-plan-container {\n position: absolute;\n top: 0;\n left: 50%;\n width: 50%;\n z-index: 3;\n height: 100%;\n padding: 8px;\n display: flex;\n font-size: 16px;\n font-weight: 500;\n text-align: center;\n align-items: center;\n flex-direction: column;\n justify-content: center;\n background-color: #fff4;\n border-radius: 0 8px 8px 0;\n backdrop-filter: blur(4px);\n border: 1px solid var(--velotax-background-color-ghost);\n border-left: none;\n .ant-btn {\n margin-top: 8px;\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n width: 100%;\n padding: 24px;\n border-radius: 0;\n &.first {\n padding: 24px;\n }\n &.second {\n padding: 24px !important;\n border-top: 1px solid #efefef;\n }\n h3 {\n font-size: 18px;\n }\n .no-plan-container {\n font-size: 14px;\n line-height: 18px;\n }\n .anticon.anticon-right.ant-collapse-arrow {\n top: 24px;\n }\n .desc-content {\n column-gap: 2px;\n flex-direction: row;\n &.is-editting {\n row-gap: 8px;\n flex-direction: column;\n button {\n width: 100%;\n }\n }\n &:not(.is-editting) {\n span {\n line-height: 32px;\n }\n button {\n margin-top: -2px;\n }\n }\n &.ml-40 {\n margin-left: 0;\n }\n }\n }\n`;\n","import { useEffect, useState } from \"react\";\nimport { Col, Form, Modal, Row } from \"antd\";\nimport { FooterModal } from \"../FooterModal\";\nimport api from \"../../services/apiExterior\";\n\ninterface FormModalProps {\n add: any;\n edit: any;\n title: string;\n onCancel: any;\n itemToEdit: any;\n visibility: boolean;\n hideFooter?: boolean;\n rows: any[];\n disableButtons?: boolean;\n preFormChildren?: React.ReactNode;\n loading?: boolean;\n cancelButtonText?: string;\n saveButtonText?: string;\n}\n\nexport const FormModal: React.FC = ({\n add,\n edit,\n rows,\n title,\n onCancel,\n visibility,\n itemToEdit,\n children,\n disableButtons,\n preFormChildren,\n hideFooter,\n loading,\n cancelButtonText,\n saveButtonText\n}) => {\n const [form] = Form.useForm();\n const [, setUpdate] = useState(false);\n\n useEffect(() => {\n if (!visibility) {\n form.resetFields();\n } else if (itemToEdit) {\n setUpdate((update) => !update);\n form.setFieldsValue(itemToEdit);\n }\n }, [visibility, form, itemToEdit]);\n\n const onValuesChange = async (changedValues: any, values: any) => {\n const inputs = rows\n .flat()\n .map((input) =>\n typeof input === \"function\" ? input({ data: values }) : input\n );\n if (\n inputs\n .filter((item) => item.mustUpdate)\n .map((item) => item.name)\n .includes(Object.keys(changedValues)[0])\n ) {\n setUpdate((update) => !update);\n }\n let data;\n if (changedValues?.date?.replace(\"_\", \"\").length === 10) {\n const response = await api.get(\"/dividendos/get-cambio\", {\n params: { date: changedValues.date }\n });\n values.taxaCambio = response?.data;\n }\n data = {\n ...values,\n ...inputs\n .filter((input) => input.onChange)\n .reduce(\n (acc, cur) => ({\n ...acc,\n [cur.name]: cur.onChange?.(changedValues, { ...values, ...acc }),\n }),\n {}\n ),\n };\n const dataEntries = Object.entries(data);\n form.setFields(dataEntries.map(([name, value]) => ({ name, value })));\n form.resetFields(\n dataEntries\n .filter(([, value]) => value === undefined || value === null)\n .map(([name]) => name)\n );\n };\n\n const onFinish = (data: any) => {\n if (itemToEdit) {\n edit(data);\n } else {\n add(data);\n }\n };\n\n return (\n \n {preFormChildren}\n \n \n );\n};\n","import { useState } from \"react\";\nimport { useNavigate } from \"react-router-dom\";\nimport { Divider, message, Modal, Typography } from \"antd\";\nimport apis from \"../../../../services/apis\";\nimport Button from \"../../../../components/Button\";\nimport { useAuth } from \"../../../../contexts/AuthContext\";\nimport { useBroker } from \"../../../../contexts/BrokerContext\";\nimport { errorMessage, monthsExtended } from \"../../../../utils\";\n\ninterface IntegrationModalProps {\n visible: boolean;\n onCancel: () => void;\n}\n\nexport const IntegrationModal: React.FC = ({\n visible,\n onCancel,\n}) => {\n const { user } = useAuth();\n const navigate = useNavigate();\n const [loading, setLoading] = useState(false);\n const { currentBroker, integration, handleIntegrationInit, currentPage } =\n useBroker();\n\n const integrationDate = new Date(integration.lastUpdate);\n\n const handleIntegrate = () => {\n if (integration.key) {\n setLoading(true);\n (currentPage?.api || apis)\n .post(`${currentBroker.apiUrl}/integrate`, {\n key: integration.key,\n email: user.user.email,\n })\n .then(() => {\n setLoading(false);\n handleIntegrationInit();\n })\n .catch((err) => {\n setLoading(false);\n if (err?.response?.data?.message?.includes(\"Aguarde\")) {\n message.error(err?.response?.data?.message);\n } else {\n message.error(errorMessage);\n }\n });\n } else {\n navigate(`/${currentBroker.path}/integration`);\n }\n };\n\n return (\n \n \n A sua última integração foi feita no mês de{\" \"}\n {monthsExtended[integrationDate.getMonth()]} de{\" \"}\n {integrationDate.getFullYear()}. Para gerenciar operações e gerar DARFs\n dos meses seguintes ao da última integração, você precisa integrar\n novamente.\n \n \n \n Integrar\n \n \n );\n};\n","import { Typography } from \"antd\";\n\nexport const DatePickerExtraFooter: React.FC = () => {\n return (\n <>\n Situação: \n \n Imposto apurado \n
\n \n Imposto devido \n
\n >\n );\n};\n","import React, { useState } from \"react\";\nimport { LoadingOutlined } from \"@ant-design/icons\";\nimport { Col, Divider, message, Modal, Row } from \"antd\";\nimport { Button } from \"@mui/material\";\n\ninterface IDeleteConfirmationModalProps {\n title: string;\n visibility: boolean;\n body: React.ReactNode;\n onCancel: () => void;\n onOk?: () => Promise;\n updateData?: () => void;\n setVisibility: (param: any) => void;\n buttonLabel?: string;\n showToast?: boolean;\n successMessage?: string;\n}\n\nconst DeleteConfirmationModal: React.FC = ({\n title,\n visibility,\n onCancel,\n onOk,\n body,\n updateData,\n setVisibility,\n buttonLabel,\n showToast = true,\n successMessage\n}) => {\n const [loading, setLoading] = useState(false);\n const handleOnOk = () => {\n setLoading(true);\n const finish = () => {\n setLoading(false);\n updateData?.();\n setVisibility(false);\n showToast && message.success(successMessage || \"Deletado com sucesso\");\n };\n if (onOk) {\n onOk()\n .then(() => finish())\n .catch((err) => {\n showToast && message.error(\"Algo deu errado\");\n setLoading(false);\n });\n } else {\n finish();\n }\n };\n return (\n \n {body}
\n \n \n \n \n Cancelar\n \n \n \n \n {loading && }\n {buttonLabel || \"Deletar\"}\n \n \n
\n \n );\n};\n\nexport default DeleteConfirmationModal;\n","import clsx from \"clsx\";\nimport { months } from \"../../../../utils\";\nimport {\n YearResumeStatusCSSClassEnum,\n YearResumeStatusFromBackEnum,\n} from \"../../../../constants/darf\";\n\nexport interface YearResume {\n year: number;\n months: {\n month: number;\n payed: boolean;\n status: keyof typeof YearResumeStatusFromBackEnum;\n }[];\n}\n\ninterface DatePickerMonthCellProps {\n event: moment.Moment;\n yearResume: YearResume[];\n}\n\nexport const DatePickerMonthCell: React.FC = ({\n event,\n yearResume,\n}) => {\n const date = new Date();\n const year = event.year();\n const month = event.month();\n const item = yearResume\n .find((list) => list.year === year)\n ?.months.find((item) => item.month - 1 === month);\n const disabled = year === date.getFullYear() && month >= date.getMonth();\n return (\n \n {months[month]}\n
\n );\n};\n","import clsx from \"clsx\";\nimport moment from \"moment\";\nimport { Button } from \"@mui/material\";\nimport { FaRegEdit } from \"react-icons/fa\";\nimport { BsCalendar3, BsDownload } from \"react-icons/bs\";\nimport { useLocation, useNavigate } from \"react-router-dom\";\nimport { BiChevronLeft, BiChevronRight } from \"react-icons/bi\";\nimport { AiOutlineDelete, AiOutlineLock } from \"react-icons/ai\";\nimport { useCallback, useEffect, useMemo, useState, ReactNode } from \"react\";\nimport { LoadingOutlined, PlusOutlined } from \"@ant-design/icons\";\nimport {\n Col,\n Collapse,\n DatePicker,\n Descriptions,\n List,\n message,\n Modal,\n Row,\n Skeleton,\n Space,\n Spin,\n Tooltip,\n Typography,\n} from \"antd\";\nimport { Container, Content } from \"./styles\";\nimport { Page } from \"../../constants/brokers\";\nimport AntButton from \"../../components/Button\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport { FormModal } from \"../../components/FormModal\";\nimport { apiPayment } from \"../../services/apiPayment\";\nimport { useBroker } from \"../../contexts/BrokerContext\";\nimport apis, { NOT_AUTHORIZED } from \"../../services/apis\";\nimport { PaymentModal } from \"../../components/PaymentModal\";\nimport { IntegrationModal } from \"./Components/IntegrationModal\";\nimport { DatePickerExtraFooter } from \"./Components/DatePickerExtraFooter\";\nimport DeleteConfirmationModal from \"../../components/DeleteConfirmationModal\";\nimport {\n DatePickerMonthCell,\n YearResume,\n} from \"./Components/DatePickerMonthCell\";\nimport {\n antDatePickerLocale,\n download,\n errorMessage,\n formatCurrency,\n isMobile,\n monthsExtended,\n} from \"../../utils\";\nimport {\n DarfResultDescriptions,\n DarfFormItemRows,\n defaultDarf,\n DarfModal,\n IDarf,\n DarfImpostosDevidosDescriptions,\n minDarfPrice,\n maxDarfPrice,\n YearResumeStatusFromBackEnum,\n} from \"../../constants/darf\";\n\ninterface DarfProps {\n item: Page;\n view?: boolean;\n viewEdit?: boolean;\n closeModal?: () => void;\n darf?: {\n id: string;\n fullPath?: string;\n };\n date?: {\n year?: number;\n month?: number;\n };\n}\n\nexport const Darf: React.FC = ({\n item,\n view,\n date,\n darf,\n viewEdit,\n closeModal,\n}) => {\n const {\n user,\n hasPlan: hasPremiunPlan,\n showUserPlanModal,\n userPlanModal,\n } = useAuth();\n const hasPlan = hasPremiunPlan;\n\n const navigate = useNavigate();\n const { state } = useLocation();\n const { currentBroker, currentPage } = useBroker();\n const today = new Date();\n const queryYear = (state as any)?.year as number;\n const queryMonth = (state as any)?.month as number;\n const currentMonth = today.getMonth();\n const currentYear = today.getFullYear();\n const maxPixPayment = 10000;\n const maxCreditCardPayment = 10000;\n\n const initialMonth = queryMonth\n ? queryMonth - 1\n : today.getMonth() - 1 >= 0\n ? today.getMonth() - 1\n : 11;\n const initialYear = queryYear\n ? queryYear\n : today.getMonth() - 1 >= 0\n ? today.getFullYear()\n : today.getFullYear() - 1;\n const defaultValue = moment({ month: initialMonth, year: initialYear });\n const currentDate = moment({\n month: currentMonth,\n year: currentYear,\n }).subtract(1, \"M\");\n window.history.replaceState({}, document.title);\n\n const year = date?.year ?? initialYear;\n const month = date?.month ?? initialMonth;\n\n const [asset, setAsset] = useState();\n const [loading, setLoading] = useState(true);\n const [emitting, setEmitting] = useState(false);\n const [darfModal, setDarfModal] = useState();\n const [helpModal, setHelpModal] = useState();\n const [paymentModal, setPaymentModal] = useState(false);\n const [paymentData, setPaymentData] = useState();\n const [, setYear] = useState(initialYear);\n const [, setMonth] = useState(initialMonth);\n const [data, setData] = useState(defaultDarf);\n const [showAssetModal, setShowAssetModal] = useState(false);\n const [showDeleteModal, setShowDeleteModal] = useState(false);\n const [yearResume, setYearResume] = useState([]);\n const [gettingYearResume, setGettingYearResume] = useState(false);\n const [datePickerYear, setDatePickerYear] = useState(initialYear);\n const [notAuthorized, setNotAuthorized] = useState(false);\n const [showNotAuthorizedModal, setShowNotAuthorizedModal] = useState(false);\n const [loadingDarfButton, setLoadingDarfButton] = useState(false);\n\n const transactionFeature = item.features[0];\n const taxFeature = item.features[1];\n const yearResumeFeature = item.features[2];\n\n // const userPlanInfoStatus = useMemo(\n // () => user.user.userPlanInfoVelotax,\n // [user.user.userPlanInfoVelotax]\n // );\n\n const { emitted } = useMemo(() => {\n if (gettingYearResume) {\n }\n const monthResume = yearResume\n .find((resume) => resume.year === year)\n ?.months.find((m) => m.month - 1 === month);\n return {\n emitted:\n monthResume?.status === YearResumeStatusFromBackEnum.PAYED ||\n monthResume?.status === YearResumeStatusFromBackEnum.NOT_PAYED,\n payed: !!monthResume?.payed,\n status: monthResume?.status,\n };\n }, [yearResume, year, month, gettingYearResume]);\n\n const getDarf = useCallback(\n (id: string) => {\n view && setLoading(true);\n view &&\n (currentPage?.api || apis)\n .get(`${taxFeature.apiUrl}/${id}`)\n .then((response) => {\n setYear(response.data?.year);\n setMonth(response.data?.month - 1);\n setData({\n ...response.data,\n emitted: true,\n });\n })\n .catch(() => message.error(errorMessage))\n .finally(() => setLoading(false));\n },\n [view, taxFeature.apiUrl, currentPage]\n );\n\n const getYearResume = useCallback(\n (year: number, force?: boolean, url?: string) => {\n setGettingYearResume(true);\n !view &&\n year <= new Date().getFullYear() &&\n year >= currentBroker.initialYear &&\n (force ||\n (!yearResumeFeature.disabled &&\n !yearResume.find((resume) => resume.year === year))) &&\n (currentPage?.api || apis)\n .get(yearResumeFeature.apiUrl, { params: { year } })\n .then((response) => {\n const index = yearResume.findIndex(\n (resume) => resume.year === year\n );\n if (index >= 0) {\n setYearResume((yearResume) => {\n yearResume.splice(index, 1, response.data);\n return yearResume;\n });\n } else {\n setYearResume((yearResume) => [...yearResume, response.data]);\n }\n\n if (url && !isMobile()) {\n download(url);\n }\n })\n .catch((err) => console.log(err))\n .finally(() => setGettingYearResume(false));\n },\n [\n yearResumeFeature,\n yearResume,\n view,\n currentBroker.initialYear,\n currentPage,\n ]\n );\n\n const getTaxes = useCallback(() => {\n (!hasPlan || !view) && setLoading(true);\n (!hasPlan || !view) &&\n (currentPage?.api || apis)\n .get(transactionFeature.apiUrl, { params: { month: month + 1, year } })\n .then((response) => {\n setData(response.data);\n setNotAuthorized(false);\n })\n .catch((err) => {\n setData(defaultDarf);\n if (err?.response?.data === NOT_AUTHORIZED) {\n setNotAuthorized(true);\n setShowNotAuthorizedModal(true);\n } else {\n setNotAuthorized(false);\n message.error(errorMessage);\n }\n })\n .finally(() => setLoading(false));\n }, [view, currentPage?.api, transactionFeature.apiUrl, month, year, hasPlan]);\n\n const handleDarf = (justSave?: boolean) => {\n setEmitting(true);\n onCloseDarfModal();\n (currentPage?.api || apis)\n .post(taxFeature.apiUrl, {\n ...data,\n year,\n month: month + 1,\n name: user.user.name,\n memoriaCalculo: [],\n transactions: data.memoriaCalculo,\n paymentInfo: {\n clientInfo: {\n name: user?.user?.name,\n email: user?.user?.email,\n document: user?.user?.cpf,\n },\n },\n insertPix: true,\n justSave,\n payed: false,\n })\n .then((res) => {\n setData((data) => ({ ...data, emitted: true }));\n getYearResume(year, true, justSave ? \"\" : res.data.darfUrl);\n setEmitting(false);\n closeModal?.();\n if (!justSave) {\n message.success(\"O DARF foi enviado para o seu e-mail\");\n }\n })\n .catch((err) => {\n message.error(err?.response?.data?.message || errorMessage);\n setEmitting(false);\n });\n };\n\n const handleRegularize = (regular: boolean) => {\n setEmitting(true);\n (currentPage?.api || apis)\n .post(\"transaction/regularize\", {\n year,\n month: month + 1,\n regular,\n impostoDevido: data.impostoDevido,\n totalImpostoDevido: data.totalImpostoDevido,\n })\n .then((res) => {\n getYearResume(year, true, res.data.darfUrl);\n getTaxes();\n message.success(\"Dados salvos\");\n handleNavigateToHistoric();\n })\n .catch(() => message.error(errorMessage))\n .finally(() => setEmitting(false));\n };\n\n const onChangeMonth = (value: moment.Moment | null, dateString: string) => {\n const [month, year] = dateString.split(\"-\");\n setMonth(parseInt(month) - 1);\n setYear(parseInt(year));\n };\n\n const unsetAsset = () => {\n setAsset(undefined);\n };\n\n const handleCloseDeleteModal = () => {\n setShowDeleteModal(false);\n unsetAsset();\n };\n\n const handleCloseAssetModal = () => {\n setShowAssetModal(false);\n unsetAsset();\n };\n\n const onCloseHelpModal = () => {\n setHelpModal(undefined);\n };\n\n const onCloseDarfModal = () => {\n setDarfModal(undefined);\n };\n\n const handleOpenDarfModal = () => {\n setDarfModal(DarfModal(emitted, onCloseDarfModal, () => handleDarf()));\n };\n\n const handleOpenPaymentModal = () => {\n setPaymentModal(!paymentModal);\n };\n\n const handleAdd = (event: React.MouseEvent) => {\n event.stopPropagation();\n setShowAssetModal(true);\n };\n\n const handleEdit = (item: any, index: number) => {\n setShowAssetModal(true);\n setAsset({\n ...item,\n data: moment.utc(item.data.toLocaleString()).format(\"DD/MM/YYYY\"),\n dataVenda: moment\n .utc(item?.dataVenda?.toLocaleString())\n ?.format(\"DD/MM/YYYY\"),\n dataCompra: moment\n .utc(item?.dataCompra?.toLocaleString())\n ?.format(\"DD/MM/YYYY\"),\n });\n };\n\n const handleRemove = (item: any, index: number) => {\n setShowDeleteModal(true);\n setAsset(item);\n };\n\n const addAsset = (asset: any) => {\n setLoading(true);\n (currentPage?.api || apis)\n .post(`${transactionFeature.apiUrl}?manual=true`, asset, {\n headers: {\n \"x-email\": user.user.email,\n },\n })\n .then(() => {\n getTaxes();\n getYearResume(year, true);\n })\n .catch(() => message.error(errorMessage))\n .finally(() => {\n setLoading(false);\n handleCloseAssetModal();\n });\n };\n\n const editAsset = (data: any) => {\n setLoading(true);\n (currentPage?.api || apis)\n .put(`${transactionFeature.apiUrl}/${data.id}`, data)\n .then(() => {\n getTaxes();\n getYearResume(year, true);\n })\n .catch(() => message.error(errorMessage))\n .finally(() => {\n setLoading(false);\n handleCloseAssetModal();\n });\n };\n\n const removeAsset = () => {\n setLoading(true);\n (currentPage?.api || apis)\n .delete(`${transactionFeature.apiUrl}/${asset.id}`)\n .then(() => {\n getTaxes();\n getYearResume(year, true);\n })\n .catch(() => message.error(errorMessage))\n .finally(() => {\n setLoading(false);\n handleCloseDeleteModal();\n });\n };\n\n const handlePlanModal = () => {\n navigate('/planos')\n };\n\n const handleSaveMonth = async () => {\n setLoadingDarfButton(true);\n try {\n const retPlanInfo = await apiPayment.get(\"/user-plan/plan-info\");\n const hasExteriorCalcActive =\n retPlanInfo?.data?.permissions?.[\"crypto-warren\"];\n\n retPlanInfo.data.active || hasExteriorCalcActive\n ? handleDarf(true)\n : handlePlanModal();\n } catch (err) {\n console.error(err);\n }\n setLoadingDarfButton(false);\n };\n\n const handleEventDarfButton = async () => {\n setLoadingDarfButton(true);\n let retPlanInfo = {\n data: {\n active: false,\n },\n };\n try {\n retPlanInfo = await apiPayment.get(\"/user-plan/plan-info\");\n } catch (err) {\n console.error(err);\n }\n setLoadingDarfButton(false);\n\n !retPlanInfo.data.active ? handlePlanModal() : handleOpenPaymentModal();\n };\n\n const handleLabelButton: ReactNode = useMemo(() => {\n // if (!userPlanInfoStatus) return \"Assinar Plano Emissão de DARF\";\n\n if (data?.totalImpostoDevido <= maxDarfPrice) return \"Pagar DARF\";\n else {\n if (emitted) return \"Pagar DARF\";\n else return taxFeature?.buttonLabel;\n }\n }, [data?.totalImpostoDevido, emitted, taxFeature?.buttonLabel]);\n\n const downloadAction = (record: any) => {\n (currentPage?.api || apis)\n .get(`/darf/download/${record._id}`)\n .then((res) => {\n if (res.data.url) {\n download(res.data.url);\n }\n });\n };\n\n const handleNavigateToHistoric = () => {\n navigate(item.settings?.historicUrl ?? `/${currentBroker.path}/historic`, {\n state: { year },\n });\n };\n\n useEffect(() => {\n getTaxes();\n }, [getTaxes, month, year]);\n\n useEffect(() => {\n if (view && darf) {\n getDarf(darf.id);\n }\n }, [getDarf, view, darf]);\n\n useEffect(() => {\n getYearResume(year);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n useEffect(() => {\n const dataToPayment = {\n ...data,\n year,\n month: month + 1,\n name: user.user.name,\n memoriaCalculo: [],\n transactions: data.memoriaCalculo,\n };\n\n setPaymentData(dataToPayment);\n }, [data.totalImpostoDevido, data, month, user.user.name, year]);\n\n const impostoCharge = useMemo(() => {\n return !view\n ? data.impostoDevido +\n data.impostoAcumulado +\n (data.impostoDevido + data.impostoAcumulado < minDarfPrice\n ? 0\n : data.multa + data.juros)\n : Math.max(\n (data.impostoDevido || 0) +\n (data.impostoAcumulado || 0) +\n ((data.impostoDevido || 0) + (data.impostoAcumulado || 0) <\n minDarfPrice\n ? 0\n : data.multa + data.juros),\n 0\n );\n }, [data, view]);\n\n return (\n \n \n \n \n \n \n \n {!view && !viewEdit && (\n <>\n {currentBroker.initialYear === year && month === 0 ? (\n
\n ) : (\n {\n if (month === 0) {\n setYear((year: number) => {\n const newYear = year - 1;\n getYearResume(newYear);\n setDatePickerYear(newYear);\n return newYear;\n });\n setMonth(11);\n } else {\n setMonth((month) => month - 1);\n }\n }}\n />\n )}\n >\n )}\n {monthsExtended[month]} de {year}\n {!view && !viewEdit && (\n <>\n {currentYear === year && currentMonth - 1 === month ? (\n
\n ) : (\n {\n if (month === 11) {\n setYear((year) => {\n const newYear = year + 1;\n getYearResume(newYear);\n setDatePickerYear(newYear);\n return newYear;\n });\n setMonth(0);\n } else {\n setMonth((month) => month + 1);\n }\n }}\n />\n )}\n >\n )}\n \n \n {!view && !viewEdit && (\n \n }\n defaultValue={defaultValue}\n locale={antDatePickerLocale}\n value={moment({ year: datePickerYear, month })}\n onSelect={(e) => {\n if (e.month() === month && year !== datePickerYear) {\n onChangeMonth(e, `${month + 1}-${datePickerYear}`);\n }\n }}\n onPanelChange={(e) => {\n getYearResume(e.year());\n setDatePickerYear(e.year());\n }}\n getPopupContainer={(trigger) => trigger.parentElement!}\n renderExtraFooter={() =>\n yearResumeFeature.disabled ? (\n <>>\n ) : (\n \n )\n }\n disabledDate={(current) =>\n current &&\n (current > currentDate.endOf(\"month\") ||\n current < moment(`${currentBroker.initialYear}-01-01`) ||\n current < moment(`2020-01-01`))\n }\n monthCellRender={(e) => (\n \n )}\n />\n \n )}\n
\n \n \n \n \n {hasPlan && view && darf?.fullPath && (\n \n \n downloadAction(darf)}\n startIcon={\n \n }\n >\n BAIXAR DARF\n \n \n
\n )}\n\n \n {!hasPlan && (\n
\n
\n Contrate o plano premium para ver os impostos devidos\n \n
}\n onClick={() => {\n handlePlanModal();\n closeModal?.();\n }}\n >\n PREMIUM\n \n
\n )}\n\n
\n {DarfImpostosDevidosDescriptions.map((description) => (\n \n {loading ? (\n \n ) : description.render ? (\n description.render(data, view)\n ) : (\n formatCurrency(\n Number(data[description.id as keyof IDarf] || 0)\n )\n )}\n \n ))}\n \n
\n\n \n\n \n \n Vendas realizadas no mês\n \n }\n >\n {!notAuthorized && !view && (\n }\n >\n Adicionar\n \n )}\n (\n handleEdit(item, index)}\n icon={ }\n />,\n handleRemove(item, index)}\n icon={ }\n />,\n ]\n }\n >\n \n \n {item.ativoTranslated\n ? item.ativoTranslated\n : item.ativo}\n \n }\n description={\n \n
\n Data de liquidação:{\" \"}\n {moment\n .utc(item?.data?.toLocaleString())\n .format(\"DD/MM/YYYY\")}\n
\n {/*
\n Data de liquidação:{\" \"}\n \n {formatCurrency(\n item.ganho ??\n item.quantidade *\n (item.precoVenda - item.custoCompra) -\n item.taxas ??\n 0\n )}\n \n
*/}\n {item.quantidade && (\n
Quantidade: {item.quantidade}
\n )}\n {view && (\n <>\n
\n Valor de venda:{\" \"}\n \n {formatCurrency(item.precoVenda)}\n \n
\n
\n Valor de compra:{\" \"}\n \n {formatCurrency(item.custoCompra)}\n \n
\n
\n Taxas:{\" \"}\n {formatCurrency(item.taxas)} \n
\n >\n )}\n
Ganho: {formatCurrency(item.ganho)}
\n
\n }\n />\n \n \n )}\n />\n
\n \n\n {hasPlan && }\n\n {hasPlan && (\n \n Imposto total \n \n {loading ? (\n \n ) : (\n formatCurrency(impostoCharge)\n )}\n \n
\n )}\n\n {hasPlan && (\n \n {DarfResultDescriptions.map((description, i) => (\n \n {loading ? (\n \n ) : description.Component ? (\n \n ) : (\n formatCurrency(\n Number(data[description.id as keyof IDarf] || 0)\n )\n )}\n \n ))}\n \n )}\n\n {!notAuthorized && !view && (\n <>\n \n {!(month === currentMonth && year === currentYear) &&\n (data.impostoDevido + data.impostoAcumulado ?? 0) <\n minDarfPrice &&\n data.memoriaCalculo.filter(\n (el: any) =>\n (el.operation === \"SELL\" &&\n (el.type === \"TRADE\" || el.type === \"WITHDRAW\")) ||\n el.type === \"INTEREST\"\n )?.length > 0 && (\n \n \n \n O valor do imposto devido é menor que{\" \"}\n {formatCurrency(minDarfPrice)}. Você não precisa\n emitir um DARF para este mês.\n >\n }\n >\n handleRegularize(true)}\n startIcon={emitting && }\n >\n Salvar sem emitir DARF\n \n \n \n )}\n {data.impostoDevido + data.impostoAcumulado >= minDarfPrice && (\n \n \n target.parentElement!}\n title={\n month === currentMonth && year === currentYear\n ? \"Aguarde o encerramento do mês para pagar seu DARF\"\n : \"\"\n }\n >\n \n }\n disabled={\n loading ||\n emitting ||\n loadingDarfButton ||\n (month === currentMonth && year === currentYear)\n }\n >\n {handleLabelButton}\n {\" \"}\n \n \n )}\n
\n {!(month === currentMonth && year === currentYear) && (\n \n \n }\n >\n Salvar apenas\n \n \n
\n )}\n >\n )}\n {!view && false && (\n <>\n setShowNotAuthorizedModal(false)}\n />\n \n
\n
\n (1) Os cálculos de imposto de renda foram realizados levando\n em consideração somente as operações de compra e venda\n realizadas na Xtage. Caso você tenha realizado operações com\n criptoativos em outras corretoras, realize as alterações\n manualmente.\n
\n
\n
\n
\n (2) Caso você tenha realizado a transferência de custódia de\n outras corretoras de criptoativos para a Xtage, realize\n manualmente a inserção do custo médio de compra dos\n respectivos ativos transferidos.\n
\n
\n
\n
\n (3) O Usuário é exclusivamente responsável pela conferência\n e validação das informações utilizadas na apuração do\n imposto devido, conforme{\" \"}\n \n Termos e Condições Gerais de Uso.\n \n
\n
\n
\n >\n )}\n \n \n\n \n {helpModal?.content}\n \n \n {darfModal?.content}\n \n \n maxPixPayment}\n disableCreditCardOption={\n data?.totalImpostoDevido > maxCreditCardPayment\n }\n onCancel={handleOpenPaymentModal}\n paymentData={{\n ...paymentData,\n impostoTotalFinal: data.totalImpostoDevido + data.impostoAcumulado,\n }}\n callDarf={() => {\n // HandleTag(\"67\");\n handleOpenDarfModal();\n }}\n maxPixValue={maxPixPayment}\n maxCCValue={maxCreditCardPayment}\n selectedYear={year}\n />\n Promise.resolve(removeAsset())}\n body=\"Você realmente quer deletar esse ativo?\"\n />\n \n );\n};\n","import styled from \"styled-components\";\n\nexport const CardContainer = styled.div`\n width: 160px;\n height: 180px;\n border-radius: 8px;\n transition: opacity 0.3s ease-in-out;\n background-color: var(--velotax-background-color);\n cursor: pointer;\n position: relative;\n box-shadow: 0px 2px 10px 0px rgba(0, 0, 0, 0.15);\n\n &.disabled {\n cursor: default;\n :before {\n content: \"Em breve!\";\n top: 6px;\n left: -6px;\n position: absolute;\n color: #000;\n width: auto;\n height: auto;\n padding: 4px 8px;\n font-size: 12px;\n font-weight: 600;\n display: flex;\n align-items: center;\n justify-content: center;\n background-color: var(--ant-primary-color);\n }\n }\n\n &,\n a,\n .MuiButtonBase-root {\n display: flex;\n align-items: flex-start;\n justify-content: flex-start;\n flex-direction: column;\n border-radius: 8px;\n }\n\n a,\n .MuiButtonBase-root {\n width: 100%;\n height: 100%;\n padding: 16px;\n }\n\n figure {\n margin: 0 0 4px;\n\n svg {\n width: 32px;\n height: 32px;\n &:not(.no-fill) {\n fill: var(--ant-primary-color);\n }\n }\n }\n\n .ant-btn.ant-btn-primary.btn-premium {\n top: 16px;\n right: -6px;\n }\n\n irpf {\n box-shadow: none;\n animation-name: irpf-card;\n animation-duration: 1.75s;\n animation-direction: alternate;\n animation-iteration-count: infinite;\n animation-timing-function: cubic-bezier(0.68, -0.55, 0.265, 1.55);\n }\n\n @keyframes irpf-card {\n 0% {\n box-shadow: 0px 0px 8px 4px var(--ant-primary-color-outline);\n }\n 50% {\n box-shadow: 0px 0px 1px 1px var(--ant-primary-color-outline);\n }\n 100% {\n box-shadow: 0px 0px 8px 4px var(--ant-primary-color-outline);\n }\n }\n\n .content {\n font-weight: 600;\n font-size: 0.875rem;\n line-height: 1.125rem;\n color: var(--velotax-font-color-light);\n\n .ant-typography {\n color: inherit;\n position: relative;\n text-align: left;\n text-transform: uppercase;\n\n .available-until {\n position: absolute;\n bottom: -24px;\n left: 8px;\n width: calc(100% - 16px);\n opacity: 0.8;\n font-weight: 300;\n font-size: 0.5625rem;\n letter-spacing: 0.25px;\n line-height: 11px;\n text-align: center;\n }\n }\n\n .description {\n left: 16px;\n bottom: 16px;\n position: absolute;\n opacity: 0.7;\n font-size: 13px;\n font-weight: 300;\n max-width: calc(100% - 32px);\n text-align: left;\n .ant-typography {\n text-transform: none;\n }\n\n .warning {\n transform: translateY(4px);\n color: var(--ant-warning-color);\n margin-right: 4px;\n font-size: 16px;\n }\n }\n }\n\n &.add {\n background-color: var(--velotax-background-color-ghost);\n border: 2px solid var(--velotax-background-color);\n svg {\n fill: var(--velotax-font-color-ghost);\n }\n .content {\n color: var(--velotax-font-color-ghost);\n }\n }\n`;\n","import { ReactNode } from \"react\";\nimport { ButtonBase } from \"@mui/material\";\nimport { useNavigate } from \"react-router-dom\";\nimport { CardContainer } from \"./styles\";\nimport { isMobile } from \"../../../utils\";\n\ninterface CardProps {\n id: string;\n icon: ReactNode;\n children: ReactNode;\n link?: string;\n className?: string;\n disabled?: boolean;\n onClick?: () => void;\n}\n\nconst Container: React.FC> = ({ children, link }) => {\n return link && link.includes(\"http\") ? (\n \n {children}\n \n ) : (\n <>{children}>\n );\n};\n\nexport const Card = ({\n id,\n icon,\n link,\n onClick,\n children,\n disabled,\n className,\n}: CardProps) => {\n const navigate = useNavigate();\n\n const handleClick = () => {\n if (disabled) return;\n if (onClick) {\n onClick();\n } else if (link && !link.includes(\"http\")) {\n navigate(link);\n }\n };\n\n return (\n \n \n \n {icon} \n {children}
\n \n \n \n );\n};\n","import { Typography } from \"antd\";\nimport { GiProfit } from \"react-icons/gi\";\nimport { AiOutlineApi } from \"react-icons/ai\";\nimport { RiFileList3Line } from \"react-icons/ri\";\nimport { HiOutlineCalculator } from \"react-icons/hi\";\nimport { MdSwapVerticalCircle } from \"react-icons/md\";\nimport { isMobile } from \"../../utils\";\nimport { FaHandHoldingUsd } from \"react-icons/fa\";\n\nexport const itensBolsa = [\n {\n id: \"darf\",\n link: \"/warren/bolsa-historic\",\n content: Calculadora de DARF ,\n description: Cálculo de IR e emissão de DARF ,\n icon: (\n \n ),\n },\n // {\n // premium: true,\n // id: \"anual-report\",\n // link: \"/warren/bolsa-report\",\n // content: Informe de rendimentos ({YearReport}) ,\n // description: Relatório para declaração do IRPF ,\n // icon: (\n // \n // ),\n // },\n {\n id: \"report\",\n premium: true,\n link: \"/warren/bolsa-report\",\n content: (\n \n Relatórios {!isMobile() && \"(padrão Receita Federal)\"}\n \n ),\n icon: (\n \n ),\n description: (\n Relatórios auxiliares para declaração de IR \n ),\n },\n {\n id: \"dividendos\",\n link: \"/warren/bolsa-dividendos\",\n premium: true,\n content: Dividendos ,\n description: Gestão dos dividendos recebidos ,\n icon: ,\n },\n {\n premium: true,\n id: \"insert-manual\",\n link: \"/warren/bolsa-insert-manual\",\n content: Posição 31/12/2019 ,\n icon: (\n \n ),\n },\n {\n id: \"doacoes-heranca\",\n link: \"/warren/bolsa-doacoes-heranca\",\n content: Doações e heranças ,\n icon: (\n \n ),\n },\n {\n id: \"integration\",\n link: \"/warren/bolsa-integration\",\n content: Notas de corretagem ,\n icon: ,\n },\n];\n","import { Typography } from \"antd\";\n// import { GiProfit } from \"react-icons/gi\";\nimport { AiOutlineApi } from \"react-icons/ai\";\n// import { RiFileList3Line } from \"react-icons/ri\";\nimport { HiOutlineCalculator } from \"react-icons/hi\";\n// import { MdSwapVerticalCircle } from \"react-icons/md\";\n// import { isMobile } from \"../../utils\";\n\nexport const itensCrypto = [\n {\n premium: false,\n id: \"darf\",\n link: \"/warren/cripto-historic\",\n content: Calculadora de DARF ,\n description: Cálculo de IR e emissão de DARF ,\n icon: (\n \n ),\n },\n // {\n // premium: true,\n // id: \"anual-report\",\n // link: \"/warren/bolsa-report\",\n // content: Informe de rendimentos ({YearReport}) ,\n // description: Relatório para declaração do IRPF ,\n // icon: (\n // \n // ),\n // },\n // {\n // premium: true,\n // id: \"report\",\n // link: \"/warren/cripto-report\",\n // content: (\n // \n // Relatórios {!isMobile() && \"(padrão Receita Federal)\"}\n // \n // ),\n // icon: (\n // \n // ),\n // description: (\n // Relatórios auxiliares para declaração de IR \n // ),\n // },\n // {\n // id: \"dividendos\",\n // link: \"/warren/bolsa-dividendos\",\n // premium: true,\n // content: Dividendos ,\n // description: Gestão dos dividendos recebidos ,\n // icon: ,\n // },\n // {\n // premium: true,\n // id: \"insert-manual\",\n // link: \"/warren/bolsa-insert-manual\",\n // content: Posição 31/12/2019 ,\n // icon: (\n // \n // ),\n // },\n {\n id: \"integration\",\n premium: false,\n link: \"/warren/cripto-integration\",\n content: Notas de corretagem ,\n description: Carregar notas de corretagem ,\n\n icon: ,\n },\n];\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 16px;\n .ant-typography {\n text-align: center;\n color: var(--velotax-font-color);\n }\n h1 {\n font-size: 32px;\n }\n div.ant-typography {\n font-size: 18px;\n }\n @media only screen and (max-device-width: 812px) {\n h1 {\n font-size: 24px;\n }\n div.ant-typography {\n font-size: 14px;\n }\n }\n`;\n","import { Typography } from \"antd\";\nimport { Container } from \"./styles\";\n\ninterface PageInfoProps {\n title: React.ReactNode;\n subtitle?: React.ReactNode;\n}\n\nexport const PageInfo: React.FC = ({ title, subtitle }) => {\n return (\n \n {title} \n {subtitle && {subtitle} }\n \n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n width: 100%;\n padding: 16px;\n border-radius: 8px;\n border: 1px solid var(--velotax-background-color-ghost);\n h3 {\n margin: 0;\n font-size: 16px;\n font-weight: 600;\n text-transform: uppercase;\n color: var(--velotax-font-color-ghost);\n }\n .price {\n display: flex;\n column-gap: 8px;\n align-items: center;\n justify-content: space-between;\n margin-bottom: 24px;\n h4 {\n font-size: 22px;\n margin: 8px 0 !important;\n color: var(--ant-primary-color);\n del,\n span.per-month {\n font-size: 14px;\n }\n del {\n opacity: 0.8;\n color: var(--velotax-font-color);\n }\n }\n .ant-typography-danger {\n font-size: 11px;\n text-align: right;\n max-width: 120px;\n }\n }\n .ant-divider.ant-divider-horizontal {\n margin: 16px 0;\n }\n .other-features,\n .included-features {\n .feature {\n display: flex;\n align-items: center;\n justify-content: space-between;\n span {\n font-weight: 500;\n }\n svg.fill {\n fill: var(--velotax-font-color);\n }\n svg:not(.fill) {\n stroke: var(--velotax-font-color);\n }\n &.disabled {\n opacity: 0.4;\n }\n .feature-desc {\n display: block;\n font-size: 10px;\n font-weight: 400;\n }\n }\n }\n .other-features {\n margin-bottom: 24px;\n }\n`;\n","import clsx from \"clsx\";\nimport { BiCheck, BiX } from \"react-icons/bi\";\nimport { Divider, Row, Typography } from \"antd\";\nimport Button from \"../../Button\";\nimport { Container } from \"./styles\";\nimport { formatCurrency } from \"../../../utils\";\nimport { OtherFeatures, Plan as IPlan } from \"../../../constants/plans\";\n\ninterface PlanProps {\n plan: IPlan;\n buyed: boolean;\n handleSelect: React.MouseEventHandler;\n}\n\nexport const Plan: React.FC = ({ plan, buyed, handleSelect }) => {\n return (\n \n {plan.title} \n \n \n {formatCurrency(plan.price)} / mês \n \n \n {formatCurrency(plan.originalPrice)}{\" \"}\n / mês \n \n \n \n {plan.discount}% de desconto no plano anual\n \n
\n \n {plan.includedFeatures.map((feature) => (\n
\n {feature.name} \n {feature.icon} \n
\n ))}\n
\n \n \n {OtherFeatures.map((feature) => (\n
feature.name === f.name) <\n 0,\n })}\n >\n \n {feature.name}\n {feature.desc && (\n {feature.desc} \n )}\n \n \n {plan.otherFeatures.findIndex((f) => feature.name === f.name) <\n 0 ? (\n \n ) : (\n \n )}\n \n
\n ))}\n
\n \n {!buyed ? (\n \n Selecionar\n \n ) : (\n \n Plano Selecionado!\n \n )}\n
\n \n );\n};","import { Col, Row } from \"antd\";\nimport Button from \"../../../../Button\";\nimport React, { Dispatch, SetStateAction } from \"react\";\n\ninterface IFooterProps {\n currentStep: number;\n handleProceed: () => void;\n setCurrentStep: Dispatch>;\n typePurchase: string;\n hideTerms: boolean;\n backtoRoot?: any;\n selectedCardId?: string;\n}\n\nconst Footer: React.FC = ({\n currentStep,\n setCurrentStep,\n handleProceed,\n typePurchase,\n hideTerms,\n selectedCardId,\n backtoRoot,\n}) => {\n return (\n <>\n {/*\n {currentStep !== 2 && !hideTerms && (\n \n \n Ao realizar o pagamento você concorda com as{\" \"}\n \n \n )} \n
*/}\n \n \n \n {currentStep !== 3 && currentStep !== 4 && (\n \n \n {currentStep === 1\n ? \"PROSSEGUIR\"\n : typePurchase === \"UNIQUE\"\n ? \"FINALIZAR PAGAMENTO\"\n : \"FINALIZAR COMPRA\"}\n \n \n )}\n {currentStep <= 2 && (\n \n {\n if (currentStep === 1) {\n backtoRoot();\n return;\n }\n\n setCurrentStep((step) => step - 1);\n }}\n >\n Voltar \n \n \n )}\n
\n \n
\n >\n );\n};\n\nexport default Footer;\n","import styled from \"styled-components\";\nimport { Row } from \"antd\";\n\nexport const CardContent = styled.div`\n display: flex;\n background-color: transparent;\n align-items: center;\n width: 100%;\n justify-content: center;\n height: 115px;\n border-radius: 7px;\n margin-top: 10px;\n cursor: pointer;\n\n .full-container {\n margin: auto;\n\n .card-number {\n display: flex;\n }\n }\n`;\n\nexport const NewCardContent = styled.div`\n display: flex;\n justify-content: center;\n align-items: center;\n cursor: pointer;\n\n h3 {\n color: var(--ant-primary-color);\n }\n`;\n\nexport const Content = styled(Row)`\n margin-top: 4%;\n\n .list-cards-content {\n height: 400px;\n width: auto;\n display: grid;\n grid-template-columns: 1fr;\n overflow-y: auto;\n }\n\n .rccs {\n transform: scale(55%);\n }\n`;\n","export const taxes: Array = [\n 1.065, 1.105, 1.125, 1.145, 1.165, 1.185, 1.205, 1.225, 1.245, 1.265, 1.285,\n 1.305,\n];","import { Col, Row } from \"antd\";\nimport { taxes } from \"../constants/taxes\";\nimport { formatCurrency, numberToPercentage } from \"../utils\";\n\nexport default function getInstallments(\n value: number, // valor\n isPlan: boolean, // caso seja plano irá alterar a descrição\n fixedInstallments?: number //número de parcela fixo\n) {\n var _installments = [];\n var customInstallments = 12;\n\n if (value) {\n if (isPlan) {\n _installments.push({\n value: 12,\n label: (\n \n R$ {(value / 12).toFixed(2)}\n \n ),\n });\n } else if (fixedInstallments) {\n _installments.push({\n value: 12,\n label: (\n \n {fixedInstallments}x R${\" \"}\n {(value / fixedInstallments).toFixed(2)}\n \n ),\n });\n } else {\n\n let valorTaxa=0;\n if(value >=10 && value <=499.99){\n valorTaxa = 9.9\n }\n if(value >=500 && value <=999.99){\n valorTaxa = 14.9\n }\n if(value >=1000 && value <=1499.99){\n valorTaxa = 19.9\n }\n if(value >=1500 && value <=10000){\n valorTaxa = 29.9\n }\n\n const _valueTmp = value+valorTaxa;\n\n for (let i = 0; i < (customInstallments || 12); i++) {\n let _value = formatCurrency(((_valueTmp * taxes[i])) / (i + 1));\n let plural = i > 0 ? \"parcelas\" : \"parcela\";\n\n _installments.push({\n value: i + 1,\n label: (\n \n \n {i + 1}x {plural} - {_value.props.prefix} {_value.props.value}\n \n \n Taxa de serviço: {numberToPercentage((taxes[i] - 1) * 100)}\n \n
\n ),\n });\n }\n }\n }\n\n return _installments;\n}\n","import React, { useEffect, useState, useMemo, forwardRef } from \"react\";\nimport { message, Typography, Col, Form, Row } from \"antd\";\nimport { CardContent, NewCardContent, Content } from './styles'\nimport { apiPayment } from \"../../services/apiPayment\";\nimport Cards from \"react-credit-cards\";\nimport { LoadingOutlined } from \"@ant-design/icons\";\nimport { Plan as IPlan, UserPlan } from \"../../constants/plans\";\nimport getInstallmentsByPrice from '../../utils/getInstallmentsListByPrice';\nimport { formatCurrency } from '../../utils/format';\nimport { SelectLabled } from \"../../constants/inputs\";\n\ninterface ICardList {\n onSelectCard: (card: any) => void;\n selectedCardId: string;\n amount?: string;\n plan?: IPlan | UserPlan;\n value: number;\n onChangeInstallments?: (value: number) => void;\n installmentSelected?: number;\n}\n\nexport interface cardAttributes {\n id: string;\n first_six_digits: string;\n last_four_digits: string;\n brand: string;\n customer_id: string;\n holder_name: string;\n exp_month: number;\n exp_year: number;\n status: string;\n type: string;\n created_at: Date;\n updated_at: Date;\n billing_address: {\n street: string;\n zip_code: string;\n city: string;\n state: string;\n country: string;\n line_1: string;\n };\n}\n\nexport const CardList: React.FC = ({\n onSelectCard,\n selectedCardId,\n value,\n plan,\n onChangeInstallments,\n installmentSelected\n}) => {\n const [cardList, setCardList] = useState([]);\n const [loading, setLoading] = useState(false);\n\n\n const SelectInput = forwardRef((props, ref) => (\n \n ));\n\n const getCardList = () => {\n setLoading(true);\n apiPayment.get(`/payment/list_cards`).then((ret) => {\n setCardList(ret?.data?.list || []);\n\n if (ret?.data?.list?.length === 0 || !ret?.data?.list) {\n onSelectCard(undefined);\n }\n setLoading(false);\n }).catch((err) => {\n message.error('Ocorreu um erro ao buscar cartões');\n setLoading(false);\n });\n\n };\n\n useEffect(() => {\n getCardList();\n // eslint-disable-next-line\n }, []);\n\n\n const fixedInstallments = useMemo(() => !!plan?.type ? 12 : undefined, [plan?.type])\n\n const installments = useMemo(() => getInstallmentsByPrice(value, !!plan?.type, fixedInstallments), [plan, value, fixedInstallments]);\n\n\n return (\n \n
\n {!loading &&\n <>\n \n {/* {getMessage()} */}\n \n {value ? (\n {formatCurrency(value ?? 0)} \n ) : undefined}\n\n \n \n onSelectCard(undefined)}>Novo Cartão \n \n \n {cardList && cardList?.length > 0 && cardList?.map((card: cardAttributes) =>\n \n onSelectCard(card)}>\n {\n console.log('cb issuer isvalid', issuer, isValid)\n }}\n />\n \n \n )}\n
\n \n {value ? (\n \n \n \n \n \n ) : (\n \"\"\n )}\n >\n }\n \n );\n};\n","import { Focused } from \"react-credit-cards\";\nimport { Form, message, Spin } from \"antd\";\nimport React, {\n Dispatch,\n SetStateAction,\n useEffect,\n useState,\n useMemo,\n} from \"react\";\nimport Footer from \"./components/Footer\";\nimport { CardList } from \"../../CardList\";\nimport { CardForm } from \"./components/CardForm\";\nimport AddressForm from \"./components/AddressForm\";\nimport { apiPlan } from \"../../../services/apiPlan\";\nimport { useAuth } from \"../../../contexts/AuthContext\";\nimport { ResultScreen } from \"./components/ResultScreen\";\nimport { removeSpecialCharacters } from \"../../../utils\";\nimport { Plan as IPlan, UserPlan } from \"../../../constants/plans\";\nimport { PendingScreen } from \"./components/ResultScreen/components/PendingScreen\";\nimport { useBroker } from \"../../../contexts/BrokerContext\";\nimport { apiPayment } from \"../../../services/apiPayment\";\nimport HandleTag from \"../../../services/handleTag\";\nimport { AxiosInstance } from \"axios\";\n\nexport enum ETypePurchase {\n MEI = \"MEI\",\n UNIQUE = \"UNIQUE\",\n VELOTAX_MAIN_BASIC = \"VELOTAX_MAIN_BASIC\",\n VELOTAX_MAIN_BASIC_MONTH = \"VELOTAX_MAIN_BASIC_MONTH\",\n VELOTAX_MAIN_PRO = \"VELOTAX_MAIN_PRO\",\n VELOTAX_MAIN_PRO_MONTH = \"VELOTAX_MAIN_PRO_MONTH\",\n VELOTAX_MAIN_CONCIERGE = \"VELOTAX_MAIN_CONCIERGE\",\n}\n\ninterface ICreditCardStepProps {\n status: string;\n initialized: boolean;\n handleBack?: () => void;\n typePurchase: ETypePurchase;\n handleSuccess?: (res: any) => void;\n setInitialized: Dispatch>;\n handleClosableModal: Dispatch>;\n plan?: IPlan | UserPlan;\n darfData?: any;\n getCardCurrentStep?: (value: number) => void;\n handleIdBack?: (event: any) => void;\n customInstallments?: number;\n backToRoot?: () => void;\n cupom?: any;\n changeCardAction?: boolean;\n closeMainModal?: () => void;\n api?: AxiosInstance;\n}\n\nexport const CreditCardStep: React.FC = ({\n plan,\n api,\n handleBack,\n handleClosableModal,\n typePurchase,\n darfData,\n handleSuccess,\n handleIdBack,\n customInstallments,\n backToRoot,\n cupom,\n changeCardAction,\n closeMainModal,\n}) => {\n const [cardForm] = Form.useForm();\n const [addressForm] = Form.useForm();\n\n const { user } = useAuth();\n const [cvc, setCvc] = useState(\"\");\n const [name, setName] = useState(\"\");\n const [phone, setPhone] = useState(\"\");\n const [status, setStatus] = useState(\"\");\n const [number, setNumber] = useState(\"\");\n const [expiry, setExpiry] = useState(\"\");\n const [focused, setFocused] = useState();\n const [holderDocument, setHolderDocument] = useState(\"\");\n const [installments, setInstallments] = useState(1);\n\n const [selectedCard, setSelectedCard] = useState();\n\n const [loading, setLoading] = useState(false);\n const [currentStep, setCurrentStep] = useState(1);\n const [isValidCardForm, setIsValidCardForm] = useState(false);\n const [isValidAddressForm, setIsValidAddressForm] = useState(false);\n\n const isPlan = useMemo(() => {\n return [\n \"VELOTAX_MAIN_BASIC\",\n \"VELOTAX_MAIN_PRO\",\n \"VELOTAX_MAIN_PRO_MONTH\",\n \"VELOTAX_MAIN_CONCIERGE\",\n ].includes(plan?.type || \"\");\n }, [plan]);\n\n const value = useMemo(() => {\n if (isPlan) {\n var isMonth = (plan as any).interval === \"Mensal\";\n\n var planPrice = (!isMonth ? (plan?.price || 0) * 12 : plan?.price) || 0;\n\n if (cupom?.type === \"absolute\") {\n planPrice = planPrice - Number(cupom?.discountValue);\n } else if (cupom?.type === \"percent\") {\n planPrice = planPrice - planPrice * Number(cupom?.discountValue / 100);\n }\n\n planPrice = Number(planPrice.toFixed(2));\n\n return planPrice && isMonth ? planPrice * 12 : planPrice;\n }\n\n return darfData?.impostoTotalFinal;\n }, [\n darfData?.impostoTotalFinal,\n plan,\n isPlan,\n cupom?.discountValue,\n cupom?.type,\n ]);\n\n const { currentPage } = useBroker();\n\n const getCardInfo = () => {\n const addressInfo = addressForm.getFieldsValue();\n const monthExp = expiry?.split(\"/\")?.[0];\n const monthYear = expiry?.split(\"/\")?.[1];\n\n let cardInfo: any = {};\n\n const cardId = selectedCard?.id;\n\n if (cardId) {\n cardInfo.card_id = cardId;\n } else {\n cardInfo = {\n line_1: `${addressInfo?.neighborhood} - ${addressInfo?.street} - ${addressInfo?.number}`,\n zip_code: addressInfo?.cep?.replace(\"-\", \"\"),\n city: addressInfo?.city,\n state: addressInfo?.state.toUpperCase(),\n holder_name: removeSpecialCharacters(name),\n number,\n exp_month: monthExp,\n exp_year: monthYear,\n holderDocument,\n cvv: cvc,\n };\n }\n\n return cardInfo;\n };\n\n useEffect(() => {\n if (currentStep < 0) {\n backToRoot && backToRoot();\n }\n }, [currentStep, backToRoot]);\n\n const getCustomerInfo = () => {\n let customerInfo;\n if (selectedCard?.customer_id) {\n customerInfo = { customer_id: selectedCard?.customer_id };\n } else {\n customerInfo = {\n name: user?.user?.name,\n email: user?.user?.email,\n document: user?.user?.cpf,\n phone,\n };\n }\n\n return customerInfo;\n };\n\n const createMeiPayment = () => {\n const addressInfo = addressForm.getFieldsValue();\n\n const url = \"/payment/plan/create_payment\";\n const monthExp = expiry?.split(\"/\")?.[0];\n const monthYear = expiry?.split(\"/\")?.[1];\n\n setLoading(true);\n handleClosableModal(false);\n (currentPage?.api || apiPlan)\n .post(url, {\n type: plan?.type,\n aceitouCGV: true,\n naoPossuiPendencia: true,\n paymentForm: \"credit_card\",\n clientInfo: {\n name: user?.user?.name,\n email: user?.user?.email,\n document: user?.user?.cpf,\n phone,\n },\n cardInfo: {\n cvv: cvc,\n holderDocument,\n number: number,\n exp_year: monthYear,\n exp_month: monthExp,\n holder_name: removeSpecialCharacters(name),\n country: \"BR\",\n city: addressInfo?.city,\n state: addressInfo?.state.toUpperCase(),\n zip_code: addressInfo?.cep?.replace(\"-\", \"\"),\n line_1: `${addressInfo?.neighborhood} - ${addressInfo?.street} - ${addressInfo?.number}`,\n },\n })\n .then((res: any) => {\n setCurrentStep(2);\n if (res?.data?.status?.toLowerCase() === \"active\") {\n setStatus(\"active\");\n }\n })\n .catch((err) => {\n message.error(err.response?.data?.message);\n })\n .finally(() => {\n setLoading(false);\n handleClosableModal(true);\n });\n };\n\n const createPlanMainVelotax = () => {\n const url = \"/user-plan\";\n\n let cardInfo: any = getCardInfo();\n let clientInfo: any = getCustomerInfo();\n\n setLoading(true);\n handleClosableModal(false);\n\n apiPayment\n .post(url, {\n ...darfData,\n type: plan?.type,\n from: \"WARREN\",\n cupomCode: cupom?.cupomCode.toUpperCase(),\n paymentForm: \"credit_card\",\n installments: 12,\n cardInfo,\n clientInfo,\n acceptTerms: true,\n })\n .then((res: any) => {\n if (res.data.paymentId || res.data.paymentRecurringId) {\n handleSuccess && handleSuccess(res.data);\n if(cupom?.cupomCode.toUpperCase() === 'VELOTAXER20') {\n HandleTag('171')\n }\n }\n setCurrentStep(4);\n })\n .catch((err) => {\n message.error(err.response?.data?.message);\n })\n .finally(() => {\n setLoading(false);\n handleClosableModal(true);\n });\n };\n const createUniquePayment = () => {\n let url = \"/darf\";\n\n let cardInfo: any = getCardInfo();\n let clientInfo: any = getCustomerInfo();\n\n setLoading(true);\n handleClosableModal(false);\n\n (api || currentPage?.api || apiPlan)\n .post(url, {\n ...darfData,\n paymentForm: \"credit_card\",\n paymentInfo: {\n installments: installments,\n cardInfo: cardInfo,\n clientInfo: clientInfo,\n },\n })\n .then((res: any) => {\n if (handleIdBack) handleIdBack(res?.data);\n setCurrentStep(3);\n })\n .catch((err) => {\n message.error(err.response?.data?.message);\n })\n .finally(() => {\n setLoading(false);\n handleClosableModal(true);\n });\n };\n\n const changeCardService = async () => {\n let cardInfo: any = getCardInfo();\n\n setLoading(true);\n\n apiPayment\n .post(`/user-plan/change_signature_card`, {\n card: cardInfo,\n })\n .then((ret) => {\n message.success(\"Cartão atualizado com sucesso\");\n })\n .catch((err) => {\n message.error(\"Ocorreu um erro ao alterar cartão\");\n })\n .finally(() => {\n setLoading(false);\n closeMainModal && closeMainModal();\n });\n };\n\n const handleNextStep = () => {\n if (getConditionToDisable()) {\n if (currentStep === 1) cardForm.validateFields();\n if (currentStep === 2) addressForm.validateFields();\n message.warning(\"Preencha corretamente o formulário\");\n return;\n }\n if (currentStep === 1) {\n if (!darfData?.impostoTotalFinal) {\n HandleTag(\"57\");\n }\n setCurrentStep(2);\n } else if (currentStep === 2 || currentStep === 0) {\n if (changeCardAction) {\n changeCardService();\n } else if (typePurchase === \"MEI\") {\n createMeiPayment();\n } else if (typePurchase === \"UNIQUE\") {\n createUniquePayment();\n } else if ([\"VELOTAX_MAIN_BASIC\", \"VELOTAX_MAIN_PRO\", \"VELOTAX_MAIN_PRO_MONTH\", \"VELOTAX_MAIN_CONCIERGE\"].includes(typePurchase)) {\n createPlanMainVelotax();\n }\n }\n };\n\n const getConditionToDisable = () => {\n if (currentStep === 0 && !selectedCard?.id) return true;\n if (currentStep === 1 && !isValidCardForm) return true;\n if (currentStep === 2 && !isValidAddressForm) return true;\n return false;\n };\n\n const changeInstallments = (value: any) => {\n setInstallments(value);\n };\n\n return (\n \n
\n {currentStep === 0 && (\n <>\n {\n setSelectedCard(card);\n\n if (!card) setCurrentStep(1);\n }}\n />\n >\n )}\n\n {currentStep === 1 && (\n <>\n \n \n >\n )}\n\n {currentStep === 2 && (\n <>\n \n \n >\n )}\n\n {currentStep === 3 && (\n \n )}\n\n {currentStep === 4 && }\n \n \n
\n );\n};\n\nexport default CreditCardStep;\n","import Payment from \"payment\";\nimport Masked from \"react-input-mask\";\nimport { TextField } from \"@mui/material\";\nimport { Focused } from \"react-credit-cards\";\nimport NumberFormat from \"react-number-format\";\nimport React, { forwardRef, useMemo } from \"react\";\nimport { Col, Form, FormInstance, Row } from \"antd\";\nimport { SelectLabled } from \"../../../../../constants/inputs\";\nimport getInstallmentsListByPrice from \"../../../../../utils/getInstallmentsListByPrice\";\nimport { cpfInputMask, getMuiInputCss } from \"../../../../../utils\";\nimport {\n validationAmexCVV,\n validationCardShelfLife,\n validationCpf,\n validationCreditCardNumber,\n validationCVV,\n validationFieldRequired,\n validationPhoneRequired,\n} from \"../../../../../utils/formValidations\";\n\ninterface ICardFormProps {\n number: string;\n focused?: Focused;\n form: FormInstance;\n uniqueValue?: number;\n setCvc: (cvc: string) => void;\n setName: (name: string) => void;\n setPhone: (value: string) => void;\n setNumber: (number: string) => void;\n setExpiry: (expiry: string) => void;\n setFocused: (focused: Focused) => void;\n setHolderDocument: (value: string) => void;\n setIsValidCardForm: (isValidCardForm: boolean) => void;\n handleBack: (event: any) => void;\n fixedInstallments?: number;\n signature?: boolean;\n installmentSelected?: number;\n customInstallments?: number;\n addInfo?: Record;\n}\n\nconst CreditCardNumber = forwardRef((props, ref) => (\n \n));\n\nconst CreditCardNumberDiner = forwardRef((props, ref) => (\n \n));\n\nconst CreditCardNumberAmex = forwardRef((props, ref) => (\n \n));\n\nconst ExpiryInput = forwardRef((props, ref) => (\n \n));\n\nconst PhoneInput = forwardRef((props, ref) => (\n \n));\n\nconst CPFInput = forwardRef((props, ref) => (\n \n));\n\nconst CVVAmexInput = forwardRef((props, ref) => (\n \n));\n\nconst CVVInput = forwardRef((props, ref) => (\n \n));\n\nexport const CardForm: React.FC = ({\n form,\n number,\n setCvc,\n setName,\n setPhone,\n setNumber,\n setExpiry,\n setFocused,\n setHolderDocument,\n setIsValidCardForm,\n uniqueValue,\n fixedInstallments,\n handleBack,\n signature,\n installmentSelected,\n customInstallments,\n addInfo\n}) => {\n const cardType = useMemo(() => Payment.fns.cardType(number), [number]);\n\n let installments = useMemo(\n () =>\n getInstallmentsListByPrice(\n uniqueValue || 0,\n !!signature,\n fixedInstallments\n ),\n [signature, fixedInstallments, uniqueValue]\n );\n\n const onValuesChange = (changed: any, values: any) => {\n setCvc(values.cvv ?? \"\");\n setExpiry(values.expiry ?? \"\");\n setName(values.holderName ?? \"\");\n setHolderDocument(values.cpf?.replace(/[_.-]/g, \"\") ?? \"\");\n setPhone(values.phone?.replace(/[()_-]/g, \"\").replace(\" \", \"\") ?? \"\");\n const key = Object.keys(changed)[0];\n setFocused(\n (key === \"holderName\" ? \"name\" : key === \"cvv\" ? \"cvc\" : key) as Focused\n );\n form.setFieldsValue({\n ...values,\n cpf: values.cpf?.replace(/[_.-]/g, \"\") ?? \"\",\n phone: values.phone?.replace(/[()_-]/g, \"\").replace(\" \", \"\") ?? \"\",\n });\n\n setTimeout(() => {\n form\n .validateFields()\n .then(() => setIsValidCardForm(true))\n .catch(() => setIsValidCardForm(false));\n }, 500);\n };\n\n const SelectInput = forwardRef((props, ref) => (\n \n ));\n\n return (\n <>\n \n >\n );\n};\n","import axios from \"axios\";\n\nconst cep = axios.create({\n method: \"GET\",\n baseURL: \"https://viacep.com.br/ws/\",\n});\n\nexport default cep;\n","import React, { forwardRef } from \"react\";\nimport { TextField } from \"@mui/material\";\nimport NumberFormat from \"react-number-format\";\nimport { Col, Form, FormInstance, Row } from \"antd\";\nimport cep from \"../../../../../services/cep\";\nimport { getMuiInputCss } from \"../../../../../utils\";\nimport {\n validationCep,\n validationFieldRequired,\n validationStateAddress,\n} from \"../../../../../utils/formValidations\";\n\ninterface IAddressFormProps {\n form: FormInstance;\n setLoading: (loading: boolean) => void;\n setIsValidAddressForm: (isValidAddressForm: boolean) => void;\n}\n\nconst AddressForm: React.FC = ({\n form,\n setLoading,\n setIsValidAddressForm,\n}) => {\n const getAddressSuggestion = (value: string) => {\n if (value?.length === 8) {\n setLoading(true);\n cep\n .get(`${value}/json`)\n .then((res) => {\n const data = res.data;\n\n if (data.cep) {\n form.setFieldsValue({\n street: data.logradouro,\n neighborhood: data.bairro,\n city: data.localidade,\n state: data.uf,\n });\n } else {\n throw Error();\n }\n })\n .catch(() => {\n form.setFieldsValue({\n street: undefined,\n neighborhood: undefined,\n city: undefined,\n state: undefined,\n number: undefined,\n });\n })\n .finally(() => {\n onFieldsChange();\n setLoading(false);\n });\n }\n };\n\n const onFieldsChange = () => {\n form\n .validateFields()\n .then(() => setIsValidAddressForm(true))\n .catch(() => setIsValidAddressForm(false));\n };\n\n const CEPInput = forwardRef((props, ref) => (\n getAddressSuggestion(value?.value)}\n > \n ));\n\n return (\n <>\n \n >\n );\n};\n\nexport default AddressForm;\n","import React from \"react\";\nimport { BiErrorAlt } from \"react-icons/bi\";\nimport { Col, Row, Typography } from \"antd\";\nimport Button from \"../../../../../../Button\";\n\ninterface IErrorScreenProps {\n setCurrentStep: (currentStep: number) => void;\n}\n\nexport const ErrorScreen: React.FC = ({\n setCurrentStep,\n}) => {\n const callBack = (cb: Function) => {\n setCurrentStep(0);\n setTimeout(() => cb(), 200);\n };\n\n return (\n \n \n \n \n\n \n \n Ocorreu um erro durante a confirmação do pagamento.\n \n \n\n {\n callBack(() => {});\n }}\n >\n Escolher outra forma de pagamento\n \n
\n );\n};\n","import React, { useEffect } from \"react\";\nimport { MdCheck } from \"react-icons/md\";\nimport { Col, Row, Typography } from \"antd\";\nimport { Plan } from \"../../../../../../../constants/plans\";\nimport { useAuth } from \"../../../../../../../contexts/AuthContext\";\nimport { useBroker } from \"../../../../../../../contexts/BrokerContext\";\n\ninterface ConfirmedScreenProps {\n plan?: Plan;\n}\n\nexport const ConfirmedScreen: React.FC = ({ plan }) => {\n const {\n user: { token },\n } = useAuth();\n const { getIntegrationStatus } = useBroker();\n\n useEffect(() => {\n getIntegrationStatus({ token, hideLoading: true });\n }, [getIntegrationStatus, token]);\n\n return (\n \n \n \n \n \n \n {plan\n ? `Pagamento confirmado com sucesso! Parabéns você contratou o plano '\n ${plan?.title.toUpperCase()}'`\n : \"Pagamento realizado com sucesso!\"}\n \n \n
\n );\n};\n","import React, { Dispatch, SetStateAction } from \"react\";\nimport { Plan } from \"../../../../../constants/plans\";\nimport { ErrorScreen } from \"./components/ErrorScreen\";\nimport { ConfirmedScreen } from \"./components/ConfirmScreen\";\n\ninterface IResultScreen {\n plan?: Plan;\n setCurrentStep: Dispatch>;\n status?: string;\n}\n\nexport const ResultScreen: React.FC = ({\n plan,\n setCurrentStep,\n status = \"pending\",\n}) => {\n return (\n <>\n {status === \"active\" ? (\n \n ) : (\n \n )}\n >\n );\n};\n","import React from \"react\";\nimport { Col, Row, Typography } from \"antd\";\nimport { BsClockHistory } from \"react-icons/bs\";\n\ninterface IPendingScreenProps {}\n\nexport const PendingScreen: React.FC = () => {\n const getMessage = () =>\n \"Aguardando confirmação de pagamento no cartão, assim que confirmado o plano será ativado automaticamente.\";\n\n return (\n \n \n \n \n\n \n \n {getMessage()}\n \n \n
\n );\n};\n","import { Form, Select } from \"antd\";\nimport React, { KeyboardEvent } from \"react\";\nimport ReactInputMask from \"react-input-mask\";\nimport { cnpjInputMask, isValidCNPJ } from \"../../utils\";\nimport {\n validationCnpj,\n validationFieldRequired,\n} from \"../../utils/formValidations\";\n\nexport enum MEIAtividadePrincipalEnum {\n SERVICO = \"Serviço\",\n COMERCIO_INDUSTRIA = \"Comércio e Indústria\",\n // COMERCIO_SERVICOS = \"Comércio e Serviços\",\n}\n\nexport const MEIAtividadePrincipalValue: {\n [key: number]: MEIAtividadePrincipalEnum;\n} = {\n 61.6: MEIAtividadePrincipalEnum.COMERCIO_INDUSTRIA,\n 65.6: MEIAtividadePrincipalEnum.SERVICO,\n // 66.6: MEIAtividadePrincipalEnum.COMERCIO_SERVICOS,\n};\n\nexport interface IdentificationData {\n companyCode: string;\n mainActivity: MEIAtividadePrincipalEnum;\n}\n\nexport const isValidAtividadePrincipal = (value: string) =>\n Object.values(MEIAtividadePrincipalEnum).includes(\n value as MEIAtividadePrincipalEnum\n );\n\nexport const validateIdentification = (data: IdentificationData) =>\n isValidCNPJ(data.companyCode) && isValidAtividadePrincipal(data.mainActivity);\n\ninterface Question {\n id: string;\n title: React.ReactNode;\n formItems: { name: string; FormItem: React.FC<{ tabIndex?: number }> }[];\n}\n\nexport const identificationQuestions: Question[] = [\n {\n id: \"companyCode\",\n title: \"Qual é o CNPJ do seu MEI?\",\n formItems: [\n {\n name: \"companyCode\",\n FormItem: ({ tabIndex }) => (\n \n \n \n ),\n },\n ],\n },\n {\n id: \"mainActivity\",\n title: \"Qual é a atividade principal do seu MEI?\",\n formItems: [\n {\n name: \"mainActivity\",\n FormItem: ({ tabIndex }) => (\n \n ) => {\n if (event.key === \"Enter\") {\n event.stopPropagation();\n }\n }}\n >\n \n {MEIAtividadePrincipalEnum.SERVICO}\n \n \n {MEIAtividadePrincipalEnum.COMERCIO_INDUSTRIA}\n \n \n \n ),\n },\n ],\n },\n];\n","import { useState } from \"react\";\nimport { ModalProps } from \"antd\";\nimport { LoadingOutlined } from \"@ant-design/icons\";\nimport { Plan } from \"./Plan\";\nimport { useBroker } from \"../../contexts/BrokerContext\";\nimport { Content, PlanModalStyled, Loading } from \"./styles\";\nimport { Plans, Plan as IPlan } from \"../../constants/plans\";\nimport CreditCardStep, { ETypePurchase } from \"../Payment/CreditCardStep\";\n\ninterface PlanModalProps extends ModalProps {\n modalPlans?: IPlan[];\n}\n\nexport const PlanModal: React.FC = ({\n title,\n modalPlans,\n onCancel,\n ...rest\n}) => {\n const {\n integration: { plans },\n } = useBroker();\n\n const [plan, setPlan] = useState();\n const [loading, setLoading] = useState(false);\n const [closable, setClosable] = useState(true);\n\n const handleSelectPlan = (plan?: IPlan) => {\n setLoading(true);\n setTimeout(() => {\n setPlan(plan);\n setLoading(false);\n }, 500);\n };\n\n const handleDeselectPlan = () => {\n setPlan(undefined);\n };\n\n return (\n {\n handleDeselectPlan();\n onCancel?.(e);\n }}\n {...rest}\n >\n \n {loading && (\n \n \n \n )}\n {!plan ? (\n (modalPlans || Plans).map((plan) => (\n handleSelectPlan(plan)}\n buyed={\n (plans?.name === plan.type && plans.status === \"active\") ??\n false\n }\n />\n ))\n ) : (\n {}}\n setInitialized={() => {}}\n getCardCurrentStep={(value: number) => {}}\n />\n )}\n \n \n );\n};\n","import { Modal } from \"antd\";\nimport styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 0 32px 128px;\n @media only screen and (max-device-width: 812px) {\n padding: 0 16px 128px;\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n margin: 0 auto;\n max-width: 600px;\n border-radius: 16px;\n background-color: var(--velotax-background-color);\n h1 {\n margin: 0 0 8px;\n font-size: 20px;\n color: var(--velotax-font-color);\n }\n h2 {\n margin: 0;\n font-size: 16px;\n font-weight: 400;\n color: var(--velotax-font-color);\n }\n .ant-list.ant-list-split {\n margin: 24px 0 0;\n padding: 0 16px;\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-ghost);\n border-radius: 8px;\n .ant-list-item-action > li {\n padding: 0;\n button {\n margin: 0 auto;\n }\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n .ant-list-item-action > li,\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color);\n }\n .ant-list-item-action-split {\n opacity: 0;\n }\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px;\n .ant-list.ant-list-split {\n .ant-list-item-action {\n display: grid;\n align-items: end;\n grid-template-rows: 1fr 0.75fr;\n grid-template-columns: 1fr 1fr;\n }\n }\n }\n`;\n\nexport const ConfirmModal = styled(Modal)`\n .p {\n font-size: 16px;\n strong {\n font-weight: 500;\n color: var(--ant-primary-color);\n }\n }\n`;\n","import { Tag } from \"antd\";\nimport styled from \"styled-components\";\n\nexport const Container = styled.div`\n display: flex;\n column-gap: 8px;\n`;\n\nexport const StyledTag = styled(Tag)`\n margin: 0;\n height: 32px;\n border-radius: 0;\n display: flex;\n align-items: center;\n justify-content: center;\n cursor: default;\n font-size: 12px;\n font-weight: 500;\n`;\n","import { Divider, Typography } from \"antd\";\nimport { InfoCircleOutlined } from \"@ant-design/icons\";\n\nexport enum MeiReportDasStatusEnum {\n EMITTED = \"EMITTED\",\n REQUESTED = \"REQUESTED\",\n}\n\nexport enum MeiReportStatusTextEnum {\n EMITTED = \"Emitido\",\n REQUESTED = \"Solicitado\",\n}\n\nexport enum MeiDasStatusTextEnum {\n EMITTED = \"Guia Emitida\",\n REQUESTED = \"Guia Solicitada\",\n}\n\nexport enum MeiDasStatusColorEnum {\n EMITTED = \"var(--ant-success-color)\",\n REQUESTED = \"var(--ant-primary-color)\",\n}\n\nexport const DasHelpModal = {\n title: \"Lista de guias MEI (DAS)\",\n content: (\n \n
\n Confirme manualmente sempre que um DAS for pago com sucesso. Esta\n funcionalidade te ajuda a manter o controle dos seus pagamentos\n realizados.\n \n
\n
\n \n Atenção: a confirmação de pagamento é feita manualmente pelo\n usuário. O Velotax não realiza este monitoramento automaticamente.\n \n
\n ),\n};\n","import { Col, Row, Typography } from \"antd\";\nimport { ExteriorModalContainer } from \"./styles\";\nimport { formatCurrency, replaceDotByComma } from \"../../../utils\";\n\nexport enum ExteriorOperationType {\n TRADE = \"TRADE\",\n DEPOSIT = \"DEPOSIT\",\n WITHDRAW = \"WITHDRAW\",\n}\n\ninterface ExteriorModalContentProps {\n asset: any;\n}\n\nexport const InterestModal = ({ asset }: ExteriorModalContentProps) => {\n return (\n \n \n \n \n {asset.type === \"DIVIDEND\" ? \"Dividendos\" : \"Juros\"} recebidos\n \n \n Valor bruto:\n \n {formatCurrency(asset.valor, \"US$ \")} \n \n Taxa de corretagem:\n \n {formatCurrency(asset.fee, \"US$ \")} \n \n Data de liquidação:\n \n {asset.data} \n \n Cotação dólar:\n \n {replaceDotByComma(asset.cambioVenda)} \n \n \n \n Rendimento total: \n {formatCurrency(asset.rendimentoAplicacao)} \n
\n \n
\n \n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 128px;\n\n .errorCard {\n display: flex;\n flex-direction: column;\n justify-content: center;\n background-color: var(--ant-primary-1);\n height: 150px;\n margin-top: 35px;\n border-radius: 16px;\n border: 1px solid var(--ant-primary-color);\n padding-left: 30px;\n\n .errorTitle {\n color: #000;\n font-weight: 600;\n font-size: 18px;\n }\n\n .errorText {\n color: #000;\n }\n }\n\n h1 {\n width: 700px;\n margin: 0 auto 1.5rem;\n font-weight: 400;\n font-size: 2rem;\n position: relative;\n line-height: 2.5rem;\n padding-bottom: 1rem;\n color: var(--velotax-font-color-light);\n\n :after {\n content: \"\";\n position: absolute;\n left: 0;\n bottom: 0;\n height: 2px;\n width: 100px;\n background-color: var(--ant-primary-color);\n }\n }\n\n h2.ant-typography {\n display: flex;\n align-items: center;\n column-gap: 8px;\n width: 700px;\n margin: 0 auto 1.5rem;\n font-size: 1rem;\n font-weight: 400;\n line-height: 1.5rem;\n color: var(--velotax-font-color-light);\n\n &.info {\n align-items: flex-start;\n font-size: 0.875rem;\n line-height: 1.125rem;\n opacity: 0.85;\n margin: 0 auto 2.5rem;\n svg {\n margin-top: 4px;\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n\n h1 {\n width: calc(100% - 48px);\n margin: 0 24px 1.5rem;\n padding: 24px 0 1rem;\n }\n\n h2.ant-typography {\n width: 100%;\n margin: 0 auto 1.5rem;\n padding: 0 24px;\n }\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n margin: 0 auto;\n width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n h1 {\n margin: 0 0 8px;\n font-size: 20px;\n color: var(--velotax-font-color);\n }\n h2 {\n margin: 0;\n font-size: 16px;\n font-weight: 400;\n color: var(--velotax-font-color);\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px 24px;\n border-radius: 0;\n width: 100%;\n .ant-list.ant-list-split {\n .ant-list-item-action {\n display: grid;\n align-items: end;\n grid-template-rows: 1fr 0.75fr;\n grid-template-columns: 1fr 1fr;\n }\n }\n }\n\n .monthError {\n text-align: center;\n .warning {\n transform: translateY(2px);\n color: var(--ant-warning-color);\n margin-right: 6px;\n font-size: 16px;\n }\n }\n`;\n\nexport const TitleContainer = styled.div`\n width: 100%;\n display: flex;\n align-items: center;\n justify-content: space-between;\n`","import styled from \"styled-components\";\n\nexport const ExteriorModalContainer = styled.div`\n font-size: 1rem;\n color: var(--velotax-font-color-dark);\n\n strong {\n font-weight: 600;\n }\n\n .exterior-capital-gain {\n padding: 8px;\n margin: 24px auto;\n text-align: center;\n background-color: #eee;\n }\n`;\n","import { Col, Row, Typography } from \"antd\";\nimport { ExteriorModalContainer } from \"./styles\";\nimport { formatCurrency, replaceDotByComma } from \"../../../utils\";\n\nexport enum ExteriorOperationType {\n TRADE = \"TRADE\",\n DEPOSIT = \"DEPOSIT\",\n WITHDRAW = \"WITHDRAW\",\n}\n\ninterface ExteriorModalContentProps {\n asset: any;\n}\n\nexport const ExteriorModalContent = ({ asset }: ExteriorModalContentProps) => {\n return (\n \n \n \n Quantidade: {asset.quantidade} \n \n \n \n {asset.type === ExteriorOperationType.TRADE\n ? \"Compra\"\n : asset.type === ExteriorOperationType.DEPOSIT\n ? \"Depósito\"\n : asset.type === ExteriorOperationType.WITHDRAW\n ? \"Resgate\"\n : \"Compra\"}\n \n \n Origem BRA:\n \n {formatCurrency(asset.valueOriginBRA, \"US$ \")} \n \n Origem USA:\n \n {formatCurrency(asset.valueOriginEUA, \"US$ \")} \n \n {asset.type === ExteriorOperationType.TRADE && (\n \n \n Venda\n \n \n )}\n {asset.type === ExteriorOperationType.TRADE && (\n <>\n Valor bruto:\n \n {formatCurrency(asset.valor, \"US$ \")} \n \n >\n )}\n Taxa de corretagem:\n \n {formatCurrency(asset.fee, \"US$ \")} \n \n Data de liquidação:\n \n {asset.data} \n \n Cotação dólar:\n \n {replaceDotByComma(asset.cambioVenda)} \n \n {(asset.type === ExteriorOperationType.DEPOSIT ||\n asset.type === ExteriorOperationType.WITHDRAW) && (\n <>\n Cotação dólar Referência:\n \n {replaceDotByComma(asset.cambioReferencia.toFixed(4))} \n \n >\n )}\n \n \n Ganho de capital: \n {formatCurrency(asset.type === \"INTEREST\" ? asset.ganho2 : asset.ganho)} \n
\n \n
\n \n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n margin: 0 auto;\n width: 700px;\n\n button {\n padding-left: 0px !important;\n padding-right: 0px !important;\n min-width: auto;\n }\n\n &.darf-bolsa,\n &.darf-exterior,\n &.historic-bolsa,\n &.integration-exterior,\n &.historic-exterior {\n margin-top: 24px;\n }\n\n &.dividendos-exterior {\n margin-top: -24px;\n }\n\n @media only screen and (max-device-width: 812px) {\n width: 100%;\n &.darf-bolsa,\n &.report-bolsa,\n &.dividendos-exterior,\n &.report-exterior,\n &.darf-exterior {\n padding: 0 24px;\n }\n }\n`;\n","import { Button } from \"@mui/material\";\nimport { useNavigate } from \"react-router-dom\";\nimport { Container } from \"./styles\";\n\ninterface BackButtonProps {\n to: string;\n className?: string;\n}\n\nexport const BackButton: React.FC = ({ to, className }) => {\n const navigate = useNavigate();\n return (\n \n {\n navigate(to);\n }}\n >\n Voltar\n \n \n );\n};\n","import styled from \"styled-components\";\n\nexport const ExteriorModalContainer = styled.div`\n font-size: 1rem;\n color: var(--velotax-font-color-dark);\n\n strong {\n font-weight: 600;\n }\n\n .exterior-capital-gain {\n padding: 8px;\n margin: 24px auto;\n text-align: center;\n background-color: #eee;\n }\n`;\n","import { Table } from \"antd\";\nimport { IoMdOptions } from \"react-icons/io\";\nimport Button from \"../../../components/Button\";\nimport { monthsExtended } from \"../../../utils\";\nimport { DrawerModal } from \"../../../components/DrawerModal\";\nimport { useB3Integration } from \"../../../contexts/B3IntegrationContext\";\n\nconst initialYear = 2020;\n\nexport const dayFilters: Array = [];\nfor (let d = 1; d <= 31; d++) {\n dayFilters.push({\n text: d.toString(),\n value: d,\n });\n}\n\nconst yearFilters = Array.from(\n Array(new Date().getFullYear() + 1 - initialYear)\n)\n .map((y, i) => ({ value: initialYear + i, text: `${initialYear + i}` }))\n .reverse();\n\ninterface NotasHistoricoModalProps {\n crypto?: boolean;\n}\n\nexport const NotasHistoricoModal: React.FC = ({\n crypto,\n}) => {\n const {\n notas: {\n downloadNotaCorretagem,\n historicoNotasCorretagem,\n getHistoricoNotasCorretagem,\n setHistoricoNotasCorretagem,\n loadingHistoricoNotasCorretagem,\n showHistoricoNotasCorretagemModal,\n setShowHistoricoNotasCorretagemModal,\n },\n } = useB3Integration();\n\n return (\n {\n setShowHistoricoNotasCorretagemModal(false);\n setHistoricoNotasCorretagem([]);\n }}\n >\n {\n getHistoricoNotasCorretagem({ filters, crypto });\n }}\n locale={{\n filterReset: \"Limpar\",\n }}\n columns={[\n {\n align: \"center\",\n title: \"Dia\",\n dataIndex: \"day\",\n filterIcon: ,\n filters: dayFilters,\n render: (status, item) => <>{status}>,\n },\n {\n align: \"center\",\n title: \"Mês\",\n dataIndex: \"month\",\n filterIcon: ,\n filters: monthsExtended.map((m, i) => ({\n text: m,\n value: i + 1,\n })),\n render: (status, item) => <>{status}>,\n },\n {\n align: \"center\",\n title: \"Ano\",\n dataIndex: \"year\",\n filterIcon: ,\n filters: yearFilters,\n render: (status, item) => <>{status}>,\n },\n {\n className: \"actionColumn\",\n render: (status, item) => (\n \n downloadNotaCorretagem(\n item.path,\n item.month,\n item.year,\n crypto\n )\n }\n >\n Ver nota\n \n ),\n },\n ]}\n />\n \n );\n};\n","import clsx from \"clsx\";\nimport moment from \"moment\";\nimport { Button } from \"@mui/material\";\nimport { FaRegEdit } from \"react-icons/fa\";\nimport { useLocation, useNavigate } from \"react-router-dom\";\nimport { BiChevronLeft, BiChevronRight } from \"react-icons/bi\";\nimport { AiOutlineDelete, AiOutlineLock } from \"react-icons/ai\";\nimport { BsCalendar3, BsDownload, BsEye } from \"react-icons/bs\";\nimport { LoadingOutlined, PlusOutlined } from \"@ant-design/icons\";\nimport { useCallback, useEffect, useMemo, useState, ReactNode } from \"react\";\nimport {\n Col,\n Collapse,\n DatePicker,\n Descriptions,\n List,\n message,\n Modal,\n Row,\n Skeleton,\n Space,\n Spin,\n Tooltip,\n Typography,\n} from \"antd\";\nimport { Page } from \"../../constants/brokers\";\nimport AntButton from \"../../components/Button\";\nimport HandleTag from \"../../services/handleTag\";\nimport { Container, Content } from \"../Darf/styles\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport { FormModal } from \"../../components/FormModal\";\nimport { apiPayment } from \"../../services/apiPayment\";\nimport { InterestModal } from \"./InterestModalContent\";\nimport { useBroker } from \"../../contexts/BrokerContext\";\nimport { BackButton } from \"../../components/BackButton\";\nimport apis, { NOT_AUTHORIZED } from \"../../services/apis\";\nimport { PaymentModal } from \"../../components/PaymentModal\";\nimport { ExteriorModalContent } from \"./ExteriorModalContent\";\nimport { IntegrationModal } from \"../Darf/Components/IntegrationModal\";\nimport DeleteConfirmationModal from \"../../components/DeleteConfirmationModal\";\nimport { DatePickerExtraFooter } from \"../Darf/Components/DatePickerExtraFooter\";\nimport {\n DatePickerMonthCell,\n YearResume,\n} from \"../Darf/Components/DatePickerMonthCell\";\nimport {\n antDatePickerLocale,\n download,\n errorMessage,\n formatCurrency,\n isMobile,\n monthsExtended,\n numberToPercentageWallet,\n} from \"../../utils\";\nimport {\n DarfResultDescriptionsExterior,\n ExteriorFormItemRows,\n defaultDarf,\n DarfModal,\n IDarf,\n DarfImpostosDevidosDescriptions,\n minDarfPrice,\n maxDarfPrice,\n YearResumeStatusFromBackEnum,\n IQuotes,\n historicoVendasEmptyText,\n maxPixPayment,\n maxCreditCardPayment,\n yearConfigPL4173,\n ExteriorNewFormItemRows,\n} from \"../../constants/darf\";\nimport { CurrencyFormItem } from \"../../constants/formItems\";\nimport { useDarfExterior } from \"./useDarfExterior\";\n\ninterface DarfProps {\n item: Page;\n view?: boolean;\n viewEdit?: boolean;\n closeModal?: () => void;\n darf?: {\n id: string;\n fullPath?: string;\n };\n date?: {\n year?: number;\n month?: number;\n };\n selectedYear?: number;\n}\n\nconst PrejuFormItems = {\n userPrejuizoExterior: (disabled: boolean) =>\n CurrencyFormItem({\n disabled,\n name: \"prejuizoExterior\",\n label: \"Prejuízo Exterior\",\n }),\n};\n\nexport const PrejuFormItemRows = (disabled: boolean) => [\n [PrejuFormItems.userPrejuizoExterior(disabled)],\n];\n\nexport const DarfExterior: React.FC = ({\n view,\n item,\n date,\n darf,\n viewEdit,\n closeModal,\n selectedYear\n}) => {\n const { state } = useLocation();\n const { currentBroker, currentPage } = useBroker();\n const [openPrejuizoModal, setOpenPrejuizoModal] = useState(false);\n const navigate = useNavigate()\n const { user, hasPlan: hasPremiunPlan, hasPermissionExterior, hasPermissionGeneral } = useAuth();\n const { getBuyPrice, getFee, getGain, getSellPrice } = useDarfExterior()\n\n const today = new Date();\n const queryYear = (state as any)?.year as number;\n const queryMonth = (state as any)?.month as number;\n const currentMonth = today.getMonth();\n const currentYear = today.getFullYear();\n const initialMonth = queryMonth\n ? queryMonth - 1\n : today.getMonth() - 1 >= 0\n ? today.getMonth() - 1\n : 11;\n const initialYear = queryYear\n ? queryYear\n : today.getMonth() - 1 >= 0\n ? today.getFullYear()\n : today.getFullYear() - 1;\n const defaultValue = moment({ month: initialMonth, year: initialYear });\n const currentDate = moment({\n month: currentMonth,\n year: currentYear,\n }).subtract(1, \"M\");\n window.history.replaceState({}, document.title);\n\n const year = date?.year ?? initialYear;\n const month = date?.month ?? initialMonth;\n const oldModel = year < yearConfigPL4173;\n const currentDarfDate = new Date(year, month, 15)\n const dateAllowedOnBasic = new Date(today.getFullYear(), today.getMonth() - 1, 1)\n const hideOnBasic = (currentDarfDate.getTime() < dateAllowedOnBasic.getTime() && user.user?.userPlanInfoVelotax?.type?.includes('BASIC'))\n const hasPlan = (hasPremiunPlan || hasPermissionExterior || hasPermissionGeneral) && !hideOnBasic;\n\n const [asset, setAsset] = useState();\n const [loading, setLoading] = useState(false);\n const [, setYear] = useState(initialYear);\n const [, setMonth] = useState(initialMonth);\n const [emitting, setEmitting] = useState(false);\n const [darfModal, setDarfModal] = useState();\n const [helpModal, setHelpModal] = useState();\n const [data, setData] = useState(defaultDarf);\n const [paymentModal, setPaymentModal] = useState(false);\n const [paymentData, setPaymentData] = useState();\n const [quotations, setQuotations] = useState({});\n const [notAuthorized, setNotAuthorized] = useState(false);\n const [movimentType, setMovimentType] = useState(\"Venda de ativos\");\n const [showAssetModal, setShowAssetModal] = useState(false);\n const [showDeleteModal, setShowDeleteModal] = useState(false);\n const [yearResume, setYearResume] = useState([]);\n const [datePickerYear, setDatePickerYear] = useState(initialYear);\n const [gettingYearResume, setGettingYearResume] = useState(false);\n const [loadingDarfButton, setLoadingDarfButton] = useState(false);\n const [showNotAuthorizedModal, setShowNotAuthorizedModal] = useState(false);\n\n const transactionFeature = item.features[0];\n const taxFeature = item.features[1];\n const yearResumeFeature = item.features[2];\n\n const memoriaCalculoFiltered = data.memoriaCalculo.filter(\n (el: any) =>\n (el.operation === \"SELL\" &&\n (el.type === \"TRADE\" || (el.type === \"WITHDRAW\" || el.type === \"WITHDRAWAL\"))) ||\n el.type === \"INTEREST\" ||\n (el.type === \"DIVIDEND\" && !oldModel)\n );\n\n const { emitted } = useMemo(() => {\n if (gettingYearResume) {\n }\n const monthResume = yearResume\n .find((resume) => resume.year === year)\n ?.months.find((m) => m.month - 1 === month);\n return {\n emitted:\n monthResume?.status === YearResumeStatusFromBackEnum.PAYED ||\n monthResume?.status === YearResumeStatusFromBackEnum.NOT_PAYED,\n payed: !!monthResume?.payed,\n status: monthResume?.status,\n };\n }, [yearResume, year, month, gettingYearResume]);\n\n const getDarf = useCallback(\n (id: string) => {\n view && setLoading(true);\n view &&\n (currentPage?.api || apis)\n .get(`${taxFeature.apiUrl}/${id}`)\n .then((response) => {\n setYear(response.data?.year);\n setMonth(response.data?.month - 1);\n setData({\n ...response.data,\n emitted: true,\n });\n })\n .catch(() => message.error(errorMessage))\n .finally(() => setLoading(false));\n },\n [view, taxFeature.apiUrl, currentPage]\n );\n\n const getYearResume = useCallback(\n (year: number, force?: boolean, url?: string) => {\n setGettingYearResume(true);\n !view &&\n year <= new Date().getFullYear() &&\n year >= currentBroker.initialYear &&\n (force ||\n (!yearResumeFeature.disabled &&\n !yearResume.find((resume) => resume.year === year))) &&\n (currentPage?.api || apis)\n .get(yearResumeFeature.apiUrl, { params: { year } })\n .then((response) => {\n const index = yearResume.findIndex(\n (resume) => resume.year === year\n );\n if (index >= 0) {\n setYearResume((yearResume) => {\n yearResume.splice(index, 1, response.data);\n return yearResume;\n });\n } else {\n setYearResume((yearResume) => [...yearResume, response.data]);\n }\n\n if (url && !isMobile()) {\n download(url);\n }\n })\n .catch((err) => console.log(err))\n .finally(() => setGettingYearResume(false));\n },\n [\n yearResumeFeature,\n yearResume,\n view,\n currentBroker.initialYear,\n currentPage,\n ]\n );\n\n const getTaxes = useCallback(() => {\n (!hasPlan || !view) && setLoading(true);\n (!hasPlan || !view) &&\n (currentPage?.api || apis)\n .get(transactionFeature.apiUrl, { params: { month: month + 1, year, atualizouImpostoAcumulado: true } })\n .then((response) => {\n setData(response.data);\n setNotAuthorized(false);\n })\n .catch((err) => {\n setData(defaultDarf);\n if (err?.response?.data === NOT_AUTHORIZED) {\n setNotAuthorized(true);\n setShowNotAuthorizedModal(true);\n } else {\n setNotAuthorized(false);\n message.error(errorMessage);\n }\n })\n .finally(() => setLoading(false));\n }, [hasPlan, view, currentPage?.api, transactionFeature.apiUrl, month, year]);\n\n const getQuotations = useCallback(() => {\n !view &&\n (currentPage?.api || apis)\n .get(\"/warren/get-cotacoes\")\n .then((response) => {\n const { value: cotacoes } = response.data;\n const pojo = {};\n for (let cotacao of cotacoes) {\n const {\n dataHoraCotacao,\n cotacaoCompra: compra,\n cotacaoVenda: venda,\n } = cotacao;\n const [year, month, day] = String(dataHoraCotacao)\n .split(\"T\")[0]\n .split(\"-\");\n Object.assign(pojo, {\n [`${day}/${month}/${year}`]: {\n compra,\n venda,\n },\n });\n }\n setQuotations(pojo);\n })\n .catch((err) => {\n setData(defaultDarf);\n if (err?.response?.data === NOT_AUTHORIZED) {\n setNotAuthorized(true);\n setShowNotAuthorizedModal(true);\n } else {\n setNotAuthorized(false);\n message.error(errorMessage);\n }\n });\n // eslint-disable-next-line\n }, [view]);\n\n const handleDarf = (justSave?: boolean) => {\n setEmitting(true);\n onCloseDarfModal();\n (currentPage?.api || apis)\n .post(taxFeature.apiUrl, {\n ...data,\n year,\n month: month + 1,\n name: user.user.name,\n memoriaCalculo: [],\n transactions: data.memoriaCalculo,\n paymentInfo: {\n clientInfo: {\n name: user?.user?.name,\n email: user?.user?.email,\n document: user?.user?.cpf,\n },\n },\n insertPix: true,\n justSave,\n payed: false,\n })\n .then((res) => {\n setData((data) => ({ ...data, emitted: true }));\n getYearResume(year, true, justSave ? \"\" : res.data.darfUrl);\n setEmitting(false);\n closeModal?.();\n if (!justSave) {\n oldModel\n ? message.success(\"O DARF foi enviado para o seu e-mail\")\n : message.success(\"O Relatório foi enviado para o seu e-mail\"); \n }\n })\n .catch((err) => {\n message.error(err?.response?.data?.message || errorMessage);\n setEmitting(false);\n });\n };\n\n const handleRegularize = (regular: boolean) => {\n setEmitting(true);\n (currentPage?.api || apis)\n .post(\"transaction/regularize\", {\n year,\n month: month + 1,\n regular,\n impostoDevido: data.impostoDevido,\n totalImpostoDevido: data.totalImpostoDevido,\n })\n .then((res) => {\n getYearResume(year, true, res.data.darfUrl);\n getTaxes();\n closeModal?.();\n message.success(\"Dados salvos\");\n })\n .catch(() => message.error(errorMessage))\n .finally(() => setEmitting(false));\n };\n\n const onChangeMonth = (value: moment.Moment | null, dateString: string) => {\n const [month, year] = dateString.split(\"-\");\n setMonth(parseInt(month) - 1);\n setYear(parseInt(year));\n };\n\n const unsetAsset = () => {\n setAsset(undefined);\n };\n\n const handleCloseDeleteModal = () => {\n setShowDeleteModal(false);\n unsetAsset();\n };\n\n const handleCloseAssetModal = () => {\n setMovimentType(\"Venda de ativos\");\n setShowAssetModal(false);\n unsetAsset();\n };\n\n const onCloseHelpModal = () => {\n setHelpModal(undefined);\n };\n\n const onCloseDarfModal = () => {\n setDarfModal(undefined);\n };\n\n const handleOpenDarfModal = () => {\n setDarfModal(DarfModal(emitted, onCloseDarfModal, () => handleDarf()));\n };\n\n const handleOpenPaymentModal = () => {\n setPaymentModal(!paymentModal);\n };\n\n const handleAdd = (event: React.MouseEvent) => {\n event.stopPropagation();\n setShowAssetModal(true);\n };\n\n const handleEdit = (item: any, index: number) => {\n setShowAssetModal(true);\n const data = moment.utc(item.data.toLocaleString()).format(\"DD/MM/YYYY\");\n let i = 0;\n let key = data;\n while (!quotations[key] && i <= 7) {\n key = moment(key, \"DD/MM/YYYY\").subtract(1, \"days\").format(\"DD/MM/YYYY\");\n i++;\n }\n\n setAsset({\n ...item,\n data,\n cotacaoVenda: quotations[key]?.venda,\n cotacaoCompra: quotations[key]?.compra,\n dataVenda: moment\n .utc(item?.dataVenda?.toLocaleString())\n ?.format(\"DD/MM/YYYY\"),\n dataCompra: moment\n .utc(item?.dataCompra?.toLocaleString())\n ?.format(\"DD/MM/YYYY\"),\n });\n };\n\n const handleRemove = (item: any, index: number) => {\n setShowDeleteModal(true);\n setAsset(item);\n };\n\n const addAsset = (asset: any) => {\n setLoading(true);\n\n const ganho = getGain(asset, oldModel)\n const sellPrice = getSellPrice(asset, oldModel)\n const buyPrice = getBuyPrice(asset, oldModel)\n const fee = getFee(asset, oldModel)\n\n const dataToCreate = {\n ...{[sellPrice.key]: sellPrice.value},\n ...{[fee.key]: fee.value},\n ...{[ganho.key]: ganho.value},\n ...{[buyPrice.key]: buyPrice.value},\n data: asset.data, // date liquidação venda\n dataCompra: asset.dataCompra, // pegar a cotação\n operation: \"SELL\", // fixo\n type: \"TRADE\", // fixo\n ativo: asset.ativo, // Tiker que o usuário colocar (Ativo)\n year,\n month,\n quantidade: asset.quantidade,\n // \"saldoBRA\": 10000,\n // \"saldoEUA\": 870.7999999999993,\n // \"capitalGainBRA_BRL\": 0, (colocar no ganho)\n // \"capitalGainEUA_BRL\": 0,\n // ganho: asset.ganho, // (capitalGainBRA_BRL)\n cambioVenda: asset.cambioVenda, // (C) o cambio venda na pagina 1 ta na venda\n cambioCompra: asset.cambioCompra, // (C) o cambio venda na pagina 1 ta na venda\n // \"cambioReferencia\": 0,\n valueOriginEUA: asset.valueOriginEUA, // Origem USA (US$) ??\n movimentType: asset.movimentType\n };\n\n const dataFeeToCreate = {\n ...{[ganho.key]: ganho.value},\n // ganho: asset.ganhoFee,\n ...{[sellPrice.key]: sellPrice.value},\n valor: asset.valor, // (Valor Bruto (US$))\n ...{[fee.key]: fee.value}, // TODO: VALIDATE THIS\n fee: asset.fee, // (Taxa de Corretagem (US$)) \n //\n ativo: asset.ativo, // Tiker que o usuário colocar (Ativo)\n data: asset.dataFee,\n dataCompra: asset.dataFee, // date liquidação venda\n operation: \"BUY\", // fixo\n type: asset.movimentType === 'Dividendos' ? \"DIVIDEND\" : \"INTEREST\", // fixo\n year,\n month,\n rendimentoAplicacao: asset.ganhoFee, // (capitalGainBRA_BRL)\n cambioVenda: asset.cambioVendaFee, // (C) o cambio venda na pagina 1 ta na venda\n movimentType: asset.movimentType\n };\n \n (currentPage?.api || apis)\n .post(\n `${transactionFeature.apiUrl}?manual=true`,\n asset.movimentType !== \"Venda de ativos\"\n ? dataFeeToCreate\n : dataToCreate,\n {\n headers: {\n \"x-email\": user.user.email,\n },\n }\n )\n .then(() => {\n getTaxes();\n getYearResume(year, true);\n })\n .catch(() => message.error(errorMessage))\n .finally(() => {\n setLoading(false);\n handleCloseAssetModal();\n });\n };\n\n const editAsset = (data: any) => {\n const ganho = getGain(data);\n const sellPrice = getSellPrice(data, oldModel)\n const buyPrice = getBuyPrice(data, oldModel)\n const fee = getFee(data, oldModel)\n // ...{[sellPrice.key]: sellPrice.value},\n // ...{[fee.key]: fee.value},\n // ...{[ganho.key]: ganho.value},\n // ...{[buyPrice.key]: buyPrice.value},\n const dataToEdit = {\n ...{[ganho.key]: ganho.value}, // (capitalGainBRA_BRL)\n ...{[fee.key]: fee.value},\n fee: data.fee, // (Taxa de Corretagem (US$))\n ...{[sellPrice.key]: sellPrice.value},\n valor: data.valor, // (Valor Bruto (US$))\n ...{[buyPrice.key]: buyPrice.value},\n valueOriginBRA: data.valueOriginBRA, // Origem BRA (US$) ??\n valueOriginEUA: data.valueOriginEUA, // Origem USA (US$) ??\n //\n data: data.data, // date liquidação venda\n dataCompra: data.dataCompra, // pegar a cotação\n operation: \"SELL\", // fixo\n type: \"TRADE\", // fixo\n ativo: data.ativo, // Tiker que o usuário colocar (Ativo)\n quantidade: data.quantidade,\n // \"saldoBRA\": 10000,\n // \"saldoEUA\": 870.7999999999993,\n // \"capitalGainBRA_BRL\": 0, (colocar no ganho)\n // \"capitalGainEUA_BRL\": 0,\n cambioVenda: data.cambioVenda, // (C) o cambio venda na pagina 1 ta na venda\n cambioCompra: data.cambioCompra, // (C) o cambio venda na pagina 1 ta na venda\n // \"cambioReferencia\": 0,\n };\n setLoading(true);\n (currentPage?.api || apis)\n .put(`${transactionFeature.apiUrl}/${asset._id}`, dataToEdit)\n .then(() => {\n getTaxes();\n getYearResume(year, true);\n })\n .catch(() => message.error(errorMessage))\n .finally(() => {\n setLoading(false);\n handleCloseAssetModal();\n });\n };\n\n const removeAsset = () => {\n setLoading(true);\n (currentPage?.api || apis)\n .delete(`${transactionFeature.apiUrl}/${asset._id}`)\n .then(() => {\n getTaxes();\n getYearResume(year, true);\n })\n .catch(() => message.error(errorMessage))\n .finally(() => {\n setLoading(false);\n handleCloseDeleteModal();\n });\n };\n\n const handlePlanModal = () => {\n navigate('/planos')\n };\n\n const handleSaveMonth = async () => {\n setLoadingDarfButton(true);\n try {\n const retPlanInfo = await apiPayment.get(\"/user-plan/plan-info\");\n const hasExteriorCalcActive = retPlanInfo?.data?.permissions?.[\"exterior-warren\"];\n\n retPlanInfo.data.active || hasExteriorCalcActive || hasPermissionGeneral\n ? handleDarf(true)\n : handlePlanModal();\n } catch (err) {\n console.error(err);\n }\n setLoadingDarfButton(false);\n };\n\n const handleEventDarfButton = async () => {\n setLoadingDarfButton(true);\n let retPlanInfo = {\n data: {\n active: false,\n permissions: {\n \"exterior-warren\": false,\n },\n },\n };\n try {\n retPlanInfo = await apiPayment.get(\"/user-plan/plan-info\");\n } catch (err) {\n console.error(err);\n }\n setLoadingDarfButton(false);\n\n const hasntPlan = !retPlanInfo.data.active;\n const hasExteriorCalcActive = retPlanInfo.data.permissions[\"exterior-warren\"];\n\n //@ts-ignore\n hasntPlan && !hasExteriorCalcActive && !retPlanInfo.data.cupomDueDate\n ? handlePlanModal()\n : handleOpenPaymentModal();\n };\n\n const handleLabelButton: ReactNode = useMemo(() => {\n if(!oldModel) return \"Emitir relatório\"\n else if (data?.totalImpostoDevido <= maxDarfPrice) return \"Pagar DARF\";\n else {\n if (emitted) return \"Emitir novo DARF\";\n else return taxFeature?.buttonLabel;\n }\n }, [data?.totalImpostoDevido, emitted, oldModel, taxFeature?.buttonLabel]);\n\n const downloadAction = (record: any) => {\n (currentPage?.api || apis)\n .get(`/darf/download/${record._id}`)\n .then((res) => {\n if (res.data.url) {\n download(res.data.url);\n }\n });\n };\n\n const calcReturn = (\n sellPrice: number,\n buyPrice: number,\n quantity: number,\n fee: number,\n ganho?: number\n ) => {\n let ret = 0\n if(ganho) {\n const qtt = quantity ? quantity : 1;\n ret =\n sellPrice && buyPrice\n ? (sellPrice * qtt - (fee || 0)) / (buyPrice * qtt) - 1\n : 0;\n }\n return (\n = 0\n ? \"var(--velotax-green-amount)\"\n : \"var(--velotax-red-prejuizo)\"\n }`,\n }}\n >\n {ret ? numberToPercentageWallet(ret) : \"-\"}\n \n );\n };\n\n useEffect(() => {\n getTaxes();\n }, [getTaxes, month, year]);\n\n useEffect(() => {\n getQuotations();\n }, [getQuotations, month, year, item]);\n\n useEffect(() => {\n if (view && darf) {\n getDarf(darf.id);\n }\n }, [getDarf, view, darf]);\n\n useEffect(() => {\n getYearResume(year);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n useEffect(() => {\n const dataToPayment = {\n ...data,\n year,\n month: month + 1,\n name: user.user.name,\n memoriaCalculo: [],\n transactions: data.memoriaCalculo,\n };\n\n setPaymentData(dataToPayment);\n }, [data.totalImpostoDevido, data, month, user.user.name, year]);\n\n const impostoCharge = useMemo(() => {\n // Para verificar se é janeiro (month == 0) e antes de 2024\n const taxes = (oldModel || month == 0)\n ? ((data.impostoDevido ?? 0) + (data.impostoAcumulado ?? 0)) - (data.irrfExterior ?? 0) \n : (data.impostoDevido ?? 0) - (data.irrfExterior ?? 0);\n const isExceedTax = taxes < minDarfPrice;\n const additionalTax = oldModel ? data.multa + data.juros : 0;\n return !view\n ? taxes +\n (isExceedTax ? 0 : additionalTax) \n : Math.max(\n taxes +\n (isExceedTax ? 0 : additionalTax) ,\n 0\n );\n }, [data, view, oldModel]);\n\n const verifyFromApi = (item = {} as any) => {\n return (Object.values(item).includes(item.saldoBRA) \n || Object.values(item).includes(item.rendimentoAplicacao)) && oldModel\n }\n\n return (\n \n \n \n \n {selectedYear && selectedYear <= 2023 && \n \n \n {!view && !viewEdit && (\n <>\n {currentBroker.initialYear === year && month === 0 ? (\n
\n ) : (\n {\n if (month === 0) {\n setYear((year) => {\n const newYear = year - 1;\n getYearResume(newYear);\n setDatePickerYear(newYear);\n return newYear;\n });\n setMonth(11);\n } else {\n setMonth((month) => month - 1);\n }\n }}\n />\n )}\n >\n )}\n {monthsExtended[month]} de {year}\n {!view && !viewEdit && (\n <>\n {currentYear === year && currentMonth - 1 === month ? (\n
\n ) : (\n {\n if (month === 11) {\n setYear((year) => {\n const newYear = year + 1;\n getYearResume(newYear);\n setDatePickerYear(newYear);\n return newYear;\n });\n setMonth(0);\n } else {\n setMonth((month) => month + 1);\n }\n }}\n />\n )}\n >\n )}\n \n }\n {!view && !viewEdit && (\n \n }\n defaultValue={defaultValue}\n locale={antDatePickerLocale}\n value={moment({ year: datePickerYear, month })}\n onSelect={(e) => {\n if (e.month() === month && year !== datePickerYear) {\n onChangeMonth(e, `${month + 1}-${datePickerYear}`);\n }\n }}\n onPanelChange={(e) => {\n getYearResume(e.year());\n setDatePickerYear(e.year());\n }}\n getPopupContainer={(trigger) => trigger.parentElement!}\n renderExtraFooter={() =>\n yearResumeFeature.disabled ? (\n <>>\n ) : (\n \n )\n }\n disabledDate={(current) =>\n current &&\n (current > currentDate.endOf(\"month\") ||\n current < moment(`${currentBroker.initialYear}-01-01`) ||\n current < moment(`2020-01-01`))\n }\n monthCellRender={(e) => (\n \n )}\n />\n \n )}\n
\n \n \n\n \n \n {hasPlan && view && darf?.fullPath && (\n \n \n downloadAction(darf)}\n startIcon={\n \n }\n >\n BAIXAR DARF\n \n \n
\n )}\n\n {selectedYear && selectedYear <= 2023 && \n {(!hasPlan || hideOnBasic) && (\n
\n
\n Contrate o plano premium para ver os impostos devidos\n \n
}\n onClick={() => {\n handlePlanModal();\n closeModal?.();\n }}\n >\n PREMIUM\n \n
\n )}\n
\n {DarfImpostosDevidosDescriptions.map((description) => {\n if (description.id === 'prejuizoExterior' && oldModel) return null\n\n return (\n {setOpenPrejuizoModal(true)}\n )}\n >\n {loading ? (\n \n ) : description.render ? (\n description.render(data, view, year)\n ) : (\n formatCurrency(\n Number(data[description.id as keyof IDarf] || 0)\n )\n )}\n \n )\n })}\n \n
}\n\n \n \n \n { selectedYear && selectedYear <= 2023?' Vendas realizadas no mês': 'Movimentações'}\n \n }\n >\n {!notAuthorized && !view && (\n }\n >\n Adicionar\n \n )}\n (\n handleEdit(item, index)}\n icon={\n verifyFromApi(item) || (item.type !== 'TRADE' && !oldModel) ? (\n \n ) : (\n \n )\n }\n />,\n handleRemove(item, index)}\n icon={ }\n />,\n ]\n }\n >\n {oldModel ? \n {item.ativoTranslated ? item.ativoTranslated : item.ativo}\n }\n description={\n \n
\n Data de liquidação:{\" \"}\n {moment\n .utc(item?.data?.toLocaleString())\n .format(\"DD/MM/YYYY\")}\n
\n
\n {item.type !== \"INTEREST\"\n ? \"Ganho\"\n : \"Rendimento total\"}\n :{\" \"}\n \n {formatCurrency(\n item.type === \"INTEREST\"\n ? item.rendimentoAplicacao\n : getGain(item, oldModel).value\n )}\n \n
\n {view && (\n <>\n
\n {item.type !== \"INTEREST\"\n ? \"Valor de venda\"\n : \"Valor bruto\"}\n :{\" \"}\n \n {formatCurrency(getSellPrice(item).value, \"US$ \")}\n \n
\n
\n Taxa de corretagem:{\" \"}\n \n {formatCurrency(getFee(item).value, \"US$ \")}\n \n
\n
\n Cotação dólar:{\" \"}\n {item.cambioVenda} \n
\n {item.type !== \"INTEREST\" && (\n <>\n
\n Valor de compra (Origem BRA):{\" \"}\n \n {formatCurrency(\n item.valueOriginBRA,\n \"US$ \"\n )}\n \n
\n
\n Valor de compra (Origem EUA):{\" \"}\n \n {formatCurrency(\n item.valueOriginEUA,\n \"US$ \"\n )}\n \n
\n >\n )}\n >\n )}\n
\n }\n />\n : \n {item.ativoTranslated ? item.ativoTranslated : item.ativo}\n }\n description={\n \n
\n Data:{\" \"}\n {moment\n .utc(item?.data?.toLocaleString())\n .format(\"DD/MM/YYYY\")}\n
\n {item.type === \"TRADE\" &&
Quantidade: {item.quantidade}
}\n
\n {item.type === \"TRADE\"\n ? \"Ganho\"\n : \"Rendimento total\"}\n :{\" \"}\n \n {formatCurrency(\n item.type !== \"TRADE\"\n ? item.rendimentoAplicacao\n : getGain(item, oldModel).value\n )}\n \n
\n {item.type === 'TRADE' &&
\n Retorno: {calcReturn(\n getSellPrice(item, oldModel).value,\n getBuyPrice(item, oldModel).value,\n item.quantidade,\n getFee(item, oldModel).value,\n getGain(item, oldModel).value\n )}\n
}\n
\n }\n />\n }\n \n )}\n />\n
\n \n\n {hasPlan && }\n\n {hasPlan && selectedYear && selectedYear <= 2023 && (\n \n Imposto total \n \n {loading ? (\n \n ) : (\n formatCurrency(Math.max(impostoCharge, 0))\n )}\n \n
\n )}\n\n {hasPlan && selectedYear && selectedYear <= 2023 && (\n \n {DarfResultDescriptionsExterior.map((description, i) => {\n if(['multa', 'juros'].includes(description.id) && !oldModel) return <>>\n if(['impostoAcumulado'].includes(description.id) && !oldModel && month !== 0) return <>>\n return (\n \n {loading ? (\n \n ) : description.Component ? (\n \n ) : (\n formatCurrency(\n Number(data[description.id as keyof IDarf] || 0)\n )\n )}\n \n )\n })}\n \n )}\n\n {!notAuthorized && !view && (\n <>\n \n {!(month === currentMonth && year === currentYear) &&\n oldModel &&\n (data.impostoDevido + data.impostoAcumulado - (data?.irrfExterior ?? 0) ?? 0) <\n minDarfPrice &&\n memoriaCalculoFiltered?.length > 0 && (\n \n \n \n O valor do imposto devido é menor que{\" \"}\n {formatCurrency(minDarfPrice)}. Você não precisa\n emitir um DARF para este mês.\n >\n }\n >\n handleRegularize(true)}\n startIcon={emitting && }\n >\n Salvar sem emitir DARF\n \n \n \n )}\n {(((data.impostoDevido + data.impostoAcumulado - (data?.irrfExterior ?? 0)) >= minDarfPrice) || (!oldModel && memoriaCalculoFiltered.length > 0 )) && (\n <>{selectedYear && selectedYear <= 2023 && \n \n target.parentElement!}\n title={\n month === currentMonth && year === currentYear && oldModel\n ? \"Aguarde o encerramento do mês para pagar seu DARF\"\n : \"\"\n }\n >\n {\n oldModel \n ? handleEventDarfButton()\n : handleDarf()\n }}\n startIcon={\n (emitting || loadingDarfButton) && \n }\n disabled={\n loading ||\n emitting ||\n loadingDarfButton ||\n (month === currentMonth && year === currentYear && oldModel)\n }\n >\n {handleLabelButton}\n {\" \"}\n \n }>\n )}\n
\n {/* {!oldModel && ( handleDarf(false)} style={{color: \"red\"}}>TESTE )} */}\n {(!(month === currentMonth && year === currentYear) && oldModel) && (\n \n \n }\n >\n Salvar apenas\n \n \n
\n )}\n >\n )}\n\n {!view && (\n <>\n setShowNotAuthorizedModal(false)}\n />\n \n
\n
\n (1) Os cálculos de imposto de renda foram realizados levando\n em consideração somente as operações de compra e venda\n realizadas na Warren Internacional. Caso você tenha realizado\n operações com ativos no exterior em outras corretoras,\n realize as alterações manualmente.\n
\n
\n {selectedYear && selectedYear <= 2023 &&
\n
\n (2) Caso você tenha realizado a transferência de custódia de\n outras corretoras de ativos no exterior para a Warren\n Internacional, realize manualmente a inserção do custo médio\n de compra dos respectivos ativos transferidos.\n
\n
}\n {/*
\n
\n (3) O Usuário é exclusivamente responsável pela conferência e\n validação das informações utilizadas na apuração do imposto\n devido, conforme{\" \"}\n \n Termos e Condições Gerais de Uso.\n \n
\n
*/}\n
\n >\n )}\n {!view && !viewEdit && (\n \n )}\n \n \n \n {helpModal?.content}\n \n \n {darfModal?.content}\n \n \n ) : (\n Object.values(asset).includes(asset.saldoBRA) && oldModel && (\n \n )\n )\n }\n />\n {/* {setOpenPrejuizoModal(false)}} \n onCancel={() => {setOpenPrejuizoModal(false)}}>\n Prejuízo a compensar \n \n */}\n {}}\n edit={() => {}}\n onCancel={() => {setOpenPrejuizoModal(false)}}\n itemToEdit={{}}\n rows={PrejuFormItemRows(false)}\n />\n maxPixPayment}\n callDarf={() => {\n HandleTag(\"67\");\n handleOpenDarfModal();\n }}\n disableCreditCardOption={\n data?.totalImpostoDevido > maxCreditCardPayment\n }\n paymentData={{\n ...paymentData,\n impostoTotalFinal: data.totalImpostoDevido + data.impostoAcumulado,\n }}\n />\n Promise.resolve(removeAsset())}\n body=\"Você realmente quer deletar esse ativo?\"\n />\n \n );\n};\n","// Retorno: {calcReturn(item.valor ?? item.sellPrice_BRL, item.valueOriginBRA ?? item.buyPrice_BRL, item.quantidade, item.fee ?? item.fee_BRL, item.ganho ?? item.varTotalBRL)}\n\nexport function useDarfExterior() {\n const gainKeys = ['ganho', 'varTotalBRL'];\n const sellPriceKeys = ['valor', 'sellPrice_BRL'];\n const buyPriceKeys = ['valueOriginBRA', 'buyPrice_BRL'];\n const feeKeys = ['fee', 'fee_BRL'];\n\n function getByKey(item: any, keys: string[], isOldModel?: boolean) {\n if (isOldModel !== undefined) {\n if (isOldModel) {\n return { key: keys[0], value: item[keys[0]] ?? null };\n }\n return { key: keys[1], value: item[keys[1]] ?? null };\n }\n const k = keys.find((key) => item[key] !== null && item[key] !== undefined);\n const value = k ? item[k] : null;\n const key = k ?? keys[0];\n return { key, value };\n }\n\n function getGain(item: any, isOldModel?: boolean) {\n return getByKey(item, gainKeys, isOldModel)\n }\n\n function getSellPrice(item: any, isOldModel?: boolean) {\n return getByKey(item, sellPriceKeys, isOldModel)\n }\n\n function getBuyPrice(item: any, isOldModel?: boolean) {\n return getByKey(item, buyPriceKeys, isOldModel)\n }\n\n function getFee(item: any, isOldModel?: boolean) {\n return getByKey(item, feeKeys, isOldModel)\n }\n\n return { getBuyPrice, getGain, getSellPrice, getFee }\n}","import styled from \"styled-components\";\n\nexport const CardContainer = styled.div`\n width: 100%;\n height: 64px;\n cursor: pointer;\n position: relative;\n background-color: transparent;\n border-bottom: 1px solid var(--velotax-border-color);\n\n &,\n a,\n .MuiButtonBase-root {\n display: flex;\n align-items: center;\n justify-content: space-between;\n\n & > svg {\n width: 20px;\n height: 20px;\n stroke: var(--velotax-font-color-light);\n }\n }\n\n a,\n .MuiButtonBase-root {\n width: 100%;\n height: 100%;\n padding: 12px 12px 12px 0;\n }\n\n figure {\n margin: 0 16px 0 0;\n\n svg {\n width: 24px;\n height: 24px;\n &:not(.no-fill) {\n fill: var(--ant-primary-color);\n }\n }\n }\n\n .content {\n flex-grow: 1;\n display: flex;\n align-items: center;\n font-weight: 400;\n font-size: 1rem;\n line-height: 1.125rem;\n color: var(--velotax-font-color-light);\n\n .ant-typography {\n color: inherit;\n position: relative;\n text-align: left;\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n &.not-free {\n .content {\n .content-children {\n width: 100%;\n .ant-row {\n width: 100%;\n .ant-col {\n width: 100%;\n .ant-typography {\n width: calc(100% - 64px);\n }\n }\n }\n }\n }\n }\n }\n`;\n","import { ReactNode } from \"react\";\nimport { ButtonBase } from \"@mui/material\";\nimport { useNavigate } from \"react-router-dom\";\nimport { HiOutlineChevronRight } from \"react-icons/hi\";\nimport { isMobile } from \"../../utils\";\nimport { CardContainer } from \"./style\";\n\ninterface CardXPProps {\n id: string;\n icon: ReactNode;\n children: ReactNode;\n link?: string;\n className?: string;\n disabled?: boolean;\n onClick?: () => void;\n}\n\nconst Container: React.FC> = ({ children, link }) => {\n return link && link.includes(\"http\") ? (\n \n {children}\n \n ) : (\n <>{children}>\n );\n};\n\nexport const CardXP: React.FC = ({\n id,\n icon,\n link,\n onClick,\n children,\n disabled,\n className,\n}) => {\n const navigate = useNavigate();\n\n const handleClick = () => {\n if (disabled) return;\n if (onClick) {\n onClick();\n } else if (link && !link.includes(\"http\")) {\n navigate(link);\n }\n };\n\n return (\n \n \n \n \n \n \n \n \n );\n};\n","import { Tooltip } from \"antd\";\nimport { Button } from \"@mui/material\";\n\nexport const PayButton: React.FC<{\n fullWidth?: boolean;\n onClick?: () => void;\n disabled?: boolean;\n title?: React.ReactNode;\n isReport?: boolean\n}> = ({ onClick, fullWidth, disabled, title, isReport }) => (\n target.parentElement!}>\n \n {isReport ? 'Emitir relatório' : 'Pagar'}\n {\" \"}\n \n);\n","import { Typography } from \"antd\";\nimport { GiProfit } from \"react-icons/gi\";\nimport { AiOutlineApi } from \"react-icons/ai\";\nimport { BiTransferAlt } from \"react-icons/bi\";\nimport { RiFileList3Line } from \"react-icons/ri\";\nimport { HiOutlineCalculator } from \"react-icons/hi\";\nimport { isMobile } from \"../../utils\";\n\nexport const itensExterior = [\n {\n id: \"darf\",\n premium: true,\n showToBasic: false,\n link: \"/warren/exterior-historic\",\n content: Calculadora de DARF ,\n description: Cálculo de IR e mensal para o ano de 2023 ,\n icon: (\n \n ),\n },\n // {\n // premium: true,\n // id: \"anual-report\",\n // link: \"/warren/exterior-report\",\n // content: Informe de rendimentos ({YearReport}) ,\n // description: Relatório para declaração do IRPF ,\n // icon: (\n // \n // ),\n // },\n {\n id: \"report\",\n basic: true,\n premium: true,\n link: \"/warren/exterior-report\",\n content: (\n \n Informe de rendimentos\n \n ),\n description: (\n Relatórios auxiliares para declaração de IR \n ),\n icon: (\n \n ),\n },\n // {\n // id: \"report\",\n // basic: true,\n // premium: true,\n // link: \"/warren/exterior-report\",\n // content: (\n // \n // Relatórios {!isMobile() && \"(padrão Receita Federal)\"}\n // \n // ),\n // description: (\n // Relatórios auxiliares para declaração de IR \n // ),\n // icon: (\n // \n // ),\n // },\n // {\n // id: \"dividendos\",\n // premium: true,\n // link: \"/warren/exterior-dividendos\",\n // content: Dividendos ,\n // icon: ,\n // description: Gestão dos dividendos recebidos ,\n // },\n // {\n // id: \"insert-manual\",\n // link: \"/warren/transferencia-custodia\",\n // content: Transferência de custódia ,\n // icon: (\n // \n // ),\n // },\n // {\n // id: \"integration\",\n // link: \"/warren/exterior-integration\",\n // content: Integração com Warren ,\n // icon: ,\n // },\n {\n id: \"insert-manual\",\n premium: false,\n link: \"/warren/transferencia-custodia\",\n content: Transferência de custódia ,\n icon: (\n \n ),\n // description: Gestão de transferências de custódia ,\n }\n];\n","export default __webpack_public_path__ + \"static/media/check.d361ff1f.svg\";","import styled from \"styled-components\";\nimport check from \"../../../assets/icons/check.svg\";\n\nexport const PayDarfButtonContainer = styled.div`\n display: flex;\n cursor: pointer;\n position: relative;\n align-items: center;\n justify-content: center;\n font-size: 12px;\n font-weight: 600;\n line-height: 16px;\n color: var(--white);\n height: 24px;\n border-radius: 12px;\n transition: filter 0.3s;\n padding: 4px 24px 4px 12px;\n background-color: var(--em-aberto);\n\n :hover {\n filter: contrast(0.75) brightness(1.15);\n }\n\n :after {\n content: \"\";\n top: 6px;\n right: 6px;\n width: 12px;\n height: 12px;\n border-radius: 50%;\n position: absolute;\n background-color: var(--white);\n }\n\n &.checked {\n background-color: var(--green);\n &::after {\n top: 5px;\n width: 14px;\n height: 14px;\n background-size: 100%;\n background-position: center;\n background-repeat: no-repeat;\n background-color: var(--green);\n background-image: url(${check});\n }\n }\n`;\n","import clsx from \"clsx\";\nimport { Tooltip } from \"antd\";\nimport { PayDarfButtonContainer } from \"./styles\";\n\ninterface PayDarfButtonProps {\n checked?: boolean;\n onChange?: () => void;\n checkedChildren?: React.ReactNode;\n unCheckedChildren?: React.ReactNode;\n}\n\nexport const PayDarfButton: React.FC = ({\n checked,\n onChange,\n checkedChildren,\n unCheckedChildren,\n}) => {\n return (\n \n \n {checked ? checkedChildren : unCheckedChildren}\n \n \n );\n};\n","import axios from 'axios';\n\nexport const instanceMsCalc = axios.create({\n baseURL: process.env.REACT_APP_MS_CALC_URL ?? 'http://localhost:5001',\n headers: { Authorization: process.env.REACT_APP_AUTH_TOKEN ?? '' },\n});","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n display: flex;\n flex-direction: column;\n svg {\n width: 20px;\n height: 20px;\n min-width: 20px;\n }\n .ant-checkbox {\n margin-top: -2px;\n }\n\n .checkbox-wrapper {\n display: flex;\n align-items: center;\n margin-left: 12px;\n\n .label {\n margin-left: 6px;\n }\n }\n\n .loading-wrapper {\n display: flex;\n align-items: center;\n margin-left: -2px;\n svg {\n width: 20px;\n height: 20px;\n }\n }\n\n .ant-checkbox-wrapper {\n margin-left: 0;\n display: flex;\n align-items: center;\n\n & + label {\n margin-top: 12px;\n }\n\n .ant-checkbox-checked .ant-checkbox-inner {\n background-color: var(--ant-primary-color);\n border-color: var(--ant-primary-color);\n }\n\n .ant-checkbox-inner {\n background: transparent;\n border: 2px solid var(--ant-primary-color);\n width: 20px;\n height: 20px;\n border-radius: 2px;\n }\n }\n\n .ant-checkbox + span {\n margin-left: -10px;\n }\n`;\n","import { Checkbox, Row } from \"antd\";\nimport { IoMdGlobe } from \"react-icons/io\";\nimport { BiHomeAlt } from \"react-icons/bi\";\nimport { IconBaseProps } from \"react-icons\";\nimport { RiStockLine } from \"react-icons/ri\";\nimport React, { useEffect, useState } from \"react\";\nimport { BsCurrencyBitcoin } from \"react-icons/bs\";\nimport { LoadingOutlined } from \"@ant-design/icons\";\nimport { Container } from \"./styles\";\n\ninterface IProps {\n data: string[];\n saveModules: (\n modules: string[],\n moduleName: string,\n deleted: boolean\n ) => void;\n loading: any;\n}\n\ninterface ISelect {\n id: string;\n icon: IconBaseProps;\n label: string;\n}\n\nconst VelotaxCalculatorsSelect: ISelect[] = [\n {\n id: \"bolsa\",\n icon: ,\n label: \"Investimentos na Bolsa de Valores\",\n },\n {\n id: \"cripto\",\n icon: ,\n label: \"Investimentos em Criptoativos\",\n },\n {\n id: \"exterior\",\n icon: ,\n label: \"Investimentos no exterior\",\n },\n {\n id: \"autonomos\",\n icon: ,\n label: \"Autônomos e Aluguel\",\n },\n];\n\nexport const SelectBoxes: React.FC = ({\n data = {},\n saveModules,\n loading,\n}) => {\n const [values, setValues] = useState((data || []) as string[]);\n\n useEffect(() => {\n if (Array.isArray(data)) {\n setValues(data);\n }\n }, [data]);\n\n const handleSetValue = (\n array: string[],\n moduleName: string,\n deleted: boolean\n ) => {\n setValues(array);\n saveModules(array, moduleName, deleted);\n };\n\n const handleChange = (value: string): void => {\n if (values.includes(value)) {\n handleSetValue(\n values.filter((vl) => vl !== value),\n value,\n true\n );\n } else {\n handleSetValue([...values, value], value, false);\n }\n };\n\n return (\n \n {VelotaxCalculatorsSelect.map((op: any, index: number) => (\n \n {loading?.[op.id] === true ? (\n \n
\n
\n {op.icon && op.icon}\n
{op.label}
\n
\n
\n ) : (\n loading[key] === true\n )}\n checked={values.includes(op.id)}\n onChange={() => handleChange(op.id)}\n >\n \n {op.icon && op.icon}\n
{op.label}
\n
\n \n )}\n
\n ))}\n \n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 128px;\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n margin: 0 auto;\n max-width: 600px;\n border-radius: 16px;\n background-color: var(--velotax-background-color);\n h1 {\n font-size: 24px;\n margin: 0 0 24px;\n position: relative;\n color: var(--velotax-font-color);\n\n &::before {\n content: \"\";\n top: 0;\n width: 4px;\n left: -32px;\n height: 100%;\n position: absolute;\n background-color: var(--ant-primary-color);\n }\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px;\n border-radius: 0;\n min-height: calc(100vh - 64px);\n h2 {\n &::before {\n left: -24px;\n }\n }\n }\n`;\n\nexport const CalculatorsSelect = styled.div`\n h2 {\n font-size: 18px;\n font-weight: 500;\n margin: 0 0 24px;\n color: var(--velotax-font-color);\n }\n`;\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 128px;\n &.view-edit {\n padding: 0;\n }\n &.view {\n padding: 0;\n }\n\n h1 {\n width: 700px;\n margin: 0 auto 1.5rem;\n font-weight: 400;\n font-size: 2rem;\n position: relative;\n line-height: 2.5rem;\n padding-bottom: 1rem;\n color: var(--velotax-font-color-light);\n\n :after {\n content: \"\";\n position: absolute;\n left: 0;\n bottom: 0;\n height: 2px;\n width: 100px;\n background-color: var(--ant-primary-color);\n }\n }\n\n & > span.ant-typography {\n display: block;\n width: 700px;\n margin: 0 auto 1rem;\n font-size: 1rem;\n line-height: 1.5rem;\n color: var(--velotax-font-color-light);\n\n span {\n cursor: pointer;\n font-weight: 500;\n color: var(--ant-primary-color);\n }\n }\n\n .edition-button-corretagem {\n display: flex;\n align-items: center;\n margin-top: 12px;\n background: var(--velotax-background-color);\n\n button {\n display: flex;\n align-items: center;\n\n span {\n margin-left: 8px;\n }\n }\n }\n\n .juridic-messages-container {\n width: 100%;\n opacity: 0.8;\n padding: 24px 0 0;\n\n .message {\n display: flex;\n flex-direction: row;\n justify-content: flex-start;\n margin-bottom: 1rem;\n }\n\n p {\n font-size: 0.75rem;\n font-weight: 400;\n text-align: justify;\n line-height: 1.4rem;\n color: var(--velotax-font-color-light);\n margin-bottom: 0;\n\n i {\n font-weight: bold;\n color: var(--velotax-font-color-light);\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 0 0 64px;\n\n .juridic-messages-container {\n margin: 0;\n width: 100%;\n p {\n margin-left: 0;\n }\n }\n\n h1 {\n max-width: calc(100% - 48px);\n margin: 0 24px 1.5rem;\n padding: 24px 0 1rem;\n }\n\n & > span.ant-typography {\n width: 100%;\n margin: 0 auto 1rem;\n padding: 0 24px;\n }\n }\n`;\n\nexport const ButtonContent = styled.div`\n width: 700px;\n margin: 0 auto 8px;\n div {\n display: flex;\n justify-content: flex-end;\n }\n\n @media only screen and (max-device-width: 812px) {\n width: 100%;\n div {\n width: 100%;\n padding: 16px 24px 0;\n button {\n span {\n font-size: 11px;\n white-space: nowrap;\n text-overflow: ellipsis;\n }\n }\n }\n }\n\n @media only screen and (max-device-width: 576px) {\n div {\n button {\n width: 100%;\n }\n }\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n &.second.view,\n &.second.view-edit {\n padding: 0 32px 32px;\n }\n &.second {\n padding: 32px;\n border: none;\n background-color: transparent;\n }\n .ant-divider {\n margin: 8px 0 24px;\n }\n h3.date-title {\n font-size: 24px;\n font-weight: 700;\n }\n margin: 0 auto;\n max-width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n h3 {\n display: flex;\n align-items: center;\n column-gap: 16px;\n font-size: 20px;\n color: var(--velotax-font-color-light);\n margin-bottom: 0;\n user-select: none;\n }\n .desc-label {\n display: flex;\n align-items: center;\n svg {\n fill: var(--ant-primary-color);\n }\n }\n .month-chevron {\n width: 32px;\n height: 32px;\n cursor: pointer;\n &.disabled {\n opacity: 0.15;\n cursor: default;\n }\n }\n .desc-content {\n display: flex;\n align-items: flex-start;\n justify-content: center;\n column-gap: 8px;\n svg {\n fill: var(--ant-primary-color);\n }\n span {\n align-self: center;\n }\n button {\n min-width: 32px;\n }\n }\n .market {\n opacity: 0.6;\n font-weight: bold;\n }\n .ml-40 {\n margin-left: 40px;\n }\n .btn-icon {\n padding: 0;\n width: 48px;\n display: flex;\n .MuiTouchRipple-root {\n width: 48px;\n }\n }\n .add {\n float: right;\n min-width: 128px;\n margin-bottom: 16px;\n }\n .ant-descriptions-view {\n border-radius: 4px !important;\n .ant-descriptions-item-label {\n width: 60%;\n padding: 10px 16px;\n color: var(--velotax-font-color-light) !important;\n }\n .ant-descriptions-item-content {\n padding: 10px 16px;\n text-align: center;\n background-color: var(--velotax-border-color) !important;\n span {\n color: var(--velotax-font-color-light) !important;\n }\n }\n }\n .ant-list.ant-list-split {\n clear: both;\n padding: 0 16px;\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-ghost);\n border-radius: 4px;\n .ant-list-item-action > li {\n padding: 0;\n }\n }\n .ant-list-item-action > li,\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color-light);\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n .list-description {\n display: flex;\n flex-direction: column;\n }\n .min-darf-price {\n margin-top: 24px;\n display: flex;\n align-items: center;\n column-gap: 8px;\n }\n .text-center {\n display: block;\n text-align: center;\n }\n .ant-collapse-header {\n padding: 12px 0 !important;\n }\n .ant-collapse-content-box {\n padding: 16px 0 !important;\n }\n .ant-collapse\n > .ant-collapse-item\n > .ant-collapse-header\n .ant-collapse-arrow\n svg {\n transform: rotate(-90deg);\n }\n .ant-collapse-item-disabled {\n & > .ant-collapse-header {\n cursor: default;\n .ant-collapse-arrow {\n opacity: 0;\n }\n }\n }\n .ant-collapse-header-text {\n width: 100%;\n }\n .total-tax-header {\n width: calc(100% - 48px);\n display: flex;\n align-items: center;\n justify-content: space-between;\n &.original {\n h3:last-of-type {\n width: 40%;\n }\n }\n h3 {\n margin: 0 0 8px;\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n width: 100%;\n padding: 24px;\n border-radius: 0;\n &.first {\n padding: 24px;\n }\n &.second {\n padding: 24px !important;\n border-top: 1px solid #efefef;\n }\n h3 {\n font-size: 18px;\n }\n .anticon.anticon-right.ant-collapse-arrow {\n top: 24px;\n }\n .desc-content {\n column-gap: 2px;\n flex-direction: row;\n &.is-editting {\n row-gap: 8px;\n flex-direction: column;\n button {\n width: 100%;\n }\n }\n &:not(.is-editting) {\n span {\n line-height: 32px;\n }\n button {\n margin-top: -2px;\n }\n }\n &.ml-40 {\n margin-left: 0;\n }\n }\n .market {\n font-size: 11px;\n letter-spacing: -0.1px;\n }\n }\n`;\n\nexport const BolsaOperations = styled.div`\n border-radius: 4px;\n position: relative;\n color: var(--velotax-font-color-light);\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-background-color-ghost);\n\n .no-plan {\n span {\n user-select: none;\n }\n }\n\n .no-plan-container {\n position: absolute;\n top: 55px;\n left: 40%;\n width: 60%;\n z-index: 3;\n display: flex;\n padding: 8px;\n font-size: 16px;\n font-weight: 500;\n text-align: center;\n align-items: center;\n flex-direction: column;\n justify-content: center;\n background-color: #fff4;\n border-radius: 0 0 8px 0;\n height: calc(100% - 55px);\n backdrop-filter: blur(4px);\n border-left: 1px solid var(--velotax-background-color-ghost);\n .ant-btn {\n margin-top: 8px;\n }\n }\n\n div.strong span {\n font-weight: 700 !important;\n }\n div.opacity {\n opacity: 0.8;\n }\n\n div.header,\n div.sub-header,\n div.row {\n display: grid;\n grid-template-columns: 2fr 1fr 1fr 1fr;\n span + span {\n text-align: center;\n justify-content: center;\n }\n }\n\n div.sub-header,\n div.row {\n height: auto;\n min-height: 48px;\n border-top: none;\n & + div.sub-header {\n border-top: 1px solid var(--velotax-background-color-ghost);\n }\n & > span + span {\n background-color: var(--velotax-border-color);\n }\n & > span {\n padding: 8px 16px;\n display: flex;\n align-items: center;\n border-left: none;\n & + span {\n border-left: 1px solid var(--velotax-background-color-ghost);\n }\n }\n &:last-of-type > span:last-of-type {\n border-radius: 0 0 4px 0;\n }\n }\n\n div.header {\n padding: 16px;\n border-radius: 4px 4px 0 0;\n background-color: var(--velotax-background-color-ghost);\n span {\n font-weight: 700;\n }\n button {\n display: none;\n }\n }\n\n div.sub-header {\n font-size: 14px;\n & > span {\n font-weight: 300;\n }\n }\n\n div.row {\n opacity: 0.85;\n font-size: 13px;\n border-top: 1px solid var(--velotax-background-color-ghost);\n }\n\n @media only screen and (max-device-width: 812px) {\n position: relative;\n overflow-x: hidden;\n span {\n font-size: 12px;\n button {\n max-width: 24px;\n min-width: 24px;\n }\n }\n .no-plan-container {\n left: 50%;\n width: 50%;\n font-size: 14px;\n line-height: 18px;\n }\n div.header,\n div.sub-header,\n div.row {\n width: 200%;\n height: auto;\n min-height: 48px;\n grid-template-columns: 1fr 1fr 1fr 1fr;\n }\n div.sub-header,\n div.row {\n & > span {\n padding: 12px;\n }\n }\n\n div.header {\n padding: 16px 0;\n }\n div.header > span + span {\n padding: 0 8px;\n justify-content: space-between;\n }\n\n div.header > span:first-of-type,\n div.sub-header > span:first-of-type {\n position: sticky;\n left: 0;\n }\n div.header > span.show,\n div.sub-header > span.show {\n opacity: 1;\n z-index: 2;\n }\n div.header > span,\n div.sub-header > span {\n opacity: 0;\n z-index: 1;\n }\n div.header > span {\n display: flex;\n align-items: center;\n justify-content: center;\n column-gap: 8px;\n }\n div.header {\n button {\n display: flex;\n }\n }\n div.header > span.show svg polyline {\n stroke: var(--velotax-font-color);\n }\n }\n`;\n","import { v4 } from \"uuid\";\nimport { message } from \"antd\";\nimport { useNavigate } from \"react-router-dom\";\nimport {\n Dispatch,\n SetStateAction,\n createContext,\n useCallback,\n useContext,\n useEffect,\n useState,\n} from \"react\";\nimport { errorMessage } from \"../utils\";\nimport apiBolsa from \"../services/apiBolsa\";\nimport { useAuth } from \"../contexts/AuthContext\";\nimport { useBroker } from \"../contexts/BrokerContext\";\nimport moment from \"moment\";\n\ninterface IOldTransactionsContext {\n changed: boolean;\n loadingPut: boolean;\n loadingGet: boolean;\n setLoadingPut: Dispatch>;\n setLoadingGet: Dispatch>;\n oldPositionEdit: any;\n doacaoHerancaEdit: any;\n oldPositionList: any[];\n doacoesHerancas: any[];\n showAddPositionModal: boolean;\n setOldPositionEdit: Dispatch>;\n setDoacaoHerancaEdit: Dispatch>;\n setShowAddPositionModal: Dispatch>;\n downloadOldTransactionsExcel: () => Promise;\n downloadDoacoesHerancasExcel: () => Promise;\n saveOldPositions: () => void;\n saveDoacoesHerancas: () => void;\n addOldPosition: (item: any) => void;\n addDoacaoHeranca: (item: any) => void;\n editOldPosition: (item: any) => void;\n editDoacaoHeranca: (item: any) => void;\n handleEditOldPosition: (item: any, index: number) => void;\n handleRemoveOldPosition: (item: any, index: number) => void;\n handleRemoveDoacaoHeranca: (item: any, index: number) => void;\n handleEditDoacaoHeranca: (item: any, index: number) => void;\n getOldTransactionsInit: () => void;\n getDoacoesHerancasInit: () => void;\n}\n\nconst OldTransactionsContext = createContext({} as IOldTransactionsContext);\n\nexport const OldTransactionsProvider: React.FC = ({ children }) => {\n const { user } = useAuth();\n const navigate = useNavigate();\n const { initIntegration, currentBroker } = useBroker();\n const [changed, setChanged] = useState(false);\n const [loadingGet, setLoadingGet] = useState(false);\n const [loadingPut, setLoadingPut] = useState(false);\n const [oldPositionEdit, setOldPositionEdit] = useState({});\n const [doacaoHerancaEdit, setDoacaoHerancaEdit] = useState({});\n const [oldPositionList, setOldPositionList] = useState(null);\n const [doacoesHerancas, setDoacoesHerancas] = useState(null);\n const [showAddPositionModal, setShowAddPositionModal] = useState(false);\n\n const handleEditOldPosition = (item: any, index: number) => {\n setShowAddPositionModal(true);\n setOldPositionEdit(item);\n };\n\n const handleEditDoacaoHeranca = (item: any, index: number) => {\n setShowAddPositionModal(true);\n setDoacaoHerancaEdit(item);\n };\n\n const handleRemoveOldPosition = (item: any, index: number) => {\n const pojo = oldPositionList\n .filter((e: any) =>\n \"lendingSide\" in item ? e.lendingSide : !e.lendingSide\n )\n .find((item: any, i: number) => i === index);\n const listOldPosition = oldPositionList.filter(\n (item: any) => JSON.stringify(pojo) !== JSON.stringify(item)\n );\n setOldPositionList(listOldPosition);\n setChanged(true);\n };\n\n const handleRemoveDoacaoHeranca = (item: any, index: number) => {\n const itemParam = item;\n const list = doacoesHerancas.map(\n (item: any) => {\n const newOrders = item?.orders.filter((or: any) => {\n delete or.dateView;\n if (JSON.stringify(itemParam) !== JSON.stringify(or)) return or;\n })\n return { ...item, orders: newOrders };\n }\n );\n setDoacoesHerancas(list);\n setChanged(true);\n }\n\n const editOldPosition = (editItem: any) => {\n const listOldPositions = oldPositionList.map((item: any) =>\n editItem.id === item.id\n ? {\n ...item,\n ...editItem,\n price: editItem?.avgBuyPrice,\n operation:\n editItem?.lendingSide && editItem?.lendingSide === \"Tomador\"\n ? \"V\"\n : \"C\",\n }\n : item\n );\n setOldPositionList(listOldPositions);\n setShowAddPositionModal(false);\n setOldPositionEdit({});\n setChanged(true);\n };\n\n const editDoacaoHeranca = (editItem: any) => {\n const list = doacoesHerancas.map((item: any) => {\n const newOrders = item?.orders.map((or: any) => {\n delete or.dateView;\n const newDate = new Date(\n `${moment(editItem?.dateView, 'DD/MM/YYYY', true).format(\n 'YYYY-MM-DD',\n )}T03:00:00.000Z`);\n return editItem?._id === or?._id\n ? {\n ...or,\n date: newDate,\n code: editItem?.code,\n price: editItem?.price,\n quantity: editItem?.quantity,\n totalValue: editItem?.price * editItem?.quantity,\n }\n : or\n })\n return { ...item, orders: newOrders };\n }\n );\n setDoacoesHerancas(list);\n setShowAddPositionModal(false);\n setDoacaoHerancaEdit({});\n setChanged(true);\n };\n\n const addOldPosition = (item: any) => {\n const listOldPosition: any[] = [...oldPositionList];\n const newPostion = {\n operation:\n item?.lendingSide && item?.lendingSide === \"Tomador\" ? \"V\" : \"C\",\n avgBuyPrice: item?.avgBuyPrice,\n price: item?.avgBuyPrice,\n code: item?.code,\n feeB3: item?.feeB3 || 0,\n institution: \"Desconhecida\",\n date: new Date(2019, 11, 31),\n market: \"Mercado a Vista\",\n quantity: item?.quantity,\n month: 11,\n year: 2019,\n id: v4(),\n };\n\n if (item.lendingSide)\n Object.assign(newPostion, { lendingSide: item?.lendingSide });\n listOldPosition.push(newPostion);\n\n setOldPositionList(listOldPosition);\n setShowAddPositionModal(false);\n setChanged(true);\n };\n\n const addDoacaoHeranca = (addItem: any) => {\n const list = [...doacoesHerancas];\n const month = Number((addItem?.dateView)?.split('/')[1]);\n const year = Number((addItem?.dateView)?.split('/')[2]);\n const now = new Date();\n const newDate = new Date(\n `${moment(addItem?.dateView, 'DD/MM/YYYY', true).format(\n 'YYYY-MM-DD',\n )}T03:00:00.000Z`);\n const retExisting = doacoesHerancas.filter((item: any) => {\n const newOrders = item?.orders;\n if (item?.month == month && item?.year == year) {\n newOrders.push(\n {\n code: addItem?.code,\n operation: \"C\",\n date: newDate,\n quantity: addItem?.quantity,\n price: addItem?.price,\n totalValue: addItem?.price * addItem?.quantity,\n totalFeeNota: 0,\n name: addItem?.code,\n market: \"Mercado a Vista\",\n expirationDate: null,\n institution: \"Desconhecida\",\n operationType: \"Credito\"\n }\n )\n return { ...item, orders: newOrders };\n }\n })\n if (!retExisting || retExisting.length == 0) {\n const newOperation = {\n cpf: user?.user?.cpf,\n month: month,\n year: year,\n provider: 'b3',\n createdAt: now,\n updatedAt: now,\n orders: [\n {\n code: addItem?.code,\n operation: \"C\",\n date: newDate,\n quantity: addItem?.quantity,\n price: addItem?.price,\n totalValue: addItem?.price * addItem?.quantity,\n totalFeeNota: 0,\n name: addItem?.code,\n market: \"Mercado a Vista\",\n expirationDate: null,\n institution: \"Desconhecida\",\n operationType: \"Credito\"\n }\n ]\n };\n \n list.push(newOperation);\n }\n\n setDoacoesHerancas(list);\n setShowAddPositionModal(false);\n setChanged(true);\n };\n\n const saveOldPositions = useCallback(async () => {\n const orders = oldPositionList.map((item: any) => ({\n ...item,\n id: undefined,\n }));\n try {\n setLoadingPut(true);\n await apiBolsa.post(\"/operation\", { orders });\n // message.success(\"Dados salvos com sucesso!\");\n navigate(`/${currentBroker?.path}/bolsa-historic`);\n } catch (err) {\n message.error(errorMessage);\n throw Error;\n } finally {\n setLoadingPut(false);\n }\n }, [oldPositionList, currentBroker, navigate]);\n\n const saveDoacoesHerancas = useCallback(async () => {\n const operations = doacoesHerancas.map((item: any) => {\n const newOrders = item?.orders.map((or: any) => {\n return { ...or }\n })\n return { ...item, orders: newOrders };\n }\n )\n try {\n setLoadingPut(true);\n await apiBolsa.post(\"/operation/donations\", { operations });\n message.success(\"Dados salvos com sucesso!\");\n } catch (err) {\n message.error(errorMessage);\n throw Error;\n } finally {\n navigate(`/${currentBroker?.path}/bolsa-historic`);\n setLoadingPut(false);\n }\n }, [doacoesHerancas, currentBroker, navigate]);\n\n const downloadOldTransactionsExcel = async () => {\n return await apiBolsa\n .get(\"/operation/download?month=12&year=2019\", {\n responseType: \"blob\",\n })\n .then((res) => {\n return res.data\n })\n .catch(() => {\n message.error(errorMessage);\n return null\n })\n }\n\n const downloadDoacoesHerancasExcel = async () => {\n return await apiBolsa\n .get(\"/operation/download/donations?operationType=Credito\", {\n responseType: \"blob\",\n })\n .then((res) => {\n return res.data\n })\n .catch(() => {\n message.error(errorMessage);\n return null\n })\n }\n\n const getOldTransactionsInit = useCallback(\n (setShowModalPrePreenchida?: (p: boolean) => void) => {\n setLoadingGet(true);\n apiBolsa\n .get(\"/operation?month=12&year=2019\", {})\n .then((response) => {\n setOldPositionList(\n response.data?.[0]?.orders.length\n ? response.data?.[0].orders.map((item: any) => ({\n ...item,\n avgBuyPrice: item.price,\n id: item._id,\n }))\n : []\n );\n setLoadingGet(false);\n })\n .catch(() => {\n setLoadingGet(false);\n setOldPositionList([]);\n message.error(errorMessage);\n })\n .finally(() => setShowModalPrePreenchida?.(false));\n },\n [setOldPositionList]\n );\n\n const getDoacoesHerancasInit = useCallback(\n (setShowModalPrePreenchida?: (p: boolean) => void) => {\n setLoadingGet(true);\n apiBolsa\n .get(\"/operation/donations?operationType=Credito\", {})\n .then((response) => {\n setDoacoesHerancas(response.data || []);\n setLoadingGet(false);\n })\n .catch(() => {\n setLoadingGet(false);\n setDoacoesHerancas([]);\n message.error(errorMessage);\n })\n .finally(() => setShowModalPrePreenchida?.(false));\n },\n [setDoacoesHerancas]\n );\n\n useEffect(() => {\n if (user?.user?.cpf) {\n getOldTransactionsInit();\n }\n }, [getOldTransactionsInit, initIntegration, user.user]);\n\n return (\n \n {children}\n \n );\n};\nexport const useOldTransactions = () => useContext(OldTransactionsContext);","import FileSaver from \"file-saver\";\nimport { message } from \"antd\";\n\nexport const handleDownloadSheet = async (buffer: any, title: string) => {\n try {\n const data = Uint8Array.from(buffer)\n const blob = new Blob([data], { type: 'attach.contentType' });\n FileSaver.saveAs(blob, `${title}.xlsx`);\n } catch (error: any) {\n message.error(error)\n }\n}\n","import { Table, Typography } from \"antd\";\nimport { DrawerModal } from \"../DrawerModal\";\nimport Button from \"../Button\";\nimport { useNavigate } from \"react-router-dom\";\n\ninterface ModalZeroBuyPrice {\n title?: string;\n visible?: boolean;\n dataZeroBuyPrice: any[];\n setModalZeroBuyPrice: Function;\n setIgnoreZeroBuyPrice: Function;\n}\n\nexport const ModalZeroBuyPrice: React.FC = ({\n title,\n visible,\n dataZeroBuyPrice,\n setModalZeroBuyPrice,\n setIgnoreZeroBuyPrice\n}) => {\n const navigate = useNavigate();\n\n return (\n setModalZeroBuyPrice(false)}\n >\n \n Neste mês, verificamos que alguns de seus ativos estão com custo de compra igual a zero.\n \n (\n {value} \n ),\n },\n {\n align: \"center\",\n title: \"Motivo\",\n dataIndex: \"reason\",\n render: (value) => (\n <>\n {value?.text ?\n <>\n {value?.text}\n { navigate(value?.link) }} target=\"_blank\" rel=\"noreferrer\">\n {\" \"} {value?.textLink} {\" \"}\n \n >\n :\n <>\n Entre em contato através de\n \n {\" \"} suporte@velotax.com.br {\" \"}\n \n para avaliarmos a situação.\n >\n }\n >\n ),\n },\n ]}\n />\n {\n setIgnoreZeroBuyPrice(true);\n setModalZeroBuyPrice(false);\n }}\n >\n ESTOU CIENTE. EMITIR DARF, MESMO COM O CUSTO DE COMPRA ZERADO.\n \n \n )\n}","import clsx from \"clsx\";\nimport moment from \"moment\";\nimport { Button } from \"@mui/material\";\nimport { FaRegEdit } from \"react-icons/fa\";\nimport { BsCalendar3, BsDownload } from \"react-icons/bs\";\nimport { useLocation, useNavigate } from \"react-router-dom\";\nimport { GrFormNext, GrFormPrevious } from \"react-icons/gr\";\nimport { BiChevronLeft, BiChevronRight } from \"react-icons/bi\";\nimport {\n AiOutlineDelete,\n AiOutlineDownload,\n AiOutlineLock,\n} from \"react-icons/ai\";\nimport {\n LoadingOutlined,\n PlusOutlined,\n QuestionCircleOutlined,\n} from \"@ant-design/icons\";\nimport {\n ReactNode,\n useCallback,\n useEffect,\n useImperativeHandle,\n useMemo,\n useRef,\n useState,\n} from \"react\";\nimport {\n Col,\n Collapse,\n DatePicker,\n Descriptions,\n List,\n message,\n Modal,\n Row,\n Skeleton,\n Space,\n Spin,\n Tooltip,\n Typography,\n} from \"antd\";\nimport apis from \"../../services/apis\";\nimport apiBolsa from \"../../services/apiBolsa\";\nimport { Page } from \"../../constants/brokers\";\nimport AntButton from \"../../components/Button\";\nimport HandleTag from \"../../services/handleTag\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport { FormModal } from \"../../components/FormModal\";\nimport { apiPayment } from \"../../services/apiPayment\";\nimport { useBroker } from \"../../contexts/BrokerContext\";\nimport { BackButton } from \"../../components/BackButton\";\nimport { PaymentModal } from \"../../components/PaymentModal\";\nimport { Container, Content, BolsaOperations } from \"./styles\";\nimport { historicoVendasEmptyText } from \"../../constants/darf\";\nimport { IntegrationModal } from \"../Darf/Components/IntegrationModal\";\nimport { useOldTransactions } from \"../../contexts/OldTransactionsContext\";\nimport DeleteConfirmationModal from \"../../components/DeleteConfirmationModal\";\nimport { DatePickerExtraFooter } from \"../Darf/Components/DatePickerExtraFooter\";\nimport {\n DatePickerMonthCell,\n YearResume,\n} from \"../Darf/Components/DatePickerMonthCell\";\nimport {\n DarfModal,\n DarfResultDescriptions,\n minDarfPrice,\n maxDarfPrice,\n IDarfBolsa,\n defaultDarfBolsa,\n DarfBolsaFormItemRows,\n PrejuFormItemRows,\n YearResumeStatusFromBackEnum,\n DarfBolsaTabs,\n IrrfFormItemRows,\n CorretagemFormItemRows,\n} from \"../../constants/darfBolsa\";\nimport {\n antDatePickerLocale,\n download,\n errorMessage,\n formatCurrency,\n getZeroBuyPrice,\n isMobile,\n monthsExtended,\n numberToPercentageWallet,\n} from \"../../utils\";\nimport { handleDownloadSheet } from \"../../utils/handleDownloadSheet\";\nimport { ModalZeroBuyPrice } from \"../../components/ModalZeroBuyPrice\";\n\ninterface DarfBolsaProps {\n item: Page;\n view?: boolean;\n viewEdit?: boolean;\n closeModal?: () => void;\n darf?: {\n id: string;\n fullPath?: string;\n };\n date?: {\n year?: number;\n month?: number;\n };\n darfBolsaRef: React.MutableRefObject;\n}\n\nexport const DarfBolsa: React.FC = ({\n view,\n item,\n date,\n darf,\n viewEdit,\n closeModal,\n darfBolsaRef,\n}) => {\n const navigate = useNavigate();\n const { state } = useLocation();\n const { oldPositionList } = useOldTransactions();\n const { currentBroker, currentPage } = useBroker();\n const {\n user,\n showUserPlanModal,\n userPlanModal,\n hasPlan: hasPremiumPlan,\n hasPermissionGeneral,\n } = useAuth();\n const today = new Date();\n const queryYear = (state as any)?.year as number;\n const queryMonth = (state as any)?.month as number;\n const currentMonth = today.getMonth();\n const currentYear = today.getFullYear();\n const [hasItems, sethasItems] = useState(false);\n const maxPixPayment = 10000;\n const maxCreditCardPayment = 10000;\n\n const initialMonth = queryMonth\n ? queryMonth - 1\n : today.getMonth() - 1 >= 0\n ? today.getMonth() - 1\n : 11;\n const initialYear = queryYear\n ? queryYear\n : today.getMonth() - 1 >= 0\n ? today.getFullYear()\n : today.getFullYear() - 1;\n const defaultValue = moment({ month: initialMonth, year: initialYear });\n const currentDate = moment({\n month: currentMonth,\n year: currentYear,\n }).subtract(1, \"M\");\n window.history.replaceState({}, document.title);\n\n const year = date?.year ?? initialYear;\n const month = date?.month ?? initialMonth;\n const currentDarfDate = new Date(year, month, 15)\n const dateAllowedOnBasic = new Date(today.getFullYear(), today.getMonth() - 1, 1)\n const hideOnBasic = (currentDarfDate.getTime() < dateAllowedOnBasic.getTime() && user.user?.userPlanInfoVelotax?.type?.includes('BASIC'))\n const hasPlan = (hasPremiumPlan || hasPermissionGeneral) && !hideOnBasic;\n\n\n const [asset, setAsset] = useState();\n const [, setYear] = useState(initialYear);\n const [, setMonth] = useState(initialMonth);\n const [loading, setLoading] = useState(false);\n const [emitting, setEmitting] = useState(false);\n const [darfModal, setDarfModal] = useState();\n const [helpModal, setHelpModal] = useState();\n const [paymentModal, setPaymentModal] = useState(false);\n const [dataToEditPreju, setDataToEditPreju] = useState();\n const [showAssetModal, setShowAssetModal] = useState(false);\n const [paymentData, setPaymentData] = useState();\n const [showDeleteModal, setShowDeleteModal] = useState(false);\n const [data, setData] = useState(defaultDarfBolsa);\n const [buffer, setBuffer] = useState>();\n const [yearResume, setYearResume] = useState([]);\n const [gettingYearResume, setGettingYearResume] = useState(false);\n const [datePickerYear, setDatePickerYear] = useState(initialYear);\n const [editedAccumulatedTax, setEditedAccumulatedTax] = useState(false);\n const [showNotAuthorizedModal, setShowNotAuthorizedModal] = useState(false);\n const [loadingDarfButton, setLoadingDarfButton] = useState(false);\n const [showEditPrejuConfirmation, setShowEditPrejuConfirmation] =\n useState(false);\n\n useImperativeHandle(darfBolsaRef, () => {\n return {\n editedAccumulatedTax,\n setEditedAccumulatedTax,\n handleDarf: (justSave?: boolean) => {\n handleDarf(justSave);\n },\n };\n });\n\n const isAntesTerceiroDiaUtil = () => {\n const today = moment();\n // Verifica se a data de hoje é um sábado\n if (today.isoWeekday() === 6) {\n return false;\n }\n // Verifica se a data de hoje é anterior ao terceiro dia útil\n const firstDayOfMonth = moment(today).startOf(\"month\");\n let businessDaysCount = 0;\n while (businessDaysCount < 2) {\n if (firstDayOfMonth.isoWeekday() < 6) {\n businessDaysCount++;\n }\n firstDayOfMonth.add(1, \"day\");\n }\n return today.isBefore(firstDayOfMonth);\n };\n\n const thirdBusinessDayAndMonth = (item: any) => {\n if (disableCurrentMonth(item))\n return { disabled: true, title: \"Aguarde o encerramento do mês para pagar seu DARF\" }\n else if (item.month + 1 == currentMonth && item.year == currentYear && isAntesTerceiroDiaUtil())\n return { disabled: true, title: \"Aguarde o terceiro dia útil deste mês para pagar seu DARF\" }\n else return { disabled: false, title: \"\" }\n }\n\n const hideOldPositionsText =\n oldPositionList?.length < 1 || oldPositionList === null;\n\n const valorPrincipal = Math.max(\n 0,\n Number(\n (data.impostoComumFinal || 0) +\n (data.impostoDTFinal || 0) +\n (data.impostoFIIFinal || 0)\n ) +\n (Number(data[\"impostoAcumulado\"] || 0) - Number(data[\"irrfTotal\"] || 0))\n );\n const impostoCharge = useMemo(() => {\n return (\n valorPrincipal +\n (valorPrincipal < minDarfPrice\n ? 0\n : (data.multaAux ?? 0) + (data.jurosAux ?? 0))\n );\n }, [data, valorPrincipal]);\n\n const transactionFeature = item.features[0];\n const taxFeature = item.features[1];\n const yearResumeFeature = item.features[2];\n const prejuizoFeature = item.features[3];\n const irrfFeature = item.features[4];\n\n const { emitted } = useMemo(() => {\n if (gettingYearResume) {\n }\n const monthResume = yearResume\n .find((resume) => resume.year === year)\n ?.months.find((m) => m.month - 1 === month);\n return {\n emitted:\n monthResume?.status === YearResumeStatusFromBackEnum.PAYED ||\n monthResume?.status === YearResumeStatusFromBackEnum.NOT_PAYED,\n payed: !!monthResume?.payed,\n status: monthResume?.status,\n };\n }, [yearResume, year, month, gettingYearResume]);\n\n const handleLabelButton: ReactNode = useMemo(() => {\n // if (!userPlanInfoStatus) return \"Assinar Plano Emissão de DARF\";\n\n if ((data?.impostoTotalFinal || 0) <= maxDarfPrice) return \"Pagar DARF\";\n else {\n if (emitted) return \"Emitir novo DARF\";\n else return taxFeature?.buttonLabel;\n }\n }, [data?.impostoTotalFinal, taxFeature?.buttonLabel, emitted]);\n\n const getDarf = useCallback(\n (id: string) => {\n view && setLoading(true);\n view &&\n (currentPage?.api || apis)\n .get(`${taxFeature.apiUrl}/${id}`)\n .then((response) => {\n setYear(response.data?.year);\n setMonth(response.data?.month - 1);\n setData({\n ...response.data,\n emitted: true,\n });\n })\n .catch((err) => {\n message.error(err.response?.data?.message || errorMessage);\n })\n .finally(() => setLoading(false));\n },\n [view, taxFeature.apiUrl, currentPage]\n );\n\n const getYearResume = useCallback(\n (year: number, force?: boolean, url?: string) => {\n setGettingYearResume(true);\n !view &&\n year <= new Date().getFullYear() &&\n year >= moment(`2020-01-01`).get(\"year\") &&\n (force ||\n (!yearResumeFeature.disabled &&\n !yearResume.find((resume) => resume.year === year))) &&\n (currentPage?.api || apis)\n .get(yearResumeFeature.apiUrl, { params: { year } })\n .then((response) => {\n const index = yearResume.findIndex(\n (resume) => resume.year === year\n );\n if (index >= 0) {\n setYearResume((yearResume) => {\n yearResume.splice(index, 1, response.data);\n return yearResume;\n });\n } else {\n setYearResume((yearResume) => [...yearResume, response.data]);\n }\n if (url && !isMobile()) {\n download(url);\n }\n })\n .catch((err) => console.log(err))\n .finally(() => setGettingYearResume(false));\n },\n [yearResumeFeature, yearResume, view, currentPage]\n );\n\n const getTaxes = useCallback(\n (\n nAtualizarDarfsPassadas?: boolean,\n nAtualizarIrrf?: boolean,\n atualizouImpostoAcumulado?: boolean,\n nAtualizarPrejuizosAcumulados?: boolean,\n atualizarTotalmente?: boolean,\n atualizouCorretagem?: boolean\n ) => {\n (!hasPlan || !view) && setLoading(true);\n (!hasPlan || !view) &&\n (currentPage?.api || apis)\n .get(transactionFeature.apiUrl, {\n params: {\n month: month + 1,\n year,\n nAtualizarDarfsPassadas: nAtualizarDarfsPassadas || false,\n nAtualizarIrrf: nAtualizarIrrf || false,\n atualizouImpostoAcumulado: atualizouImpostoAcumulado || false,\n nAtualizarPrejuizosAcumulados:\n nAtualizarPrejuizosAcumulados || false,\n atualizarTotalmente: atualizarTotalmente || false,\n atualizouCorretagem: atualizouCorretagem || false,\n },\n })\n .then((response) => {\n if (response.data.memoriaCalculo) {\n setData({\n ...response.data,\n multa: response.data.multaAux,\n juros: response.data.jurosAux,\n impostoDevido: response.data.impostoTotal,\n memoriaCalculo: response.data.memoriaCalculo.map(\n (memoCalc: any) => ({\n ...memoCalc,\n date: moment(memoCalc.date.slice(0, 10)).format(\n \"DD/MM/YYYY\"\n ),\n originalDate: memoCalc.date,\n market:\n memoCalc.market === \"Opção de Compra\" ||\n memoCalc.market === \"Opção de Venda\"\n ? \"Mercado de opções\"\n : memoCalc.market,\n })\n ),\n });\n }\n // setNotAuthorized(false);\n })\n .catch((err) => {\n setData(defaultDarfBolsa);\n if (err?.response?.data === \"NOT_AUTHORIZED\") {\n // setNotAuthorized(true);\n setShowNotAuthorizedModal(true);\n } else {\n // setNotAuthorized(false);\n message.error(errorMessage);\n }\n })\n .finally(() => setLoading(false));\n },\n [month, year, transactionFeature, view, hasPlan, currentPage]\n );\n\n const handleDarf = (justSave?: boolean) => {\n setEmitting(true);\n onCloseDarfModal();\n (currentPage?.api || apis)\n .post(taxFeature.apiUrl, {\n ...data,\n memoriaCalculo: [],\n transactions: data.memoriaCalculo,\n totalImpostoDevido: data.impostoTotalFinal,\n impostoDevido: data.impostoTotal,\n juros: data.jurosAux,\n multa: data?.multaAux,\n year,\n month: month + 1,\n name: user.user.name,\n insertPix: true,\n paymentInfo: {\n clientInfo: {\n name: user?.user?.name,\n email: user?.user?.email,\n document: user?.user?.cpf,\n },\n },\n justSave,\n payed: false,\n })\n .then((res) => {\n setData((data) => ({ ...data, emitted: true }));\n getYearResume(year, true, justSave ? \"\" : res.data.darfUrl);\n setEmitting(false);\n closeModal?.();\n if (!justSave) {\n message.success(\"O DARF foi enviado para o seu e-mail\");\n }\n })\n .catch((err) => {\n message.error(err.response?.data?.message || errorMessage);\n setEmitting(false);\n });\n };\n\n const handleRegularize = (regular: boolean) => {\n setEmitting(true);\n (currentPage?.api || apis)\n .post(\"transaction/regularize\", {\n year,\n month: month + 1,\n regular,\n impostoAcumulado:\n (data.impostoTotal ?? 0) + (data.impostoAcumulado ?? 0),\n })\n .then((res) => {\n getYearResume(year, true, res.data.darfUrl);\n getTaxes(false, false, false, true);\n message.success(\"Dados salvos\");\n closeModal?.();\n })\n .catch(() => message.error(errorMessage))\n .finally(() => setEmitting(false));\n };\n\n const onChangeMonth = (value: moment.Moment | null, dateString: string) => {\n const [month, year] = dateString.split(\"-\");\n setMonth(parseInt(month) - 1);\n setYear(parseInt(year));\n };\n\n const unsetAsset = () => {\n setAsset(undefined);\n };\n\n const handleCloseDeleteModal = () => {\n setShowDeleteModal(false);\n unsetAsset();\n };\n\n const handleCloseAssetModal = () => {\n setShowAssetModal(false);\n unsetAsset();\n };\n\n const onCloseHelpModal = () => {\n setHelpModal(undefined);\n };\n\n const onCloseDarfModal = () => {\n setDarfModal(undefined);\n };\n\n const handleOpenDarfModal = () => {\n setDarfModal(DarfModal(emitted, onCloseDarfModal, () => handleDarf()));\n };\n\n const handleOpenPaymentModal = () => {\n setPaymentModal(!paymentModal);\n };\n\n const handleAdd = (event: React.MouseEvent) => {\n event.stopPropagation();\n setShowAssetModal(true);\n };\n\n const handleEdit = (item: any, index: number) => {\n setShowAssetModal(true);\n setAsset({\n ...item,\n id: item._id,\n feeB3: item.fee,\n price: item.sellPrice,\n avgBuyPrice: item.buyPrice,\n data: moment.utc(item.date.toLocaleString()).format(\"DD/MM/YYYY\"),\n });\n };\n\n const handleRemove = (item: any, index: number) => {\n setShowDeleteModal(true);\n setAsset(item);\n };\n\n const addAsset = (asset: any) => {\n setLoading(true);\n (currentPage?.api || apis)\n .post(transactionFeature.apiUrl, {\n ...asset,\n operation: \"V\",\n avgSellPrice: asset.price,\n totalValue: asset.price * asset.quantity,\n })\n .then(() => {\n getTaxes(true, false, false, true, true);\n getYearResume(year, true);\n })\n .catch(() => message.error(errorMessage))\n .finally(() => {\n setLoading(false);\n handleCloseAssetModal();\n });\n };\n\n const editAsset = (asset: any) => {\n setLoading(true);\n (currentPage?.api || apis)\n .put(`${transactionFeature.apiUrl}/${asset.id}`, {\n ...asset,\n operation: \"V\",\n avgSellPrice: asset.price,\n totalValue: asset.price * asset.quantity,\n })\n .then(() => {\n getTaxes(true, false, false, true, true);\n getYearResume(year, true);\n })\n .catch(() => message.error(errorMessage))\n .finally(() => {\n setLoading(false);\n handleCloseAssetModal();\n });\n };\n\n const removeAsset = () => {\n setLoading(true);\n (currentPage?.api || apis)\n .delete(`${transactionFeature.apiUrl}/${asset.id ?? asset._id}`, {\n params: {\n year,\n month,\n },\n })\n .then(() => {\n getTaxes(true, false, false, true, true);\n getYearResume(year, true);\n })\n .catch(() => message.error(errorMessage))\n .finally(() => {\n setLoading(false);\n handleCloseDeleteModal();\n });\n };\n\n /* Manipular as tabs de imposto no mobile */\n const [tab, setTab] = useState(1);\n const bolsaOperationsRef = useRef(null);\n\n const handleNext = () => {\n setTab((tab) => tab + 1);\n const span = document.querySelector(\"div.sub-header > span.show\");\n bolsaOperationsRef.current?.scrollBy({ left: span?.clientWidth ?? 155 });\n };\n\n const handlePrev = () => {\n setTab((tab) => tab - 1);\n const span = document.querySelector(\"div.sub-header > span.show\");\n bolsaOperationsRef.current?.scrollBy({ left: -(span?.clientWidth ?? 155) });\n };\n /* Fim da manipulação das tabs de imposto no mobile */\n\n /* Editar prejuízo acumulado */\n const [resetedPreju, setResetedPreju] = useState(false);\n const [showPrejuModal, setShowPrejuModal] = useState(false);\n const [showCorretagemModal, setShowCorretagemModal] = useState(false);\n const [loadingResetPreju] = useState(false);\n const [loadingEditPreju, setLoadingEditPreju] = useState(false);\n const [ignoreZeroBuyPrice, setIgnoreZeroBuyPrice] = useState(false);\n const [dataZeroBuyPrice, setDataZeroBuyPrice] = useState([]);\n const [modalZeroBuyPrice, setModalZeroBuyPrice] = useState(false);\n\n const preju = useMemo(\n () => ({\n userPrejuizoComum: resetedPreju\n ? data.prejuizoComum\n : data.userPrejuizoComum ?? data.prejuizoComum,\n userPrejuizoDayTrade: resetedPreju\n ? data.prejuizoDayTrade\n : data.userPrejuizoDayTrade ?? data.prejuizoDayTrade,\n userPrejuizoFII: resetedPreju\n ? data.prejuizoFII\n : data.userPrejuizoFII ?? data.prejuizoFII,\n }),\n [data, resetedPreju]\n );\n\n const custoCorretagem = useMemo(\n () => ({\n userCustoCorretagemComum: data.userCustoCorretagemComum || 0,\n userCustoCorretagemDT: data.userCustoCorretagemDT || 0,\n userCustoCorretagemFII: data.userCustoCorretagemFII || 0,\n }),\n [data]\n );\n\n // const handleResetPreju = () => {\n // setResetedPreju(true);\n // };\n\n // const handleResetPreju = () => {\n // setLoadingResetPreju(true);\n // (currentPage?.api || apis)\n // .put(`${prejuizoFeature.apiUrl}/update-all?month=${month + 1}&year=${year}`)\n // .then(() => {\n // message.success('Prejuízos resetados com sucesso.');\n // getTaxes(false, false, false, true, true);\n // })\n // .catch(() => {\n // message.error(\"Não foi possível resetar os valores\");\n // })\n // .finally(() => {\n // setLoadingResetPreju(false);\n // handleClosePrejuModal();\n // });\n // };\n\n const handleEditPreju = () => {\n setResetedPreju(false);\n setShowPrejuModal(true);\n };\n\n const handleEditPrejuData = (preju: any) => {\n setDataToEditPreju(preju);\n setShowEditPrejuConfirmation(true);\n };\n\n const handleEditCorretagem = () => {\n setShowCorretagemModal(true);\n };\n\n const handleClosePrejuModal = () => {\n setShowPrejuModal(false);\n };\n\n const handleCloseCorretagemModal = () => {\n setShowCorretagemModal(false);\n };\n\n const editPreju = () => {\n setLoadingEditPreju(true);\n setShowEditPrejuConfirmation(false);\n\n (currentPage?.api || apis)\n .put(\n `${prejuizoFeature.apiUrl}?month=${month + 1}&year=${year}`,\n dataToEditPreju\n )\n .then((res) => {\n message.success(\"Dados salvos\");\n getTaxes(true, true, false, true);\n setLoadingEditPreju(false);\n handleClosePrejuModal();\n })\n .catch(() => {\n message.error(errorMessage);\n setLoadingEditPreju(false);\n handleClosePrejuModal();\n });\n };\n\n const editCorretagem = (data: any) => {\n handleCloseCorretagemModal();\n (currentPage?.api || apis)\n .put(\n `/warren/custoCorretagem?month=${month + 1}&year=${year}`,\n data\n )\n .then((res) => {\n message.success(\"Dados salvos\");\n getTaxes(true, true, false, true, false, true);\n })\n .catch(() => {\n message.error(errorMessage);\n });\n };\n /* Fim do editar prejuízo acumulado */\n\n /* Editar IRRF */\n const [showIrrfModal, setShowIrrfModal] = useState(false);\n\n const irrf = useMemo(\n () => ({\n userIrrfComum: data.irrfComum,\n userIrrfDayTrade: data.irrfDayTrade,\n userIrrfFII: data.irrfFII,\n userIrrfTotal: data.irrfTotal,\n }),\n [data]\n );\n\n // const handleResetIrrf = () => {\n // setResetedIrrf(true);\n // };\n\n const handleEditIrrf = () => {\n setShowIrrfModal(true);\n };\n\n const handleCloseIrrfModal = () => {\n setShowIrrfModal(false);\n };\n\n const editIrrf = (data: any) => {\n handleCloseIrrfModal();\n (currentPage?.api || apis)\n .put(`${irrfFeature.apiUrl}?month=${month + 1}&year=${year}`, data)\n .then((response) => {\n message.success(\"Dados salvos\");\n getTaxes(true, true, false, true);\n })\n .catch(() => {\n message.error(errorMessage);\n });\n };\n /* Fim do editar prejuízo acumulado */\n\n useEffect(() => {\n // if (month === 0) {\n getTaxes(true, true, false, true, false);\n // } else {\n // getTaxes(false, false, false, false, true);\n // }\n }, [getTaxes, month, year]);\n\n useEffect(() => {\n if (view && darf) {\n getDarf(darf.id);\n }\n }, [getDarf, view, darf]);\n\n useEffect(() => {\n getYearResume(year);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n useEffect(() => {\n const dataToPayment = {\n ...data,\n year,\n month: month + 1,\n name: user.user.name,\n memoriaCalculo: [],\n transactions: data.memoriaCalculo,\n };\n\n setPaymentData(dataToPayment);\n }, [data.impostoTotalFinal, data, month, user.user.name, year]);\n\n useEffect(() => {\n if (data?.memoriaCalculo?.length > 0) sethasItems(!hideOnBasic);\n setBuffer(data?.buffer?.data);\n }, [data, hideOnBasic]);\n\n const handlePlanModal = () => {\n navigate('/planos')\n };\n\n useEffect(()=> {\n if (ignoreZeroBuyPrice) handleEventDarfButton();\n }, [ignoreZeroBuyPrice])\n\n const handleEventDarfButton = async () => {\n setLoadingDarfButton(true);\n let retPlanInfo = {\n data: {\n active: false,\n },\n };\n try {\n retPlanInfo = await apiPayment.get(\"/user-plan/plan-info\");\n } catch (err) {\n console.error(err);\n }\n setLoadingDarfButton(false);\n\n if (!ignoreZeroBuyPrice) {\n const hasZeroBuyPrice = await getZeroBuyPrice(data?.memoriaCalculo);\n if (hasZeroBuyPrice.length > 0) {\n setModalZeroBuyPrice(true);\n setDataZeroBuyPrice(hasZeroBuyPrice);\n return;\n }\n }\n\n //@ts-ignore\n !retPlanInfo.data.active && !retPlanInfo.data.cupomDueDate\n ? handlePlanModal()\n : handleOpenPaymentModal();\n };\n\n const handleSaveMonth = async () => {\n setLoadingDarfButton(true);\n try {\n const retPlanInfo = await apiPayment.get(\"/user-plan/plan-info\");\n\n retPlanInfo.data.active || hasPermissionGeneral\n ? handleDarf(true)\n : handlePlanModal();\n } catch (err) {\n console.error(err);\n }\n setLoadingDarfButton(false);\n };\n\n const downloadAction = (record: any) => {\n apiBolsa.get(`/darf/download/${record._id}`).then((res) => {\n if (res.data.url && !isMobile) {\n download(res.data.url);\n }\n });\n };\n\n const calcReturn = (\n sellPrice: number,\n buyPrice: number,\n quantity: number,\n fee: number\n ) => {\n const qtt = quantity ? quantity : 1;\n const ret =\n sellPrice && buyPrice\n ? (sellPrice * qtt - (fee || 0)) / (buyPrice * qtt) - 1\n : 0;\n return (\n = 0\n ? \"var(--velotax-green-amount)\"\n : \"var(--velotax-red-prejuizo)\"\n }`,\n }}\n >\n {ret ? numberToPercentageWallet(ret) : \"-\"}\n \n );\n };\n\n const disableCurrentMonth = (item: any) =>\n item.year === currentYear && item.month === currentMonth;\n\n return (\n \n \n \n \n \n \n \n {!view && !viewEdit && (\n <>\n {currentBroker.initialYear === year && month === 0 ? (\n
\n ) : (\n {\n if (month === 0) {\n setYear((year) => {\n const newYear = year - 1;\n getYearResume(newYear);\n setDatePickerYear(newYear);\n return newYear;\n });\n setMonth(11);\n } else {\n setMonth((month) => month - 1);\n }\n }}\n />\n )}\n >\n )}\n {monthsExtended[month]} de {year}\n {!view && !viewEdit && (\n <>\n {currentYear === year && currentMonth - 1 === month ? (\n
\n ) : (\n {\n if (month === 11) {\n setYear((year) => {\n const newYear = year + 1;\n getYearResume(newYear);\n setDatePickerYear(newYear);\n return newYear;\n });\n setMonth(0);\n } else {\n setMonth((month) => month + 1);\n }\n }}\n />\n )}\n >\n )}\n \n \n {!view && !viewEdit && (\n \n }\n defaultValue={defaultValue}\n locale={antDatePickerLocale}\n value={moment({ year: datePickerYear, month })}\n onSelect={(e) => {\n if (e.month() === month && year !== datePickerYear) {\n onChangeMonth(e, `${month + 1}-${datePickerYear}`);\n }\n }}\n onPanelChange={(e) => {\n getYearResume(e.year());\n setDatePickerYear(e.year());\n }}\n getPopupContainer={(trigger) => trigger.parentElement!}\n renderExtraFooter={() =>\n yearResumeFeature.disabled ? (\n <>>\n ) : (\n \n )\n }\n disabledDate={(current) =>\n current &&\n (current > currentDate.endOf(\"month\") ||\n current < moment(`${currentBroker.initialYear}-01-01`) ||\n current < moment(`2020-01-01`))\n }\n monthCellRender={(e) => (\n \n )}\n />\n \n )}\n
\n \n \n\n \n \n {hasPlan && view && darf?.fullPath && (\n \n \n {\n downloadAction(darf)\n }}\n startIcon={\n \n }\n >\n BAIXAR DARF\n \n \n
\n )}\n \n {(!hasPlan || hideOnBasic) && (\n \n \n Contrate o plano premium para ver os impostos devidos\n \n }\n onClick={() => {\n handlePlanModal();\n closeModal?.();\n }}\n >\n PREMIUM\n \n
\n )}\n {DarfBolsaTabs(data, {\n handleEditPreju,\n handleEditCorretagem,\n view,\n month,\n year,\n }).map((container, rowIndex) => (\n \n {container.tabs.map((item, colIndex) => (\n
0 && colIndex > 0,\n }\n )}\n >\n {rowIndex === 0 && (\n }\n onClick={\n !hasPlan || colIndex === 0 || tab === 1\n ? undefined\n : handlePrev\n }\n style={{\n visibility:\n !hasPlan || colIndex === 0 || tab === 1\n ? \"hidden\"\n : \"visible\",\n }}\n />\n )}\n {!hasPlan && rowIndex > 0 && colIndex > 0\n ? formatCurrency(13443)\n : isMobile() && !hasPlan && rowIndex === 0\n ? \"\"\n : item.content}\n {rowIndex === 0 && (\n }\n onClick={\n !hasPlan || colIndex === 0 || tab === 3\n ? undefined\n : handleNext\n }\n style={{\n visibility:\n !hasPlan || colIndex === 0 || tab === 3\n ? \"hidden\"\n : \"visible\",\n }}\n />\n )}\n \n ))}\n
\n ))}\n \n\n {/* \n \n {view ? : }\n \n {view\n ? \"Exibir taxas de corretagem\"\n : \"Inserir soma das taxas de corretagem (opcional)\"}\n \n \n {!view && (\n setHelpModal(CorretagemModalProps)}>\n \n \n )}\n
*/}\n\n \n\n \n \n Vendas realizadas no mês\n \n }\n >\n {hasItems && (\n handleDownloadSheet(buffer, \"Vendas\")}\n style={{\n alignItems: \"center\",\n fontSize: \"13px\",\n paddingRight: \"0\",\n justifyContent: \"end\",\n backgroundColor: \"transparent\",\n color: \"#e02b57\",\n }}\n startIcon={\n \n }\n >\n Baixar planilha\n \n )}\n {!view && !hideOnBasic && (\n }\n >\n Adicionar\n \n )}\n \n new Date(b.originalDate).getTime() -\n new Date(a.originalDate).getTime()\n )}\n renderItem={(item, index) => (\n handleEdit(item, index)}\n icon={ }\n />,\n handleRemove(item, index)}\n icon={ }\n />,\n ]\n }\n >\n \n {item.code}}\n description={\n \n
\n Data:{\" \"}\n {moment(item.date, [\n \"DD/MM/YYYY\",\n \"YYYY-MM-DD\",\n ]).format(\"DD/MM/YYYY\")}\n
\n
Quantidade: {item.quantity}
\n {view && (\n <>\n
\n Custo médio de compra:{\" \"}\n \n {formatCurrency(item.buyPrice)}\n \n
\n
\n Preço de venda:{\" \"}\n \n {formatCurrency(item.sellPrice)}\n \n
\n >\n )}\n
\n Ganho:{\" \"}\n \n {formatCurrency(item.capitalGain)}\n \n
\n
\n Retorno:{\" \"}\n {calcReturn(\n item?.sellPrice,\n item?.buyPrice,\n item?.quantity,\n item?.fee\n )}\n
\n
\n }\n />\n \n \n )}\n />\n
\n \n\n {hasPlan && }\n\n {hasPlan && (\n \n Imposto total \n \n {loading ? (\n \n ) : (\n formatCurrency(impostoCharge)\n )}\n \n
\n )}\n\n {hasPlan && (\n \n {DarfResultDescriptions.map((description) => (\n \n {loading ? (\n \n ) : description.Component ? (\n \n ) : (\n formatCurrency(\n Number(data[description.id as keyof IDarfBolsa] || 0)\n )\n )}\n \n ))}\n \n )}\n\n {!view && !hideOnBasic && (\n <>\n \n {!(month === currentMonth && year === currentYear) &&\n valorPrincipal < minDarfPrice &&\n data?.memoriaCalculo?.length > 0 && (\n \n \n \n O valor do imposto devido é menor que{\" \"}\n {formatCurrency(minDarfPrice)}. Você não precisa\n emitir um DARF para este mês.\n >\n }\n >\n }\n // onClick={() => {\n // handleRegularize(true);\n // }}\n onClick={handleSaveMonth}\n >\n Salvar sem emitir DARF\n \n \n \n )}\n {valorPrincipal >= minDarfPrice && (\n \n \n target.parentElement!}\n title={\n thirdBusinessDayAndMonth(date).title\n }\n >\n handleEventDarfButton()}\n startIcon={\n (emitting || loadingDarfButton) && \n }\n disabled={\n loading ||\n emitting ||\n loadingDarfButton ||\n thirdBusinessDayAndMonth(date).disabled\n }\n >\n {handleLabelButton}\n {\" \"}\n \n \n )}\n
\n {!(month === currentMonth && year === currentYear) && (\n \n \n }\n >\n Salvar apenas\n \n \n
\n )}\n >\n )}\n\n {!view && (\n \n
\n {!hideOldPositionsText && (\n
\n Você tinha investimentos em\n 31/12/2019 e ainda não preencheu o seu custo médio?{\" \"}\n \n navigate(`/${currentBroker.path}/bolsa-insert-manual`)\n }\n style={{\n cursor: \"pointer\",\n color: \"var(--ant-primary-color)\",\n }}\n >\n Clique aqui.\n \n
\n )}\n
\n {/*
\n
\n ({hideOldPositionsText ? 1 : 2}) A Receita Federal admite a\n dedução dos custos de corretagem na apuração do ganho líquido\n em operações de renda variável. Entretanto, não há\n obrigatoriedade do usufruto do benefício da dedução por parte\n dos investidores (Instrução Normativa RFB nº 1.585, de 31 de\n agosto de 2015, art. 56, § 3º). As informações advindas\n somente da integração com a B3 não consideram o benefício da\n dedução dos custos de corretagem da Warren, incluindo somente as\n tarifas de liquidação, registro, termo/opções e emolumentos da\n B3. Caso deseje descontar os seus custos de corretagem dos\n ganhos tributáveis, você pode adicionar manualmente as taxas\n de corretagem em cada operação no campo “Taxas” ou carregar\n suas\n \n navigate(\n item.settings?.integrationPath ??\n `/${currentBroker.path}/integration`\n )\n }\n style={{\n cursor: \"pointer\",\n color: \"var(--ant-primary-color)\",\n }}\n >\n {\" \"}\n notas de corretagem.\n \n
\n
*/}\n {/*
*/}\n {/*
\n ({hideOldPositionsText ? 2 : 3}) As taxas apresentadas incluem\n somente as tarifas de liquidação, registro, termo/opções e\n emolumentos da B3. Caso deseje, você pode adicionar manualmente as\n taxas de corretagem em cada operação.\n
*/}\n {/*
*/}\n {/*
\n
\n ({hideOldPositionsText ? 2 : 3}) Para preencher\n automaticamente os seus prejuízos acumulados, você deve salvar\n os meses na ordem do mais antigo para o mais recente.\n
\n
*/}\n {/*
\n
\n ({hideOldPositionsText ? 3 : 4}) O Usuário é\n exclusivamente responsável pela conferência e validação\n das informações utilizadas na apuração do imposto devido,\n conforme{\" \"}\n \n Termos e Condições Gerais de Uso.\n \n
\n
*/}\n
\n )}\n \n \n \n {helpModal?.content}\n \n\n { }}\n rows={[]}\n itemToEdit={null}\n visibility={showEditPrejuConfirmation}\n saveButtonText={\"Sim\"}\n title={\"Editar prejuízos\"}\n onCancel={() => setShowEditPrejuConfirmation(false)}\n >\n \n Você tem certeza que deseja editar os valores calculados de prejuízos\n acumulados?\n
\n \n\n \n {darfModal?.content}\n \n \n { }}\n edit={(data: any) => handleEditPrejuData(data)}\n itemToEdit={preju}\n loading={loadingEditPreju}\n disableButtons={loadingResetPreju || loadingEditPreju}\n rows={PrejuFormItemRows(loadingResetPreju || loadingEditPreju)}\n visibility={showPrejuModal}\n onCancel={handleClosePrejuModal}\n title=\"Editar prejuízos a compensar\"\n // preFormChildren={\n // }\n // style={{ display: \"flex\", margin: \"0 0 16px auto\" }}\n // >\n // Resetar valores\n // \n // }\n />\n\n { }}\n edit={editCorretagem}\n itemToEdit={custoCorretagem}\n rows={CorretagemFormItemRows(!!view)}\n visibility={showCorretagemModal}\n onCancel={handleCloseCorretagemModal}\n title=\"Taxas de corretagem\"\n hideFooter={view}\n >\n {/* \n \n \n \n Caso altere algum desses valores, você pode voltar aos valores\n calculados automaticamente pelo sistema (clique em aplicar e\n salvar):\n \n \n \n \n Prejuízo Comum:\n {formatCurrency(data.prejuizoComum ?? 0)}\n Prejuízo Day Trade:\n {formatCurrency(data.prejuizoDayTrade ?? 0)}\n Prejuízo FII:\n {formatCurrency(data.prejuizoFII ?? 0)}\n
\n \n \n \n Aplicar\n \n \n
*/}\n \n { }}\n edit={editIrrf}\n itemToEdit={irrf}\n rows={IrrfFormItemRows}\n visibility={showIrrfModal}\n onCancel={handleCloseIrrfModal}\n title=\"Editar IRRF\"\n >\n {/* \n \n \n \n Caso altere algum desses valores, você pode voltar aos valores\n calculados automaticamente pelo sistema (clique em aplicar e\n salvar):\n \n \n \n \n IRRF Comum:\n {formatCurrency(data.irrfComum ?? 0)}\n IRRF Day Trade:\n {formatCurrency(data.irrfDayTrade ?? 0)}\n IRRF FII:\n {formatCurrency(data.irrfFII ?? 0)}\n IRRF Total:\n {formatCurrency(data.irrfTotal ?? 0)}\n
\n \n \n \n Aplicar\n \n \n
*/}\n \n maxPixPayment}\n paymentData={{ ...paymentData, impostoTotalFinal: impostoCharge }}\n disableCreditCardOption={\n (data?.impostoTotalFinal || 0) > maxCreditCardPayment\n }\n callDarf={() => {\n HandleTag(\"64\");\n handleOpenDarfModal();\n }}\n />\n Promise.resolve(removeAsset())}\n body=\"Você realmente quer deletar esse ativo?\"\n />\n setShowNotAuthorizedModal(false)}\n />\n {!view && !viewEdit && (\n \n )}\n \n \n \n );\n};\n","import { Modal } from \"antd\";\nimport styled from \"styled-components\";\n\nexport const Container = styled.div`\n display: flex;\n align-items: center;\n flex-direction: column;\n padding: 0 32px 128px;\n @media only screen and (max-device-width: 812px) {\n padding: 0 16px 128px;\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n min-width: 40vw;\n max-width: 500px;\n border-radius: 16px;\n background-color: var(--velotax-background-color);\n\n h2 {\n font-size: 24px;\n position: relative;\n color: var(--velotax-font-color);\n &::before {\n content: \"\";\n position: absolute;\n width: 4px;\n top: 0;\n left: -32px;\n height: 100%;\n background-color: var(--ant-primary-color);\n }\n }\n h3 {\n font-size: 20px;\n display: flex;\n align-items: center;\n column-gap: 8px;\n color: var(--velotax-font-color);\n }\n\n .ant-list.ant-list-split {\n margin-top: 24px;\n padding: 0 0 0 16px;\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-ghost);\n border-radius: 8px;\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color);\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n }\n .list-description {\n display: flex;\n flex-direction: column;\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px;\n h2 {\n &::before {\n left: -24px;\n }\n }\n }\n`;\n\nexport const ConfirmModal = styled(Modal)`\n .p {\n font-size: 16px;\n strong {\n font-weight: 500;\n color: var(--ant-primary-color);\n }\n }\n`;\n","import { Typography } from \"antd\";\nimport { RiMoneyDollarCircleLine, RiFileSearchLine } from \"react-icons/ri\";\nimport { YearReport } from \"../../constants/rendimentos\";\n\nexport const cards = [\n // {\n // id: \"irpf\",\n // icon: (\n // \n // ),\n // content: (\n // \n // Declaração IRPF\n // \n // 2023 no Velotax\n // \n // ),\n // },\n {\n id: \"anual-report\",\n icon: (\n \n ),\n content: RELATÓRIOS ANUAIS ,\n description: (\n \n Relatórios auxiliares para preenchimento da DIRPF\n \n ),\n },\n {\n id: \"gain\",\n icon: (\n \n ),\n content: INFORMES DE GANHO DE CAPITAL ,\n description: (\n \n Relatórios auxiliares para acompanhamento do ganho de capital (GCAP)\n \n ),\n },\n];\n","import styled from \"styled-components\";\n\nexport const ReportContainer = styled.div`\n height: 100%;\n display: flex;\n padding: 64px 32px;\n\n & > .content {\n margin: 0 auto;\n width: 700px;\n }\n\n h1 {\n width: 700px;\n margin: 0 auto 1.5rem;\n font-weight: 400;\n font-size: 2rem;\n position: relative;\n line-height: 2.5rem;\n padding-bottom: 1rem;\n color: var(--velotax-font-color-light);\n\n :after {\n content: \"\";\n position: absolute;\n left: 0;\n bottom: 0;\n height: 2px;\n width: 100px;\n background-color: var(--ant-primary-color);\n }\n }\n\n h3 {\n display: flex;\n align-items: center;\n column-gap: 16px;\n color: var(--velotax-font-color-light);\n margin-bottom: 0;\n }\n\n div.ant-typography {\n font-size: 1rem;\n line-height: 1.5rem;\n color: var(--velotax-font-color);\n }\n\n .ant-typography.destak {\n margin: 48px 0;\n display: block;\n font-weight: 700;\n font-size: 1rem;\n line-height: 1.5rem;\n color: var(--velotax-font-color);\n padding-left: 16px;\n border-left: 4px solid var(--ant-primary-color);\n }\n\n .cards {\n margin: 32px 0 0;\n padding-bottom: 32px;\n width: 100%;\n display: grid;\n row-gap: 1rem;\n column-gap: 1rem;\n grid-template-columns: repeat(2, 1fr);\n\n & > div {\n width: 100%;\n }\n\n .card-antd {\n position: relative;\n\n a {\n top: 0;\n left: 0;\n position: absolute;\n width: 100%;\n height: 100%;\n color: var(--velotax-font-color);\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n height: auto;\n padding: 24px 0 0;\n\n & > .content {\n width: 100%;\n h1 {\n width: 100%;\n padding: 0 24px 12px;\n margin: 0 auto 1.5rem;\n :after {\n left: 24px;\n }\n }\n\n & > div.ant-typography {\n padding: 0 24px;\n }\n }\n\n .cards {\n width: 100%;\n padding: 0 24px;\n margin: 32px 0;\n grid-template-columns: repeat(2, 1fr);\n\n & > div {\n width: auto;\n height: 240px;\n }\n }\n }\n`;\n\nexport const Container = styled.div`\n h1 {\n width: 700px;\n margin: 0 auto 1.5rem;\n font-weight: 400;\n font-size: 2rem;\n position: relative;\n line-height: 2.5rem;\n padding-bottom: 1rem;\n color: var(--velotax-font-color-light);\n\n :after {\n content: \"\";\n position: absolute;\n left: 0;\n bottom: 0;\n height: 2px;\n width: 100px;\n background-color: var(--ant-primary-color);\n }\n }\n\n h2.ant-typography {\n display: flex;\n align-items: center;\n column-gap: 8px;\n width: 700px;\n margin: 0 auto 1.5rem;\n font-size: 1rem;\n font-weight: 400;\n line-height: 1.5rem;\n color: var(--velotax-font-color-light);\n\n &.info {\n align-items: flex-start;\n font-size: 0.875rem;\n line-height: 1.125rem;\n opacity: 0.85;\n margin: 0 auto 2.5rem;\n svg {\n margin-top: 4px;\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n\n h1 {\n width: calc(100% - 48px);\n margin: 0 24px 1.5rem;\n padding: 24px 0 1rem;\n }\n\n h2.ant-typography {\n width: 100%;\n margin: 0 auto 1.5rem;\n padding: 0 24px;\n }\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n\n .ant-list.ant-list-split {\n margin: 24px 0 0;\n padding: 0 16px;\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-ghost);\n border-radius: 4px;\n .ant-list-item-action > li {\n padding: 0;\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n .ant-list-item-action > li > button {\n font-size: 16px;\n color: var(--ant-primary-color);\n }\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n font-size: 16px;\n color: var(--velotax-font-color-light);\n }\n .ant-list-item-action-split {\n opacity: 0;\n }\n }\n .historic-month-status {\n font-weight: 500;\n padding: 4px 8px;\n border-radius: 0;\n color: var(--black);\n display: flex;\n &.payed,\n &.regular {\n background-color: var(--ant-success-color);\n }\n &.pending {\n background-color: var(--ant-error-color);\n }\n &.not-payed {\n background-color: var(--ant-warning-color);\n }\n }\n .ant-btn.ant-btn-text {\n color: var(--ant-primary-color);\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px 24px 32px;\n border-radius: 0;\n width: 100%;\n .ant-list.ant-list-split {\n .ant-list-item-action {\n align-items: end;\n grid-template-rows: 1fr 0.75fr;\n grid-template-columns: 1fr 1fr;\n }\n }\n }\n`;\n","import { RiDownloadCloudLine } from \"react-icons/ri\";\nimport { Button as MaterialButton } from \"@mui/material\";\nimport { Link, useLocation, useNavigate } from \"react-router-dom\";\nimport { useCallback, useEffect, useMemo, useState } from \"react\";\nimport { InfoCircleOutlined, LoadingOutlined } from \"@ant-design/icons\";\nimport {\n Col,\n List,\n message,\n Modal,\n Row,\n Select,\n Skeleton,\n Typography,\n} from \"antd\";\nimport { cards } from \"./items\";\nimport apis from \"../../services/apis\";\nimport { Card } from \"../Exterior/Cards\";\nimport Button from \"../../components/Button\";\nimport { Page } from \"../../constants/brokers\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport { UserPlanEnum } from \"../../constants/plans\";\nimport { useBroker } from \"../../contexts/BrokerContext\";\nimport { BackButton } from \"../../components/BackButton\";\nimport { ReportContainer, Container, Content } from \"./styles\";\nimport {\n download,\n errorMessage,\n getInitialYear,\n isMobile,\n monthsExtended,\n zero,\n} from \"../../utils\";\nimport api from \"../../services/apiBolsa\";\nimport { YearReport } from \"../../constants/rendimentos\";\nimport { DrawerModal } from \"../../components/DrawerModal\";\ninterface ReportProps {\n item: Page;\n}\n\nconst ReportAlertTitle = (\n \n {\" \"}\n \n Os relatórios estão disponíveis somente no plano premium.{\" \"}\n Contrate o seu plano. \n \n \n);\n\nexport const ReportBolsa: React.FC = ({ item }) => {\n const {\n user,\n showUserPlanModal,\n userPlanModal,\n hasPlan,\n ssoToken,\n hasPermissionGeneral,\n } = useAuth();\n const { state } = useLocation();\n const navigate = useNavigate();\n\n const today = new Date();\n const currentMonth = today.getMonth();\n const currentYear = today.getFullYear();\n const stateYear = (state as any)?.year;\n const initialYear = getInitialYear(currentYear, currentMonth);\n window.history.replaceState({}, document.title);\n\n const { currentPage } = useBroker();\n\n const [action, setAction] = useState();\n\n const [data, setData] = useState();\n const [infoModal, toggleModal] = useState(null);\n const [loading, setLoading] = useState(false);\n const [downloading, setDownloading] = useState(false);\n const [loadingLinks] = useState(false);\n const [year, setYear] = useState(stateYear ?? initialYear);\n\n const historicFeature = item.features[0];\n const downloadFeature = item.features[1];\n // const sendEmailFeature = item.features[2];\n\n const userPlanInfoStatus = useMemo(\n () => user.user?.userPlanInfoVelotax,\n [user.user?.userPlanInfoVelotax]\n );\n\n const yearOptions = Array.from(Array(initialYear - 2020 + 1))\n .map((y, i) => ({\n value: 2020 + i,\n label: `${2020 + i}`,\n }))\n .reverse();\n\n const onChangeYear = (value: string) => {\n setYear(parseInt(value));\n };\n\n const getHistoric = useCallback(() => {\n if (!action) return;\n if (action === \"anual-report\")\n return setData(\n Array.from(Array(new Date().getFullYear() - 2022))\n .map((y, i) => ({\n year: 2022 + i,\n label: `${2022 + i}`,\n }))\n .reverse()\n );\n\n setLoading(true);\n (currentPage?.api || apis)\n .get(historicFeature.apiUrl, { params: { year } })\n .then((response) => setData((response.data ?? []).reverse()))\n .catch(() => message.error(errorMessage))\n .finally(() => setLoading(false));\n }, [historicFeature, year, currentPage, action]);\n\n // const getDowloadLinks = useCallback(\n // (data: any[], action: string) => {\n // setLoadingLinks(true);\n // const dataLinks = data.map((item) =>\n // !!item._id\n // ? (currentPage?.api || apis).post(\n // `${downloadFeature.apiUrl}/${action}`,\n // {\n // id: item._id,\n // year: item.year,\n // month: item.month,\n // }\n // )\n // : Promise.resolve({ data: item })\n // );\n // Promise.allSettled(dataLinks).then((response) => {\n // setData(\n // response.map((item, index) =>\n // item.status === \"fulfilled\" && item.value.data?.file\n // ? {\n // ...data?.[index],\n // url: `data:application/pdf;base64,${item.value.data.file}`,\n // }\n // : data?.[index]\n // )\n // );\n // setLoadingLinks(false);\n // });\n // },\n // [currentPage, downloadFeature]\n // );\n\n // const sendDarfToEmail = (reportId: string) => {\n // (currentPage?.api || apis)\n // .get(`${sendEmailFeature.apiUrl}/${reportId}`)\n // .then(() => {\n // message.success(\"O DARF foi enviado para o seu e-mail\");\n // })\n // .catch((error) => {\n // console.log(error);\n // });\n // };\n\n const downloadAction = (record: any, Action?: string) => {\n const ACTION = Action || action;\n setDownloading(true);\n (currentPage?.api || apis)\n .post(`${downloadFeature.apiUrl}/${ACTION}`, {\n from: \"warren\",\n id: record._id,\n month: record.month,\n year: record.year,\n })\n .then((res) => {\n if (res.data.fileUrl && !isMobile()) {\n download(res.data.fileUrl);\n }\n if (res.data.file) {\n // downloadPDF(res.data.file);\n }\n message.success(\"Relatório enviado para o seu e-mail\");\n })\n .catch(({ response }) =>\n message.error(response?.data?.message ?? errorMessage)\n )\n .finally(() => {\n setDownloading(false);\n toggleModal(null);\n });\n };\n\n const checkProcessorAction = (record: any) => {\n setLoading(true);\n let showToast = false;\n toggleModal(record);\n const itvl = setInterval(() => {\n api\n .get(`${downloadFeature.apiUrl}/anual-report?year=${record.year}`)\n .then((res) => {\n const { data } = res;\n const { year, status } = data;\n const isProcessing = record.year === year && status === \"processing\";\n if (isProcessing && showToast) return;\n if (isProcessing && !showToast) {\n showToast = true;\n return message.success(\n \"O seu Informe de Rendimentos está em geração. Aguarde alguns minutos.\"\n );\n }\n setLoading(false);\n clearInterval(itvl);\n })\n .catch(({ response }) => {\n const { data, status } = response;\n if (status === 401)\n setTimeout(\n () => navigate(\"/warren/bolsa-integration\"),\n 2000\n );\n message.error(data?.message ?? errorMessage);\n setLoading(false);\n clearInterval(itvl);\n });\n }, 5000);\n };\n\n const handlePlanModal = () => {\n showUserPlanModal(!userPlanModal, ReportAlertTitle);\n };\n\n const handleActionButton = (id: string) => {\n if (!hasPlan) {\n handlePlanModal();\n } else {\n setAction(id);\n }\n };\n\n useEffect(() => {\n getHistoric();\n }, [getHistoric, year]);\n\n return (\n \n \n
Relatórios \n
\n Confira os relatórios{\" \"}\n {action !== \"anual-report\" ? \"auxiliares\" : \"anuais\"} para a\n declaração do seu IR na bolsa de valores.\n \n\n {!action && (\n
\n {cards.map((item) => (\n
\n ) : (\n item.icon\n )\n }\n onClick={() => !loading && handleActionButton(item.id)}\n >\n {item.id === \"irpf\" ? (\n <>\n
\n {\" \"}\n \n {item.content}\n >\n ) : (\n <>\n {item.content}\n {item.description && (\n
{item.description}
\n )}\n >\n )}\n \n ))}\n
\n )}\n {!action && (\n
\n )}\n {action && (\n
\n \n Selecione o ano: \n {action !== \"anual-report\" && (\n \n )}\n \n action === \"anual-report\" ||\n item.year === year ||\n item.year === currentYear\n ) ?? []\n }\n itemLayout=\"horizontal\"\n renderItem={(item, index) => {\n const show2024Report = localStorage.getItem('show2024Report') === 'show';\n if ((item.year ===2025) && action === \"anual-report\" && !show2024Report) return <>>\n // if ((item.year === 2024 || item.year ===2025) && action === \"anual-report\" && !show2024Report) return <>>\n return \n action === \"anual-report\"\n ? checkProcessorAction(item)\n : downloadAction(\n item\n ) /*sendDarfToEmail(item._id)*/\n }\n download={`Relatório-${item.year}-${zero(\n item.month\n )}.pdf`}\n >\n Baixar\n {loadingLinks ? (\n \n ) : (\n \n )}\n ,\n ]}\n >\n \n \n \n \n }}\n />\n handleActionButton(\"\")}\n style={{ marginTop: \"1rem\" }}\n >\n Voltar\n \n
\n \n )}\n
{\n toggleModal(null);\n }}\n footer={false}\n >\n \n \n {\n if (!loading && !downloading) {\n downloadAction(infoModal, \"anual-report\");\n }\n }}\n >\n {loading || downloading ? (\n <>\n \n {downloading ? \"Baixando\" : \"Gerando\"} informe\n >\n ) : isMobile() ? (\n \"Enviar informe por email\"\n ) : (\n \"Baixar informe\"\n )}\n \n \n \n {\n if (!loading && !downloading) {\n navigate(\"/warren/bolsa-historic\");\n }\n }}\n >\n Ir para calculadora de DARF\n \n \n
\n \n
\n \n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n display: flex;\n align-items: center;\n flex-direction: column;\n padding: 64px 32px 128px;\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n }\n`;\nexport const Content = styled.div`\n padding: 32px;\n min-width: 40vw;\n max-width: 500px;\n border-radius: 16px;\n background-color: var(--velotax-background-color);\n\n h2 {\n font-size: 24px;\n position: relative;\n &::before {\n content: \"\";\n position: absolute;\n width: 4px;\n top: 0;\n left: -32px;\n height: 100%;\n background-color: var(--ant-primary-color);\n }\n }\n .last-update {\n letter-spacing: 1px;\n }\n div.ant-typography {\n font-size: 18px;\n }\n .ant-typography {\n color: var(--velotax-font-color);\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px;\n border-radius: 0;\n min-height: calc(100vh - 64px);\n h2 {\n &::before {\n left: -24px;\n }\n }\n }\n`;\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n .ant-list {\n padding: 0;\n border-radius: 4px;\n background-color: var(--velotax-background-color-ghost);\n .ant-list-item {\n padding: 16px;\n &.error-border {\n border: 1px solid var(--ant-error-color);\n }\n :first-child {\n border-radius: 4px 4px 0 0;\n }\n :last-child {\n border-radius: 0 0 4px 4px;\n }\n }\n .error-border + .error-border {\n border-top: none;\n }\n .ant-list-item-action > li {\n padding: 0;\n }\n }\n .ant-list-item-action > li,\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color-light);\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n\n .alert-icon {\n fill: var(--ant-error-color) !important;\n }\n\n .row-margin {\n margin-top: 24px;\n }\n`;\n","import { Button } from \"@mui/material\";\nimport { FaRegEdit } from \"react-icons/fa\";\nimport { Col, List, Row, Skeleton, Tooltip, Typography } from \"antd\";\nimport { Dispatch, SetStateAction, useEffect, useState } from \"react\";\nimport { AiOutlineDelete, AiOutlineInfoCircle } from \"react-icons/ai\";\nimport {\n LoadingOutlined,\n PlusCircleOutlined,\n DownloadOutlined,\n} from \"@ant-design/icons\";\nimport { Container } from \"./styles\";\nimport { formatCurrency } from \"../../../../utils\";\nimport AntButton from \"../../../../components/Button\";\nimport { FormModal } from \"../../../../components/FormModal\";\nimport { AddTransactionGOVFormItemRows } from \"../../../../constants/darfBolsa\";\n\nexport interface OldTransactionsProps {\n changed: boolean;\n loadingPut: boolean;\n loadingGet: boolean;\n setLoadingPut: Dispatch>;\n setLoadingGet: Dispatch>;\n oldPositionEdit: any;\n oldPositionList: any[];\n showAddPositionModal: boolean;\n downloadOldTransactionsExcel: () => Promise;\n setOldPositionEdit: Dispatch>;\n setShowAddPositionModal: Dispatch>;\n saveOldPositions: () => void;\n addOldPosition: (item: any) => void;\n editOldPosition: (item: any) => void;\n handleEditOldPosition: (item: any, index: number) => void;\n handleRemoveOldPosition: (item: any, index: number) => void;\n getOldTransactionsInit: () => void;\n}\n\nexport const OldTransactions: React.FC = ({\n loadingGet,\n loadingPut,\n oldPositionList,\n downloadOldTransactionsExcel,\n setOldPositionEdit,\n setShowAddPositionModal,\n handleEditOldPosition,\n handleRemoveOldPosition,\n addOldPosition,\n editOldPosition,\n showAddPositionModal,\n oldPositionEdit,\n}) => {\n const [hasLendingPosition, setHasLendingPosition] = useState(false);\n const [downloading, setDownloading] = useState(false);\n\n useEffect(() => {\n if (oldPositionEdit?._id || oldPositionEdit?.id) {\n if (oldPositionEdit?.lendingSide) {\n setHasLendingPosition(true);\n } else {\n setHasLendingPosition(false);\n }\n }\n }, [oldPositionEdit]);\n\n async function handleDownloadOldTransactions() {\n setDownloading(true);\n const file = await downloadOldTransactionsExcel();\n if (file) {\n const blob = new Blob([file], { type: \"xlsx\" });\n const element = document.createElement(\"a\");\n element.href = URL.createObjectURL(blob);\n element.download = \"carteira_31-12-19.xlsx\";\n document.body.appendChild(element);\n element.click();\n element.remove();\n }\n setDownloading(false);\n }\n\n return (\n \n \n \n \n Ativos em carteira\n \n \n \n \n {oldPositionList.length === 0 ? (\n
\n ) : (\n \n ) : (\n \n )\n }\n onClick={handleDownloadOldTransactions}>\n BAIXAR EXCEL\n \n )}\n {\n setOldPositionEdit({});\n setHasLendingPosition(false);\n setShowAddPositionModal(true);\n }}\n startIcon={\n loadingPut || loadingGet ? (\n \n ) : (\n \n )\n }>\n Adicionar Posição\n \n
\n \n \n !oldPosition.lendingSide\n ) ?? []\n }\n renderItem={(item, index) => (\n \n \n }\n type=\"text\"\n />\n ,\n ]\n : []),\n }\n onClick={() => handleEditOldPosition(item, index)}\n />,\n }\n onClick={() => handleRemoveOldPosition(item, index)}\n />,\n ]}>\n \n {item.code}}\n description={\n <>\n \n
\n Quantidade: {item.quantity} \n
\n
\n Custo médio:{\" \"}\n {formatCurrency(item.price ?? 0)} \n
\n
\n >\n }\n />\n \n \n )}\n />\n \n
\n \n \n Empréstimo de ativos \n \n \n \n ) : (\n \n )\n }\n onClick={() => {\n setOldPositionEdit({});\n setHasLendingPosition(true);\n setShowAddPositionModal(true);\n }}>\n Adicionar Posição\n \n \n \n oldPosition.lendingSide\n ) ?? []\n }\n renderItem={(item, index) => (\n \n \n }\n type=\"text\"\n />\n ,\n ]\n : []),\n }\n onClick={() => handleEditOldPosition(item, index)}\n />,\n }\n onClick={() => handleRemoveOldPosition(item, index)}\n />,\n ]}>\n \n {item.code}}\n description={\n <>\n \n
\n Quantidade: {item.quantity} \n
\n
\n Custo médio:{\" \"}\n {formatCurrency(item.price ?? 0)} \n
\n
\n >\n }\n />\n \n \n )}\n />\n \n
\n\n {\n setShowAddPositionModal(false);\n setHasLendingPosition(false);\n }}\n title={\n oldPositionEdit?._id || oldPositionEdit?.id\n ? \"Editar ativo\"\n : \"Adicionar ativo\"\n }\n itemToEdit={\n oldPositionEdit.id || oldPositionEdit._id\n ? oldPositionEdit\n : undefined\n }\n />\n \n );\n};\n","import { Typography } from \"antd\";\nimport { InfoCircleOutlined } from \"@ant-design/icons\";\nimport { BolsaManualData, Question } from \"./interfaces\";\nimport {\n OldTransactions,\n OldTransactionsProps,\n} from \"../Darf/Components/OldTransactions\";\n\nexport const validateBolsaManualData = (data: BolsaManualData) => {\n return true;\n};\n\nexport const questions: Question[] = [\n // {\n // id: \"possuiInvestimentosBolsa2019\",\n // title: \"Você tinha investimentos em 31/12/2019?\",\n // formItems: [\n // {\n // name: \"possuiInvestimentosBolsa2019\",\n // FormItem: () => (\n // \n // \n // Sim \n // Não \n // \n // \n // ),\n // },\n // ],\n // },\n {\n id: \"posicoesBolsa2019\",\n title: \"Complete sua carteira de 31/12/2019\",\n subTitle: (\n <>\n \n A B3 armazena dados somente a partir de 2020. Por isso, você precisará preencher manualmente o custo médio dos\n seus investimentos em 31/12/2019. Você pode encontrar essas informações na sua Declaração de Imposto de Renda de 2019/2020.\n >\n ),\n children: (\n \n Lembre-se de clicar em finalizar para salvar seus dados\n \n ),\n formItems: [\n {\n name: \"posicoesBolsa2019\",\n FormItem: ({ rest }) => (\n \n ),\n },\n ],\n },\n];\n","import styled from \"styled-components\";\n\nexport const BolsaManualContainer = styled.div`\n height: 100%;\n overflow-y: hidden;\n .btn-preencher-pos2019 {\n margin-bottom: 3rem;\n }\n`;\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n .ant-collapse {\n border: none;\n\n .ant-collapse-item {\n margin-bottom: 8px;\n border-radius: 4px;\n background-color: var(--velotax-ghost-white);\n border: 2px solid var(--velotax-ghost-white);\n &.ant-collapse-item-active {\n .ant-collapse-header {\n border-radius: 4px 4px 0 0;\n background-color: var(--velotax-background-color);\n .ant-typography {\n color: var(--ant-primary-color);\n button {\n svg {\n fill: var(--ant-primary-color);\n }\n }\n }\n }\n }\n &.ant-collapse-item-disabled {\n .ant-collapse-header {\n cursor: pointer;\n }\n }\n &.collapse-skip-btn {\n .ant-spin-nested-loading {\n width: 100%;\n }\n .ant-collapse-header {\n padding: 0;\n .collapse-skip-btn-content {\n padding: 24px 32px;\n }\n }\n }\n .ant-collapse-header {\n border-radius: 4px;\n padding: 24px 32px;\n background-color: var(--velotax-ghost-white);\n .ant-typography {\n width: 100%;\n display: flex;\n align-items: center;\n justify-content: space-between;\n font-size: 1rem;\n font-weight: bold;\n line-height: 32px;\n color: var(--velotax-font-color-dark);\n & > div {\n display: flex;\n align-items: center;\n }\n button {\n display: inline-flex;\n height: 32px !important;\n svg {\n width: 20px;\n height: 20px;\n fill: var(--velotax-font-color-dark);\n }\n }\n .opcao-recomendada {\n padding: 2px 4px;\n font-weight: 400;\n font-size: 0.75rem;\n line-height: 1rem;\n color: var(--white);\n white-space: nowrap;\n border-radius: 4px;\n display: flex;\n align-items: center;\n background-color: var(--ant-success-color);\n svg {\n margin-right: 4px;\n }\n }\n }\n .sub-title {\n font-size: 0.8125rem;\n color: var(--ant-primary-color);\n }\n }\n .ant-collapse-content {\n border: none;\n border-radius: 0 0 8px 8px;\n background-color: var(--velotax-background-color);\n .ant-collapse-content-box {\n padding: 16px 16px 32px;\n .ant-steps-item-title {\n margin-top: 0.25rem;\n /* font-size: 0.875rem; */\n line-height: 1.5rem;\n }\n .ant-btn {\n font-size: 14px;\n }\n }\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n .ant-collapse {\n .ant-collapse-item {\n &.collapse-skip-btn {\n .ant-collapse-header {\n .collapse-skip-btn-content {\n padding: 8px 16px;\n .ant-typography {\n font-size: 0.95rem;\n line-height: 1.25rem;\n flex-direction: column;\n align-items: flex-start;\n }\n .sub-title {\n margin-top: 4px;\n font-size: 0.75rem;\n }\n }\n }\n }\n .ant-collapse-header {\n padding: 8px 16px;\n .ant-typography {\n font-size: 0.95rem;\n line-height: 1.5rem;\n flex-direction: column;\n align-items: flex-start;\n }\n }\n }\n }\n }\n`;\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n .income-report-uploader {\n height: auto;\n display: block;\n margin-bottom: 16px;\n\n .ant-upload {\n width: 100%;\n height: 200px;\n\n .upload-text-icon {\n height: 100%;\n cursor: pointer;\n font-size: 20px;\n display: flex;\n align-items: center;\n flex-direction: column;\n justify-content: center;\n\n svg {\n fill: var(--ant-primary-color);\n }\n\n .upload-text-desc {\n font-size: 14px;\n text-align: center;\n }\n }\n\n .loading-container {\n height: 100%;\n display: flex;\n align-items: center;\n flex-direction: column;\n justify-content: center;\n\n .anticon {\n width: 40px;\n height: 40px;\n margin-bottom: 12px;\n\n svg {\n width: 100%;\n height: 100%;\n fill: var(--ant-primary-color);\n }\n }\n }\n }\n }\n`;\n","import React from \"react\";\nimport { Upload } from \"antd\";\nimport { UploadProps } from \"antd/lib/upload\";\nimport { LoadingOutlined } from \"@ant-design/icons\";\nimport { AiOutlineCloudUpload } from \"react-icons/ai\";\nimport { Container } from \"./styles\";\n\nexport interface IUploadProps extends UploadProps {\n loading?: boolean;\n descText?: string;\n customText?: string;\n tabIndex?:number;\n}\n\nconst UploadVelotax: React.FC = ({\n loading,\n descText,\n disabled,\n customText = \"Arraste ou clique\",\n ...props\n}) => {\n return (\n \n \n {loading ? (\n \n \n Isso pode demorar alguns minutos, aguarde \n
\n ) : (\n \n
\n {customText}\n {descText &&
{descText} }\n
\n )}\n \n \n );\n};\n\nexport default UploadVelotax;\n","import { message } from \"antd\";\nimport { RcFile } from \"antd/lib/upload\";\nimport { UploadRequestOption as RcCustomRequestOptions } from \"rc-upload/lib/interface\";\nimport {\n createContext,\n Dispatch,\n SetStateAction,\n useContext,\n useState\n} from \"react\";\nimport apiIrpf2023 from '../services/apiIrpf2023';\nimport { useOldTransactions } from \"./OldTransactionsContext\";\n\ninterface IUploadDecContext {\n loading: boolean;\n loadingSkip: boolean;\n confirmarModal: boolean;\n beforeUpload: (file: RcFile) => boolean;\n customRequest: (params: RcCustomRequestOptions) => void;\n setConfirmarModal: Dispatch>;\n}\n\nconst UploadDecContext = createContext({} as IUploadDecContext);\n\nexport const UploadDecProvider: React.FC = ({ children }) => {\n const [loading, setLoading] = useState(false);\n const [loadingSkip] = useState(false);\n const [confirmarModal, setConfirmarModal] = useState(false);\n const {getOldTransactionsInit} = useOldTransactions();\n\n const beforeUpload = (file: RcFile) => {\n const nameSplitted = file.name.split(\".\");\n const isDec = nameSplitted[nameSplitted.length - 1].toLowerCase() === \"dec\";\n\n if (!isDec) {\n message.error(\"Você só pode fazer upload de arquivos .dec\");\n }\n\n return isDec;\n };\n\n const customRequest = ({ file }: RcCustomRequestOptions) => {\n const formData = new FormData();\n formData.append(\"file\", file);\n setLoading(true);\n\n apiIrpf2023\n .post(\"/receitaFederal/processarDec2019IRPF\", formData)\n .then(async () => {\n message.success(\"Arquivo .dec importado com sucesso!\");\n let interval: NodeJS.Timeout | null = null;\n if (loading) {\n\n interval = setInterval( ()=>{\n\n apiIrpf2023.get(`/receitaFederal/checkStatusRecuperarDecIRPF`).then( ({data})=>{\n if(data?.processamantoDec2019){\n setLoading(false);\n getOldTransactionsInit();\n\n return clearInterval(interval as unknown as NodeJS.Timeout);\n }\n })\n\n },2000);\n }else {\n if (interval) {\n clearInterval(interval);\n }\n }\n setLoading(false);\n \n })\n .catch((err:any) => {\n const errorMessage =\n err.reponse?.message?.error ||\n \"Não foi possivel importar o arquivo .dec!\";\n message.error(errorMessage);\n setLoading(false);\n });\n };\n\n return (\n \n {children}\n \n );\n};\n\nexport const useUploadDec = () => useContext(UploadDecContext);","export default __webpack_public_path__ + \"static/media/govbr.16bc87af.svg\";","import styled from \"styled-components\";\nimport GovBrIcon from \"../../assets/govbr.svg\";\n\nexport const Container = styled.div`\n min-height: 100%;\n padding: 64px 32px;\n background-color: var(--velotax-ghost);\n\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n background-color: var(--velotax-background-color);\n }\n`;\n\nexport const Content = styled.div`\n padding: 48px;\n width: fit-content;\n max-width: 720px;\n margin: 0 auto;\n border-radius: 16px;\n border: 1px solid var(--velotax-background-color-ghost);\n background-color: var(--velotax-background-color);\n\n h1 {\n display: flex;\n align-items: center;\n margin: 0 0 24px;\n font-size: 22px;\n line-height: 30px;\n font-weight: bold;\n position: relative;\n color: var(--velotax-font-color);\n\n .gov-logo {\n width: 64px;\n height: 40px;\n margin-left: 8px;\n display: inline-block;\n background-size: contain;\n background-position: center;\n background-repeat: no-repeat;\n background-image: url(${GovBrIcon});\n }\n\n button {\n display: inline-flex;\n height: 32px !important;\n svg {\n width: 20px;\n height: 20px;\n }\n }\n\n &::before {\n content: \"\";\n top: 0;\n width: 4px;\n left: -48px;\n height: 100%;\n position: absolute;\n background-color: var(--ant-primary-color);\n }\n }\n\n h5 {\n margin-bottom: 24px;\n }\n\n a.ant-btn.ant-btn-primary.ant-btn-lg {\n padding-top: 6.4px !important;\n }\n\n div.ant-typography {\n font-size: 1rem;\n }\n\n .upload-dec-info {\n margin: 24px 0 16px;\n font-weight: 500;\n font-size: 1.125rem;\n line-height: 1.5rem;\n color: var(--ant-primary-color);\n svg {\n width: 20px;\n height: 20px;\n margin: 0 0 -2px 4px;\n fill: var(--ant-primary-color);\n }\n }\n\n .ant-divider {\n margin: 24px 0 0;\n }\n\n .skip-btn {\n padding: 0;\n margin: 0 0 -16px auto;\n svg {\n margin: 0;\n }\n }\n\n \n\n .ant-steps {\n margin-top: 24px;\n .ant-steps-item:nth-of-type(3),\n .ant-steps-item:nth-of-type(4) {\n .ant-steps-item-container {\n .ant-steps-item-content {\n padding-bottom: 48px;\n }\n }\n }\n \n .ant-steps-item {\n \n &.ant-steps-item-wait {\n .ant-steps-item-title {\n opacity: 0.8;\n }\n }\n &.ant-steps-item-active {\n .ant-steps-item-title {\n background-color: red;\n font-weight: 500;\n color: var(--velotax-font-color) !important;\n }\n }\n .ant-steps-item-container {\n .ant-steps-item-content {\n padding-bottom: 24px;\n .ant-steps-item-title {\n }\n }\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n width: 100%;\n border: none;\n border-radius: 0;\n padding: 32px 24px 96px;\n h1 {\n font-size: 1.2rem;\n letter-spacing: -0.35px;\n .gov-logo {\n width: 58px;\n height: 38px;\n margin-left: 6px;\n }\n button {\n margin-top: -4px;\n height: 40px !important;\n }\n }\n .skip-btn {\n bottom: 24px;\n font-size: 13px;\n position: fixed;\n margin: 0;\n }\n }\n`;\n\nexport const DecUploadContent = styled.div`\n display: flex;\n align-items: center;\n flex-direction: column;\n justify-content: center;\n .dec-icon {\n width: 48px;\n height: 48px;\n font-size: 24px;\n margin: 0 0 16px;\n border-radius: 8px;\n color: var(--ant-primary-color);\n border: 2px solid var(--ant-primary-color);\n display: flex;\n position: relative;\n align-items: center;\n flex-direction: column;\n justify-content: center;\n svg {\n margin-top: -4px;\n }\n &:before {\n content: \"\";\n height: 2px;\n bottom: 8px;\n position: absolute;\n width: calc(100% - 24px);\n background-color: var(--ant-primary-color);\n }\n }\n span.ant-typography {\n font-size: 16px;\n font-weight: 500;\n max-width: 320px;\n color: var(--ant-primary-color);\n }\n`;\n","import { createContext, useContext, useState } from \"react\";\n\nexport const today = new Date();\n\nconst year = today.getFullYear();\n\nconst anoCalendario = year - 1;\n\ninterface IYearContext {\n year: number;\n currentMonth: number;\n anoCalendario: number;\n changeYear: (year: number) => void;\n}\n\nconst YearContext = createContext({} as IYearContext);\n\nexport const YearProvider: React.FC = ({ children }) => {\n const [Year, setYear] = useState(year);\n const [AnoCalendario, setAnoCalendario] = useState(anoCalendario);\n\n const changeYear = (year: number) => {\n setYear(year);\n setAnoCalendario(year - 1);\n };\n\n return (\n \n {children}\n \n );\n};\n\nexport const useYear = () => useContext(YearContext);","import { useState } from \"react\";\nimport { Divider, Steps, Typography } from \"antd\";\nimport { AiOutlineArrowRight } from \"react-icons/ai\";\nimport { LoadingOutlined, QuestionCircleOutlined } from \"@ant-design/icons\";\nimport Button from \"../../components/Button\";\nimport { Container, Content } from \"./styles\";\nimport { useEcac } from \"../../contexts/EcacContext\";\nimport { useYear } from \"../../contexts/YearContext\";\nimport apiIrpf2023 from '../../services/apiIrpf2023';\nimport { useOldTransactions } from \"../../contexts/OldTransactionsContext\";\n\n\ninterface DecPrepreenchidaProps {\n successCallback?: () => void;\n setShowModalPrePreenchida: (p:boolean) => void;\n}\n\nexport const DecPrepreenchidaContent: React.FC = ({\n\n setShowModalPrePreenchida\n}) => {\n const [currentStep,setCurrentStep] = useState(0);\n const [loading, setLoading] = useState(true);\n\n const {getOldTransactionsInit} = useOldTransactions();\n \n const programLink = \n process.env.REACT_APP_NODE_ENV === \"development\"\n ? \"https://velotax-programas.s3.amazonaws.com/posicao-2019-dev/velotax.msi\"\n : \"https://velotax-programas.s3.amazonaws.com/posicao-2019/velotax.msi\";\n\n return (\n \n \n {\n setCurrentStep(1);\n setCurrentStep(2);\n setLoading(true);\n let interval: NodeJS.Timeout | null = null;\n if (loading) {\n\n interval = setInterval( ()=>{\n\n apiIrpf2023.get(`/receitaFederal/checkStatusRecuperarDecIRPF`).then( ({data})=>{\n if(data?.processamantoDec2019){\n setLoading(false);\n setShowModalPrePreenchida(false);\n\n getOldTransactionsInit();\n // window.location.reload();\n\n return clearInterval(interval as unknown as NodeJS.Timeout);\n }\n })\n\n },1000);\n }else {\n if (interval) {\n clearInterval(interval);\n }\n }\n }}\n >\n Clique aqui \n {\" \"}\n {\"para baixar o arquivo de conexão\"}\n >\n }\n />\n \n : ''}\n title={\n <>\n Execute o atalho Velotax no seu desktop, faça\n o login no gov.br e aguarde alguns minutos.\n >\n }\n />\n \n )\n};\n\nexport const DecPrepreenchida = () => {\n const { year } = useYear();\n const { loadingEcac, handleSkipEcacStep, clearEcacInterval } =\n useEcac();\n\n return (\n \n \n \n Conecte sua conta \n }\n />\n \n\n \n Você terá acesso a sua declaração de IRPF {year} pré-preenchida\n \n\n \n {\n clearEcacInterval();\n await handleSkipEcacStep();\n // updateUserEcac();\n }}\n loading={loadingEcac.skipStep}\n disabled={loadingEcac.postLogin}\n >\n Não utilizar o gov.br \n {!loadingEcac.skipStep && }\n \n \n \n );\n};\n","import { useState } from \"react\";\nimport { Collapse, Spin, Typography, } from \"antd\";\nimport { QuestionCircleOutlined } from \"@ant-design/icons\";\nimport { AiOutlineArrowUp, AiOutlineCheck } from \"react-icons/ai\";\nimport { Container } from \"./styles\";\nimport { DrawerModal, DrawerModalProps } from \"../../components/DrawerModal\";\nimport Button from \"../../components/Button\";\nimport UploadVelotax from \"../../components/UploadVelotax\";\nimport { useUploadDec } from \"../../contexts/UploadDecContext\";\nimport { DecUploadContent } from \"../DecPrepreenchida/styles\";\nimport { DecPrepreenchidaContent } from \"../DecPrepreenchida\";\nimport { UploadRequestOption as RcCustomRequestOptions } from \"rc-upload/lib/interface\";\ninterface PrePreenchidaModalProps extends DrawerModalProps {\n setShowModalPrePreenchida: (b: boolean) => void;\n}\n\nexport const PrePreenchidaModal: React.FC = ({\n visible,\n setShowModalPrePreenchida,\n ...props\n}) => {\n const [showModal, setShowModal] = useState(true);\n const [loadingSkip] = useState(false);\n const { loading, beforeUpload, customRequest } = useUploadDec();\n\n const onCancel = () => {\n setShowModal(false);\n };\n\n return (\n \n \n \n Você pode conectar a sua conta gov.br ou carregar o arquivo DEC da sua\n declaração de 2019/2020. Selecione uma das opções abaixo:\n \n\n \n \n \n \n Conecte sua conta gov.br\n \n }\n onClick={(e) => {\n e.stopPropagation();\n // openHelpDrawer();\n }}\n />\n
\n \n
\n Opção recomendada\n
\n \n }\n >\n \n \n \n \n\n \n \n Carregue arquivo .DEC referente ao IR de 2019/2020\n }\n onClick={(e) => {\n e.stopPropagation();\n }}\n />\n
\n \n }\n >\n ) => {\n customRequest(params);\n setShowModalPrePreenchida(false);\n }}\n >\n \n \n \n Arraste aqui sua declaração do ano passado (formato .DEC)\n \n \n \n \n \n \n \n );\n};\n","import styled, { css } from \"styled-components\";\n\nexport const Container = styled.div`\n height: calc(100vh - 64px);\n overflow: hidden;\n`;\n\nexport const QuestionContainer = styled.div<{ width?: string | number }>`\n display: flex;\n height: 100%;\n overflow-y: auto;\n align-items: center;\n flex-direction: column;\n h2 {\n font-size: 2rem;\n font-weight: 400;\n line-height: 2.5rem;\n position: relative;\n color: var(--velotax-text-color);\n margin: 64px auto 0;\n padding: 0 0 1rem;\n ${({ width }) =>\n width\n ? css`\n width: ${width} !important;\n `\n : \"\"}\n &::after {\n content: \"\";\n left: 0;\n bottom: 0;\n width: 100px;\n height: 2px;\n position: absolute;\n background-color: var(--ant-primary-color);\n }\n }\n\n @media only screen and (max-width: 812px) {\n padding: 32px 0 0;\n h2 {\n width: 100% !important;\n padding: 0 24px 12px;\n margin: 0;\n &::after {\n left: 24px;\n }\n }\n }\n`;\n\nexport const Question = styled.div<{ width?: string | number }>`\n height: fit-content;\n padding: 32px;\n margin: 24px 0 96px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n border: 1px solid var(--velotax-background-color-ghost);\n ${({ width }) =>\n width\n ? css`\n width: ${width} !important;\n `\n : \"\"}\n h2 {\n font-size: 24px;\n padding-bottom: 12px;\n font-weight: 400;\n position: relative;\n color: var(--velotax-text-color);\n margin: 32px 0 0;\n &::after {\n content: \"\";\n left: 0;\n bottom: 0;\n width: 100px;\n height: 2px;\n position: absolute;\n background-color: var(--ant-primary-color);\n }\n }\n @media only screen and (min-width: 1025px) {\n width: 48%;\n }\n @media only screen and (max-width: 1024px) {\n width: 72% !important;\n padding: 32px;\n h2 {\n &::before {\n left: -32px;\n }\n }\n }\n @media only screen and (max-width: 812px) {\n width: 100% !important;\n padding: 24px 24px 48px;\n border: none;\n border-radius: 0;\n margin: 24px 0 0;\n h2 {\n padding: 0 0 12px;\n &::before {\n left: -24px;\n }\n }\n .ant-descriptions {\n .ant-descriptions-item-content {\n font-size: 12px;\n }\n .ant-descriptions-item-label {\n width: 30%;\n font-size: 12px;\n }\n }\n }\n`;\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 128px;\n @media only screen and (max-device-width: 812px) {\n padding: 0 0 64px;\n }\n &.view {\n padding: 0;\n }\n\n .juridic-messages-container {\n opacity: 0.8;\n padding: 32px;\n margin: 0 auto;\n max-width: 600px;\n border-radius: 16px;\n\n .message {\n display: flex;\n flex-direction: row;\n align-items: center;\n margin-bottom: 1rem;\n }\n\n p {\n font-size: 0.85rem;\n font-weight: 400;\n text-align: justify;\n line-height: 1.4;\n color: var(--velotax-font-color);\n margin-bottom: 0;\n margin-left: 0.5rem;\n\n i {\n font-weight: bold;\n color: var(--velotax-font-color);\n }\n }\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n &.view {\n padding: 0;\n }\n margin: 0 auto;\n max-width: 600px;\n border-radius: 16px;\n background-color: var(--velotax-background-color);\n h3 {\n display: flex;\n align-items: center;\n column-gap: 16px;\n color: var(--velotax-font-color);\n margin-bottom: 0;\n }\n .desc-label {\n display: flex;\n align-items: center;\n svg {\n fill: var(--ant-primary-color);\n }\n }\n .desc-content {\n display: flex;\n align-items: flex-start;\n justify-content: center;\n column-gap: 8px;\n svg {\n fill: var(--ant-primary-color);\n }\n span {\n align-self: center;\n }\n button {\n min-width: 32px;\n }\n }\n .ml-40 {\n margin-left: 40px;\n }\n .add {\n min-width: 128px;\n margin: 0 0 16px auto;\n }\n .ant-descriptions-item-label {\n width: 60%;\n }\n .ant-descriptions-item-content {\n text-align: center;\n }\n .ant-list.ant-list-split {\n padding: 0 16px;\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-ghost);\n border-radius: 8px;\n .ant-list-item-action > li {\n padding: 0;\n }\n }\n .ant-list-item-action > li,\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color);\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n .list-description {\n display: flex;\n flex-direction: column;\n }\n .min-darf-price {\n margin-top: 24px;\n display: flex;\n align-items: center;\n column-gap: 8px;\n }\n .text-center {\n display: block;\n text-align: center;\n }\n .ant-collapse-header {\n padding: 12px 0 !important;\n }\n .ant-collapse-content-box {\n padding: 16px 0 !important;\n }\n .ant-collapse\n > .ant-collapse-item\n > .ant-collapse-header\n .ant-collapse-arrow\n svg {\n transform: rotate(-90deg);\n }\n .ant-collapse-header-text {\n width: 100%;\n }\n .total-tax-header {\n width: calc(100% - 48px);\n display: flex;\n align-items: center;\n justify-content: space-between;\n h3 {\n margin: 0;\n }\n }\n\n h5 {\n margin: 0px;\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 24px;\n border-radius: 0;\n h3 {\n font-size: 18px;\n }\n .anticon.anticon-right.ant-collapse-arrow {\n top: 24px;\n }\n .desc-content {\n row-gap: 8px;\n flex-direction: column;\n button {\n width: 100%;\n }\n &.ml-40 {\n margin-left: 0;\n }\n }\n }\n`;\n\nexport const BolsaOperations = styled.div`\n display: grid;\n grid-template-columns: 2fr 1fr;\n border-radius: 8px;\n color: var(--velotax-font-color);\n background-color: var(--velotax-ghost);\n\n div.body-deducoes-tributarias {\n display: flex;\n flex-direction: column;\n padding: 0 8px;\n border-radius: 0 8px 8px 0;\n border: 1px solid var(--velotax-ghost);\n background-color: var(--velotax-background-color);\n span {\n height: 48px;\n display: flex;\n font-weight: 700;\n align-items: center;\n column-gap: 8px;\n }\n }\n\n div.header-deducoes-tributarias {\n display: flex;\n flex-direction: column;\n padding: 0 16px;\n border-radius: 8px 0 0 8px;\n background-color: var(--velotax-ghost);\n span {\n height: 48px;\n display: flex;\n font-weight: 700;\n align-items: center;\n column-gap: 8px;\n border-top: none;\n + span {\n border-top: 1px solid var(--velotax-background-color-ghost);\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n grid-template-columns: 1.6fr 1fr;\n span {\n font-size: 12px;\n button {\n max-width: 24px;\n min-width: 24px;\n }\n }\n }\n`;\n","import { Modal } from \"antd\";\nimport styled from \"styled-components\";\n\nexport const Container = styled.div`\n display: flex;\n align-items: center;\n flex-direction: column;\n padding: 64px 32px 128px;\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n min-width: 40vw;\n max-width: 500px;\n border-radius: 16px;\n background-color: var(--velotax-background-color);\n\n h2 {\n font-size: 24px;\n position: relative;\n color: var(--velotax-font-color);\n &::before {\n content: \"\";\n position: absolute;\n width: 4px;\n top: 0;\n left: -32px;\n height: 100%;\n background-color: var(--ant-primary-color);\n }\n }\n h3 {\n font-size: 20px;\n display: flex;\n align-items: center;\n column-gap: 8px;\n color: var(--velotax-font-color);\n }\n\n .ant-list.ant-list-split {\n margin-top: 24px;\n padding: 0 0 0 16px;\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-ghost);\n border-radius: 8px;\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color);\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n }\n .list-description {\n display: flex;\n flex-direction: column;\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px 24px 128px;\n border-radius: 0;\n h2 {\n &::before {\n left: -24px;\n }\n }\n }\n`;\n\nexport const ConfirmModal = styled(Modal)`\n .p {\n font-size: 20px;\n strong {\n font-weight: 500;\n color: var(--ant-primary-color);\n }\n }\n`;\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 128px;\n\n h2.ant-typography {\n display: flex;\n align-items: center;\n column-gap: 8px;\n width: 700px;\n margin: 0 auto 1.5rem;\n font-size: 1rem;\n font-weight: 400;\n line-height: 1.5rem;\n color: var(--velotax-font-color-light);\n\n &.info {\n align-items: flex-start;\n font-size: 0.875rem;\n line-height: 1.125rem;\n opacity: 0.85;\n margin: 0 auto 2.5rem;\n svg {\n margin-top: 4px;\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n\n h2.ant-typography {\n width: 100%;\n margin: 0 auto 1.5rem;\n padding: 0 24px;\n }\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n margin: 0 auto;\n width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n\n /* .ant-list.ant-list-split {\n margin: 24px 0 0;\n padding: 0 16px;\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-ghost);\n border-radius: 4px;\n .ant-list-item-action > li {\n padding: 0;\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n .ant-list-item-action > li,\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color);\n }\n .ant-list-item-action-split {\n opacity: 0;\n }\n } */\n @media only screen and (max-device-width: 812px) {\n width: 100%;\n padding: 24px;\n border-radius: 0;\n /* .ant-list.ant-list-split {\n .ant-list-item-action {\n display: grid;\n align-items: end;\n grid-template-rows: 1fr 0.75fr;\n grid-template-columns: 1fr 1fr;\n }\n } */\n }\n`;\n\nexport const BottomAdvice = styled.div`\n margin-top: 40px;\n width: 700px;\n\n line-height: 24px;\n\n @media only screen and (max-device-width: 812px) {\n width: 100%;\n padding: 24px;\n }\n`;\n\nexport const AttentionIconContainer = styled.div`\n position: relative;\n display: inline;\n margin-right: 3px;\n width: 14px;\n\n & svg {\n position: relative;\n top: 2px;\n }\n`;","import moment from \"moment\";\nimport { message } from \"antd\";\nimport { useNavigate, useLocation } from \"react-router-dom\";\nimport {\n createContext,\n useCallback,\n useContext,\n useEffect,\n useState,\n} from \"react\";\nimport { useAuth } from \"./AuthContext\";\nimport { errorMessage } from \"../utils\";\nimport apiExterior from \"../services/apiExterior\";\nimport { useBroker } from \"./BrokerContext\";\n\ntype LoadingOperation = \"get\" | \"put\" | \"delete\";\n\ntype Loading = {\n get: boolean;\n put: boolean;\n delete: boolean;\n};\n\ninterface Ativo {\n id: string;\n codigo: string;\n quantidade: number;\n dataCompra?: string;\n valorCompraUSDOrigemBRA?: number;\n valorCompraUSDOrigemEUA?: number;\n orderId: string;\n type: string;\n}\n\ninterface ITransferenciaCustodiaContext {\n ativo?: Ativo;\n ativos: Ativo[];\n loading: Loading;\n getAtivos: any;\n showFormModal: boolean;\n showDeleteModal: boolean;\n closeFormModal: () => void;\n closeDeleteModal: () => void;\n handleAdd: () => void;\n handleEdit: (item: Ativo, index: number) => void;\n handleRemove: (item: Ativo, index: number) => void;\n add: (item: Ativo) => void;\n edit: (item: Ativo) => void;\n remove: (item: Ativo) => void;\n saveAtivos: () => void;\n isInvalidItem: (item: Ativo) => boolean;\n}\n\nexport const TransferenciaCustodiaContext =\n createContext(\n {} as ITransferenciaCustodiaContext\n );\n\nexport const TransferenciaCustodiaProvider: React.FC = ({ children }) => {\n const { user } = useAuth();\n const { currentBroker } = useBroker();\n const navigate = useNavigate();\n const { pathname } = useLocation();\n const [ativo, setAtivo] = useState();\n const [ativos, setAtivos] = useState([]);\n const [indexAtivo, setIndexAtivo] = useState(-1);\n const [redirected, setRedirected] = useState(false);\n const [showFormModal, setShowFormModal] = useState(false);\n const [showDeleteModal, setShowDeleteModal] = useState(false);\n const [loading, setLoading] = useState({\n get: false,\n put: false,\n delete: false,\n });\n\n const isInvalidItem = (item: Ativo) =>\n !item.dataCompra ||\n (!item.valorCompraUSDOrigemBRA && !item.valorCompraUSDOrigemEUA);\n\n const handleLoading = (operation: LoadingOperation, value: boolean) => {\n setLoading((loading) => ({ ...loading, [operation]: value }));\n };\n\n const closeFormModal = () => {\n setShowFormModal(false);\n setAtivo(undefined);\n setIndexAtivo(-1);\n };\n\n const closeDeleteModal = () => {\n setShowDeleteModal(false);\n setAtivo(undefined);\n setIndexAtivo(-1);\n };\n\n const handleAdd = () => {\n setShowFormModal(true);\n };\n\n const handleEdit = (item: Ativo, index: number) => {\n setShowFormModal(true);\n setIndexAtivo(index);\n const dataCompra = !item.dataCompra\n ? \"\"\n : item.dataCompra.includes(\"/\")\n ? item.dataCompra\n : moment.utc(item.dataCompra)?.format(\"DD/MM/YYYY\");\n setAtivo({\n ...item,\n dataCompra,\n });\n };\n\n const handleRemove = (item: Ativo, index: number) => {\n setShowDeleteModal(true);\n setIndexAtivo(index);\n setAtivo(item);\n };\n\n const getAtivos = useCallback(() => {\n handleLoading(\"get\", true);\n apiExterior\n .get(\"/warren/custodia\")\n .then(({ data }) => {\n setAtivos(data || []);\n closeFormModal();\n })\n .catch(() => {\n message.error(errorMessage);\n })\n .finally(() => {\n handleLoading(\"get\", false);\n });\n }, []);\n\n const saveAtivos = () => {\n handleLoading(\"put\", true);\n apiExterior\n .post(\"/warren/custodia\", ativos)\n .then(({ data }) => {\n setAtivos(data ?? []);\n navigate(\"/warren/exterior-historic\");\n })\n .catch(() => {\n message.error(errorMessage);\n })\n .finally(() => {\n handleLoading(\"put\", false);\n });\n };\n\n const add = (item: Ativo) => {\n setAtivos((ativos) => [...ativos, item]);\n closeFormModal();\n };\n\n const edit = (item: Ativo) => {\n setAtivos((ativos) =>\n ativos.map((ativo, index) =>\n index === indexAtivo ? { ...item, type: ativo.type } : ativo\n )\n );\n closeFormModal();\n };\n\n const remove = () => {\n setAtivos((ativos) =>\n ativos.filter((ativo, index) => index !== indexAtivo)\n );\n closeDeleteModal();\n };\n\n useEffect(() => {\n if (\n !redirected &&\n pathname.includes(\"exterior\") &&\n ativos.filter((item: Ativo) => isInvalidItem(item)).length > 0\n ) {\n navigate(\"/warren/transferencia-custodia\");\n setRedirected(true);\n }\n }, [pathname, navigate, ativos, redirected]);\n\n useEffect(() => {\n if (user.user && currentBroker.useTransferenciaCustodia) {\n getAtivos();\n }\n }, [user.user, currentBroker.useTransferenciaCustodia, getAtivos]);\n\n return (\n \n {children}\n \n );\n};\n\nexport const useTransferenciaCustodia = () =>\n useContext(TransferenciaCustodiaContext);\n","import { message } from \"antd\";\nimport { useRef, useState } from \"react\";\nimport { useNavigate } from \"react-router-dom\";\nimport Upload, { RcFile } from \"antd/lib/upload\";\nimport {\n UploadFile,\n UploadFileStatus,\n UploadProps,\n} from \"antd/lib/upload/interface\";\nimport apiBolsa from \"../../../services/apiBolsa\";\nimport apiExterior from \"../../../services/apiExterior\";\nimport { useTransferenciaCustodia } from \"../../../contexts/TransferenciaCustodiaContext\";\n\nconst isValidFile = (file: UploadFile) => file.type === \"application/pdf\";\nconst sameNameFile = (file: UploadFile, files: UploadFile[]) =>\n files.findIndex((item) => item.name === file.name) >= 0;\n\nexport const useNotasCorretagem = () => {\n const navigate = useNavigate();\n const sameNameFiles = useRef([]);\n const [loadingSend, setLoadingSend] = useState(false);\n const [notas, setNotas] = useState[]>([]);\n const [showNotasCorretagemModal, setShowNotasCorretagemModal] =\n useState(false);\n\n const { getAtivos } = useTransferenciaCustodia();\n\n const beforeUploadNota = (file: RcFile) => {\n if (!isValidFile(file)) {\n message.error(\"Você só pode fazer upload de arquivos pdf\");\n return Upload.LIST_IGNORE;\n }\n if (sameNameFile(file, notas)) {\n sameNameFiles.current.push(file.name);\n message.error(\"Este arquivo já foi adicionado\");\n return Upload.LIST_IGNORE;\n }\n };\n\n const onChangeNotas = ({\n file,\n fileList,\n }: {\n file: UploadFile;\n fileList: UploadFile[];\n }) => {\n if (file && file.status === \"removed\") return;\n if (!isValidFile(file)) return;\n if (sameNameFiles.current.includes(file.name)) {\n sameNameFiles.current = sameNameFiles.current.filter(\n (name) => name !== file.name\n );\n return;\n }\n setNotas(\n fileList\n .filter((file) => file.status !== \"removed\")\n .map((file) => ({ ...file, status: \"done\" as UploadFileStatus }))\n );\n };\n\n const onRemoveNota = (file: UploadFile) => {\n setNotas((notas) => notas.filter((nota) => nota.uid !== file.uid));\n };\n\n const sendNotas = () => {\n const formData = new FormData();\n notas.forEach((nota) => {\n formData.append(\"files\", nota.originFileObj as File);\n });\n setLoadingSend(true);\n apiBolsa\n .post(\"/brokerage-notes/read\", formData)\n .then(() => {\n message.success(\"Informações extraídas com sucesso\");\n navigate(\"/warren/bolsa-historic\");\n })\n .catch((err) => {\n const errorMessage =\n err.reponse?.message?.error || \"Não foi possivel extrair informações\";\n message.error(errorMessage);\n })\n .finally(() => {\n setLoadingSend(false);\n });\n };\n\n const sendNotasCustodyTransfer = (setLoadingSend: any, setShowNotasCorretagemModal: any) => {\n const formData = new FormData();\n notas.forEach((nota) => {\n formData.append(\"files\", nota.originFileObj as File);\n });\n setLoadingSend(true);\n apiExterior\n .post(\"/warren/custodia/notes\", formData)\n .then(() => {\n message.success(\"Informações extraídas com sucesso\");\n })\n .catch((err) => {\n const errorMessage =\n err.reponse?.message?.error || \"Não foi possivel extrair informações\";\n message.error(errorMessage);\n })\n .finally(() => {\n getAtivos();\n setShowNotasCorretagemModal(false);\n setLoadingSend(false);\n });\n };\n\n const uploadProps: UploadProps = {\n fileList: notas,\n multiple: true,\n listType: \"text\",\n accept: \"application/pdf\",\n onRemove: onRemoveNota,\n onChange: onChangeNotas,\n beforeUpload: beforeUploadNota,\n customRequest: () => {},\n showUploadList: {\n showRemoveIcon: true,\n showPreviewIcon: false,\n showDownloadIcon: false,\n },\n };\n\n return {\n sendNotas,\n loadingSend,\n sendNotasCustodyTransfer,\n uploadProps,\n showNotasCorretagemModal,\n setShowNotasCorretagemModal,\n };\n};\n","import { Typography } from \"antd\";\nimport {\n RiMoneyDollarCircleLine,\n RiHandCoinLine,\n RiFileSearchLine,\n} from \"react-icons/ri\";\nimport { YearReport } from \"../../constants/rendimentos\";\n\nexport const cards = [\n // {\n // id: \"irpf\",\n // icon: (\n // \n // ),\n // content: (\n // \n // Declaração IRPF\n // \n // 2023 no Velotax\n // \n // ),\n // },\n {\n id: \"anual-report\",\n icon: (\n \n ),\n content: Relatórios anuais ,\n description: (\n \n Relatório auxiliar para preenchimento da DIRPF\n \n ),\n },\n {\n id: \"gain\",\n icon: (\n \n ),\n content: INFORMES DE GANHO DE CAPITAL ,\n description: (\n \n Relatórios auxiliares para acompanhamento do ganho de capital (GCAP)\n \n ),\n },\n {\n id: \"income\",\n icon: (\n \n ),\n content: INFORMES DE DIVIDENDOS ,\n description: (\n \n Relatórios auxiliares para acompanhamento dos dividendos recebidos no\n exterior (carnê-leão)\n \n ),\n },\n];\n","import styled from \"styled-components\";\n\nexport const ReportContainer = styled.div`\n height: 100%;\n display: flex;\n padding: 64px 32px;\n\n & > .content {\n margin: 0 auto;\n width: 700px;\n }\n\n h1 {\n width: 700px;\n margin: 0 auto 1.5rem;\n font-weight: 400;\n font-size: 2rem;\n position: relative;\n line-height: 2.5rem;\n padding-bottom: 1rem;\n color: var(--velotax-font-color-light);\n\n :after {\n content: \"\";\n position: absolute;\n left: 0;\n bottom: 0;\n height: 2px;\n width: 100px;\n background-color: var(--ant-primary-color);\n }\n }\n\n h3 {\n display: flex;\n align-items: center;\n column-gap: 16px;\n color: var(--velotax-font-color-light);\n margin-bottom: 0;\n }\n\n div.ant-typography {\n font-size: 1rem;\n line-height: 1.5rem;\n color: var(--velotax-font-color);\n }\n\n .ant-typography.destak {\n margin: 48px 0;\n display: block;\n font-weight: 700;\n font-size: 1rem;\n line-height: 1.5rem;\n color: var(--velotax-font-color);\n padding-left: 16px;\n border-left: 4px solid var(--ant-primary-color);\n }\n\n .cards {\n margin: 32px 0 0;\n padding-bottom: 32px;\n width: 100%;\n display: grid;\n row-gap: 1rem;\n column-gap: 1rem;\n grid-template-columns: repeat(2, 1fr);\n\n & > div {\n width: 100%;\n }\n\n .card-antd {\n position: relative;\n\n a {\n top: 0;\n left: 0;\n position: absolute;\n width: 100%;\n height: 100%;\n color: var(--velotax-font-color);\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n height: auto;\n padding: 24px 0 0;\n\n & > .content {\n width: 100%;\n h1 {\n width: 100%;\n padding: 0 24px 12px;\n margin: 0 auto 1.5rem;\n :after {\n left: 24px;\n }\n }\n\n & > div.ant-typography {\n padding: 0 24px;\n }\n }\n\n .cards {\n width: 100%;\n padding: 0 24px;\n margin: 32px 0;\n grid-template-columns: repeat(2, 1fr);\n\n & > div {\n width: auto;\n height: 240px;\n }\n }\n }\n`;\n\nexport const Container = styled.div`\n h1 {\n width: 700px;\n margin: 0 auto 1.5rem;\n font-weight: 400;\n font-size: 2rem;\n position: relative;\n line-height: 2.5rem;\n padding-bottom: 1rem;\n color: var(--velotax-font-color-light);\n\n :after {\n content: \"\";\n position: absolute;\n left: 0;\n bottom: 0;\n height: 2px;\n width: 100px;\n background-color: var(--ant-primary-color);\n }\n }\n\n h2.ant-typography {\n display: flex;\n align-items: center;\n column-gap: 8px;\n width: 700px;\n margin: 0 auto 1.5rem;\n font-size: 1rem;\n font-weight: 400;\n line-height: 1.5rem;\n color: var(--velotax-font-color-light);\n\n &.info {\n align-items: flex-start;\n font-size: 0.875rem;\n line-height: 1.125rem;\n opacity: 0.85;\n margin: 0 auto 2.5rem;\n svg {\n margin-top: 4px;\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n\n h1 {\n width: calc(100% - 48px);\n margin: 0 24px 1.5rem;\n padding: 24px 0 1rem;\n }\n\n h2.ant-typography {\n width: 100%;\n margin: 0 auto 1.5rem;\n padding: 0 24px;\n }\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n\n .ant-list.ant-list-split {\n margin: 24px 0 0;\n padding: 0 16px;\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-ghost);\n border-radius: 4px;\n .ant-list-item-action > li {\n padding: 0;\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n .ant-list-item-action > li > button {\n font-size: 16px;\n color: var(--ant-primary-color);\n }\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n font-size: 16px;\n color: var(--velotax-font-color-light);\n }\n .ant-list-item-action-split {\n opacity: 0;\n }\n }\n .historic-month-status {\n font-weight: 500;\n padding: 4px 8px;\n border-radius: 0;\n color: var(--black);\n display: flex;\n &.payed,\n &.regular {\n background-color: var(--ant-success-color);\n }\n &.pending {\n background-color: var(--ant-error-color);\n }\n &.not-payed {\n background-color: var(--ant-warning-color);\n }\n }\n .ant-btn.ant-btn-text {\n color: var(--ant-primary-color);\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px 24px 32px;\n border-radius: 0;\n width: 100%;\n .ant-list.ant-list-split {\n .ant-list-item-action {\n align-items: end;\n grid-template-rows: 1fr 0.75fr;\n grid-template-columns: 1fr 1fr;\n }\n }\n }\n`;\n","import { RiDownloadCloudLine } from \"react-icons/ri\";\nimport { Button as MaterialButton } from \"@mui/material\";\nimport { useCallback, useEffect, useState } from \"react\";\nimport { useLocation, useNavigate } from \"react-router-dom\";\nimport { InfoCircleOutlined, LoadingOutlined } from \"@ant-design/icons\";\nimport {\n Col,\n List,\n message,\n Modal,\n Row,\n Select,\n Skeleton,\n Typography,\n} from \"antd\";\nimport { cards } from \"./items\";\nimport apis from \"../../services/apis\";\nimport { Card } from \"../Exterior/Cards\";\nimport Button from \"../../components/Button\";\nimport { Page } from \"../../constants/brokers\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport { apiPayment } from \"../../services/apiPayment\";\nimport { useBroker } from \"../../contexts/BrokerContext\";\nimport { BackButton } from \"../../components/BackButton\";\nimport { ReportContainer, Container, Content } from \"./styles\";\nimport {\n download,\n errorMessage,\n getInitialYear,\n isMobile,\n monthsExtended,\n zero,\n} from \"../../utils\";\nimport apiExterior from \"../../services/apiExterior\";\nimport { YearReport } from \"../../constants/rendimentos\";\ninterface ReportProps {\n item: Page;\n}\n\nexport const ReportAlertTitle = (\n \n {\" \"}\n \n Os relatórios estão disponíveis somente no plano premium.{\" \"}\n Contrate o seu plano \n \n \n);\n\nexport const ReportExterior: React.FC = ({ item }) => {\n const navigate = useNavigate();\n const {\n showUserPlanModal,\n userPlanModal,\n ssoToken,\n hasPlan,\n user,\n hasPermissionExteriorPlanos,\n hasPermissionGeneral,\n } = useAuth();\n const today = new Date();\n const currentMonth = today.getMonth();\n const currentYear = today.getFullYear();\n const [initialYear, setInitialYear] = useState(currentYear);\n window.history.replaceState({}, document.title);\n\n const { currentPage } = useBroker();\n\n const [action, setAction] = useState('anual-report');\n // const [action, setAction] = useState();\n\n const [data, setData] = useState();\n const [infoModal, toggleModal] = useState(null);\n const [loading, setLoading] = useState(false);\n const [loadingError, setLoadingError] = useState(true);\n\n const [loadingLinks] = useState(false);\n const [year, setYear] = useState(initialYear);\n\n const [reportDisabled, setReportDisabled] = useState(false);\n\n const historicFeature = item.features[0];\n const downloadFeature = item.features[1];\n // const sendEmailFeature = item.features[2];\n const transactionsFeature = item.features[3];\n const darfFeature = item.features[4];\n\n const yearOptions = Array.from(Array(initialYear - 2020 + 1))\n .map((y, i) => ({\n value: 2020 + i,\n label: `${2020 + i}`,\n }))\n .reverse();\n\n const onChangeYear = (value: string) => {\n setYear(parseInt(value));\n };\n\n const getHistoric = useCallback(() => {\n if (!action) return;\n if (action === \"anual-report\")\n return setData(\n Array.from(Array(new Date().getFullYear() - 2022))\n .map((y, i) => ({\n year: 2022 + i,\n label: `${2022 + i}`,\n }))\n .reverse()\n );\n\n setLoading(true);\n (currentPage?.api || apis)\n .get(`${historicFeature.apiUrl}/${action || \"gain\"}`, {\n params: { year },\n })\n .then((response) => {\n const data = (response.data ?? []).reverse();\n setData(data);\n // getDowloadLinks(data, action || \"gain\");\n })\n .catch(() => message.error(errorMessage))\n .finally(() => setLoading(false));\n }, [historicFeature, year, currentPage, action /*, getDowloadLinks*/]);\n\n const downloadAction = (record: any): Promise => {\n setLoading(true);\n return (currentPage?.api || apis)\n .post(`${downloadFeature.apiUrl}/${action || \"anual-report\"}`, {\n id: record._id,\n month: record.month,\n year: record.year,\n })\n .then((res) => {\n if (res.data.fileUrl && !isMobile()) {\n download(res.data.fileUrl);\n }\n if (res.data.file) {\n // downloadPDF(res.data.file);\n message.success(\"Relatório enviado para o seu e-mail\");\n }\n })\n .catch(({ response }) =>\n message.error(response?.data?.message ?? errorMessage)\n )\n .finally(() => setLoading(false));\n };\n\n const downloadDarf = async (data: any) => {\n setLoading(true);\n\n (currentPage?.api || apis).post(`${darfFeature.apiUrl}`, data)\n .then((res) => {\n if (!res?.data?.darfUrl ) {\n return message.error(\"Houve um erro ao baixar seu relatório!\")\n }\n download(res.data.darfUrl);\n message.success(\"Relatório emitido com sucesso\");\n })\n .catch(({ response }) =>\n message.error(response?.data?.message)\n )\n .finally(() => {\n setLoading(false);\n });\n };\n\n const handlePlanModal = () => {\n showUserPlanModal(!userPlanModal, ReportAlertTitle);\n };\n\n \n const checkDarf = (record: any) => {\n setLoading(true);\n (currentPage?.api || apis).get(`${transactionsFeature.apiUrl}`, { params: { year: record.year} })\n .then((res) => {\n const darf = res.data.find((d: any)=> {\n if (record?.month == d?.month) return d;\n })\n if (!darf || !darf?._id) {\n message.info(\"Cliente não possui relatório para este mês.\");\n setLoading(false);\n return;\n }\n delete darf?.transactions;\n const data = {\n ...darf,\n year: record.year,\n month: record.month,\n name: user?.user?.name,\n memoriaCalculo: [],\n justSave: false,\n payed: false,\n noSendMail: true,\n paymentInfo: {\n clientInfo: {\n name: user?.user?.name,\n email: user?.user?.email,\n document: user?.user?.cpf,\n },\n },\n };\n setLoading(false);\n downloadDarf(data);\n })\n .catch(({ response }) => {\n const { data, status } = response;\n if (status === 401) message.error(data?.message);\n setLoading(false);\n });\n }\n\n const handleActionButton = (id: string) => {\n if (!hasPlan) {\n handlePlanModal();\n } else {\n setAction(id);\n switch (id) {\n case \"anual-report\":\n case \"income\":\n setYear(2023);\n setInitialYear(2023);\n break;\n case \"gain\":\n setYear(2024);\n setInitialYear(2024);\n break;\n }\n }\n };\n\n const handleActionButtonBack = (id: any) => {\n setAction(id);\n };\n\n const getErrors = () => {\n apiExterior\n .get(`/transaction/hasError`, { params: { notified: true } })\n .then((res: any) => {\n setReportDisabled((res?.data?.error?.errorDates || []).length > 0)\n })\n .catch(() => {\n console.log(\"Erro no error\");\n })\n .finally(() => setLoadingError(false))\n }\n\n useEffect(() => {\n getErrors()\n }, [])\n\n useEffect(() => {\n getHistoric();\n }, [getHistoric, year]);\n\n return (\n \n \n
Relatórios \n {action == \"anual-report\" &&\n
\n Confira os relatórios anuais de operações no\n exterior para a declaração do seu IR.\n A sua declaração de IRPF pré-preenchida com as informações da Warren está\n disponível.\n \n }\n\n {loadingError ?
: reportDisabled ?\n
\n \n Não foi possível carregar as suas operações. \n Entre em contato através do e-mail suporte@velotax.com.br .\n \n
\n : !action && (\n
\n {cards.map((item) => (\n
: item.icon}\n onClick={\n item.id === \"irpf\"\n ? () => { }\n : () => !loading && handleActionButton(item.id)\n }\n >\n {item.id === \"irpf\" ? (\n <>\n
\n {\" \"}\n \n {item.content}\n >\n ) : (\n <>\n {item.content}\n {item.description && (\n
{item.description}
\n )}\n >\n )}\n \n ))}\n
\n )}\n {/* {!action && (\n
\n )} */}\n {action && (\n
\n \n Selecione o ano: \n {action !== \"anual-report\" && (\n trigger.parentElement}\n />\n )}\n \n action === \"anual-report\" ||\n item.year === year ||\n item.year === currentYear\n ) ?? []\n }\n itemLayout=\"horizontal\"\n renderItem={(item, index) => {\n // const show2024Report = localStorage.getItem('show2024Report') === 'show';\n if ((item.year ===2025) && action === \"anual-report\") return <>>\n return (\n \n {\n (action === \"anual-report\")\n ? toggleModal(item)\n : ((action === \"gain\") && year === 2024)\n ? checkDarf(item)\n : downloadAction(item)\n }\n }\n download={`Relatório-${item.year}-${zero(\n item.month\n )}.pdf`}\n >\n Baixar\n {loadingLinks ? (\n \n ) : (\n \n )}\n ,\n ]}\n >\n \n \n \n \n )\n }}\n />\n navigate(\"/warren/exterior\")}\n style={{ marginTop: \"0.5rem\" }}\n >\n Voltar\n \n {/* {!action || action ==='anual-report' && (\n \n )} */}\n
\n \n )}\n
{\n toggleModal(null);\n }}\n footer={\n \n \n {\n downloadAction(infoModal);\n toggleModal(null);\n }}\n >\n Continuar\n \n \n \n {\n navigate(\"/warren/exterior-historic\");\n }}\n >\n Ir para calculadora de DARF\n \n \n
\n }\n >\n \n \n Caso você tenha realizado operações de venda de ativos ou recebido\n cupom de bonds em {year},\n {\" \"}\n certifique-se de que todos os meses estejam com a situação \"Regular\"\n na Calculadora de DARF.\n \n \n
\n \n );\n};\n","import styled from \"styled-components\";\nimport { DrawerModal } from \"../../components/DrawerModal\";\n\nexport const ModalStyled = styled(DrawerModal)`\n &.ant-modal {\n div.ant-modal-content {\n background-color: var(--white);\n .ant-modal-close {\n top: 0;\n right: 0;\n }\n .ant-modal-close-icon {\n color: var(--ant-primary-color);\n }\n .ant-modal-header {\n padding: 48px 64px 24px 48px;\n background-color: var(--white);\n .ant-modal-title {\n font-size: 24px;\n color: var(--velotax-font-color-dark);\n }\n }\n .ant-modal-body {\n padding: 8px 48px 48px;\n }\n }\n }\n .ant-table {\n width: 100%;\n }\n .mobile-list {\n display: none;\n }\n .table-total {\n padding: 16px;\n display: flex;\n align-items: center;\n background-color: #f0f0f0;\n justify-content: space-between;\n h3 {\n margin: 0;\n font-size: 14px;\n }\n span.ant-typography {\n font-weight: 600;\n width: 116px;\n }\n &.exterior {\n margin-bottom: 16px;\n span.ant-typography {\n width: calc(30% - 22px);\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n &.ant-drawer {\n .ant-drawer-content-wrapper {\n .ant-drawer-content {\n background-color: var(--white) !important;\n .ant-drawer-wrapper-body {\n .ant-drawer-header {\n .ant-drawer-header-title {\n .ant-drawer-title {\n color: var(--velotax-font-color-dark) !important;\n }\n }\n }\n }\n }\n }\n }\n .ant-table {\n display: none;\n }\n .mobile-list {\n width: 100%;\n display: flex;\n margin: 24px 0 0;\n flex-direction: column;\n .list-item {\n width: 100%;\n padding: 16px 24px;\n border-top: 1px solid #e8e8e8;\n :last-of-type {\n border-bottom: 1px solid #e8e8e8;\n }\n p {\n font-size: 16px;\n font-weight: 600;\n margin-bottom: 4px;\n }\n .list-item-label {\n width: 48px;\n font-weight: 500;\n display: inline-block;\n }\n .list-item-value {\n width: calc(100% - 48px);\n display: inline-block;\n }\n }\n .list-empty {\n padding: 16px 0;\n margin-top: -24px;\n text-align: center;\n background-color: #e8e8e8;\n }\n &.exterior {\n .list-item-label {\n width: 100px;\n }\n .list-item-value {\n width: calc(100% - 100px);\n }\n }\n }\n .table-total {\n padding: 24px;\n span.ant-typography {\n width: auto !important;\n text-align: right;\n }\n }\n }\n`;\n\nexport const BolsaOperations = styled.div`\n margin-top: 20px;\n border-radius: 8px;\n color: var(--velotax-font-color-light);\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-background-color-ghost);\n\n div.strong span {\n font-weight: 700 !important;\n }\n div.opacity {\n opacity: 0.8;\n }\n\n span.dividendos-header {\n background-color: var(--velotax-background-color-ghost) !important;\n font-weight: 700 !important;\n }\n div.header,\n div.sub-header,\n div.row {\n display: grid;\n grid-template-columns: 0.7fr 1fr 0.5fr;\n span {\n text-align: center;\n justify-content: center;\n }\n }\n\n div.sub-header,\n div.row {\n height: auto;\n min-height: 48px;\n border-top: none;\n & + div.sub-header {\n border-top: 1px solid var(--velotax-background-color-ghost);\n }\n & > span + span {\n background-color: var(--velotax-background-color);\n }\n & > span {\n display: flex;\n align-items: center;\n border-left: none;\n }\n &:last-of-type > span:last-of-type {\n border-radius: 0 0 8px 0;\n }\n }\n\n div.header {\n padding: 16px;\n border-radius: 8px 8px 0 0;\n background-color: var(--velotax-background-color-ghost);\n span {\n font-weight: 700;\n }\n button {\n display: none;\n }\n }\n\n div.sub-header {\n font-size: 14px;\n & > span {\n font-weight: 300;\n }\n }\n\n div.row {\n opacity: 0.85;\n font-size: 13px;\n border-top: 1px solid var(--velotax-background-color-ghost);\n }\n\n @media only screen and (max-device-width: 812px) {\n div.sub-header {\n font-size: 12px;\n }\n }\n`;\n\nexport const Container = styled.div`\n width: 100%;\n display: flex;\n flex-direction: column;\n &,\n h4,\n .ant-typography {\n color: var(--velotax-font-color-dark);\n }\n .MuiButtonBase-root {\n height: 40px;\n margin: -60px 0 32px auto;\n border: 1px solid #e8e8e8;\n border-radius: 24px !important;\n text-transform: none !important;\n .MuiButton-startIcon {\n width: 18px;\n height: 18px;\n display: flex;\n margin-right: 6px;\n align-items: center;\n svg {\n fill: var(--ant-primary-color);\n }\n }\n }\n @media only screen and (max-device-width: 812px) {\n .MuiButtonBase-root {\n width: 60%;\n margin: -16px 0 0 0;\n }\n }\n`;\n","import { message } from \"antd\";\nimport apiBolsa from \"../services/apiBolsa\";\nimport apiExterior from \"../services/apiExterior\";\n\nexport const sendEmailSheet = async (type: string, orders: any, dateInfo: any, dolarQuote: any, user: any, title: string) => {\n try {\n const api = (type == 'bolsa') ? apiBolsa : apiExterior;\n api\n .post(\"/dividendos/sheet-email\", {\n orders: orders,\n dolarQuote: dolarQuote,\n title: title,\n user: user,\n month: dateInfo?.month,\n year: dateInfo?.year\n })\n .then(() => {\n message.success(\"Planilha enviada por e-mail!\");\n })\n .catch((err) => {\n message.error(err?.response?.data?.message);\n })\n .finally(() => {\n return;\n });\n } catch (error: any) {\n message.error(error)\n }\n}\n","import moment from \"moment\";\nimport { Button } from \"@mui/material\";\nimport { Table, Typography } from \"antd\";\nimport { AiOutlineDownload } from \"react-icons/ai\";\nimport { Container } from \"./styles\";\nimport { formatCurrency, isMobile } from \"../../utils\";\nimport { handleDownloadSheet } from \"../../utils/handleDownloadSheet\";\nimport { sendEmailSheet } from \"../../utils/sendEmailSheet\";\nimport { useAuth } from \"../../contexts/AuthContext\";\n\nexport const DividendosModal = ({ item, year }: any) => {\n const { user } = useAuth();\n const dateInfo = {year: year, month: item?.month}\n return (\n \n {item?.operations?.length > 0 && (\n }\n onClick={() => {\n if (isMobile()) \n {\n sendEmailSheet('bolsa', item?.operations, dateInfo, '', user, \"Dividendos\");\n }\n else {\n handleDownloadSheet(item?.buffer?.data, \"Dividendos\")\n }\n }}\n >\n Baixar planilha\n \n )}\n (\n {value} \n ),\n },\n {\n align: \"left\",\n title: \"Data\",\n dataIndex: \"referenceDate\",\n render: (value) => moment(value).format(\"DD/MM/YYYY\"),\n },\n {\n align: \"left\",\n dataIndex: \"tipo\",\n title: \"Tipo de rendimento\",\n },\n {\n align: \"left\",\n title: \"Valor\",\n width: \"150px\",\n dataIndex: \"valor\",\n render: (value) => formatCurrency(value ?? 0, \"R$ \"),\n },\n ]}\n />\n \n {item?.operations?.map((op: any, index: number) => (\n
\n
{op.ticker}
\n
Tipo: \n
{op.tipo} \n
Data: \n
\n {moment(op.referenceDate).format(\"DD/MM/YYYY\")}\n \n
Valor: \n
\n {formatCurrency(op.valor ?? 0, \"R$ \")}\n \n
\n ))}\n {!item?.operations?.length && (\n
Nenhum dividendo recebido neste mês
\n )}\n
\n {item?.operations?.length > 0 && (\n \n Total recebido no mês: \n {formatCurrency(item.value)} \n
\n )}\n \n );\n};\n","import { monthsExtended } from \"../../utils\";\n\nexport const monthList = Array.from(Array(12))\n .map((_, i) => ({\n month: i + 1,\n monthName: monthsExtended[i],\n value: 0,\n // impostoPagoExterior: 0,\n }))\n .reverse();\n","import { Spin } from \"antd\";\nimport { BsEye } from \"react-icons/bs\";\nimport { formatCurrency } from \"../../utils\";\nimport { CurrencyFormItem, TextFormItem } from \"../formItems\";\n\nconst DividendosItem = {\n id: TextFormItem({\n name: \"id\",\n label: \"Id\",\n hidden: true,\n }),\n valorRendimento: CurrencyFormItem({\n name: \"valorRendimento\",\n label: \"Rendimento\",\n }),\n impostoPagoExterior: CurrencyFormItem({\n name: \"impostoPagoExterior\",\n label: \"Imposto Pago no Exterior\",\n }),\n};\n\nexport const DividendosRows = [\n [DividendosItem.id],\n [DividendosItem.valorRendimento],\n [DividendosItem.impostoPagoExterior],\n];\n\nexport const handleValue = (\n item: any,\n index: number,\n variable: string,\n data: any\n) => {\n const dividendo = data?.find((el: any) => el.month === 12 - index);\n if (dividendo) {\n return dividendo[variable] * dividendo.dolarQuote;\n } else {\n return item[variable];\n }\n};\n\nexport const handleValueBolsa = (item: any, data: any) => {\n const month = data.find((el: any) => el.month === item.month);\n return month?.value || 0;\n};\n\nexport const handleItem = (index: number, data: any, item: any) => {\n return data?.find((el: any) => el.month === 12 - index) ?? item;\n};\n\nexport const DividendosTabs: (\n item: any,\n index: number,\n data: any,\n loading: boolean,\n setShowModal: Function,\n setItem: Function,\n setIndex: Function,\n isBolsa?: boolean\n) => { class: string; tabs: { content: React.ReactNode }[] }[] = (\n item,\n index,\n data,\n loading,\n setShowModal,\n setItem,\n setIndex,\n isBolsa\n) => [\n {\n class: \"sub-header\",\n tabs: [\n {\n content: {item.monthName}
,\n },\n {\n content: loading ? (\n \n ) : (\n formatCurrency(\n isBolsa\n ? handleValueBolsa(item, data)\n : handleValue(item, index, \"valorRendimento\", data)\n )\n ),\n },\n ...(!isBolsa\n ? [\n {\n content: loading ? (\n \n ) : (\n formatCurrency(\n handleValue(item, index, \"impostoPagoExterior\", data)\n )\n ),\n },\n ]\n : []),\n {\n content: (\n {\n setShowModal(true);\n setItem(handleItem(index, data, item));\n setIndex(index);\n }}\n />\n ),\n },\n ],\n },\n];\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n display: flex;\n align-items: center;\n flex-direction: column;\n padding: 64px 32px 64px;\n\n .juridic-messages-container {\n opacity: 0.8;\n width: 700px;\n margin: 0 auto;\n padding: 32px 0 32px 0;\n\n .message {\n display: flex;\n flex-direction: row;\n align-items: center;\n margin-bottom: 1rem;\n }\n\n p {\n font-size: 0.85rem;\n font-weight: 400;\n text-align: justify;\n line-height: 1.4rem;\n color: var(--velotax-font-color-light);\n margin-bottom: 0px;\n\n i {\n font-weight: bold;\n color: var(--velotax-font-color-light);\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n min-height: 100%;\n\n .page-title {\n font-size: 1.25rem;\n line-height: 1.75rem;\n padding-bottom: 1rem;\n }\n\n .juridic-messages-container {\n width: calc(100% - 48px);\n margin: 0 24px 48px;\n p {\n margin-left: 0;\n }\n }\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n\n .destak {\n font-size: 14px;\n font-weight: 400;\n color: var(--velotax-font-color);\n }\n\n button.text-small,\n button.text-small:hover,\n button.text-small:focus,\n button.text-small:active,\n button.text-small:visited {\n color: var(--velotax-font-color-dark) !important;\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px;\n border-radius: 0;\n width: 100%;\n .text-small {\n font-size: 11px;\n }\n h2 {\n &::before {\n left: -24px;\n }\n }\n }\n`;\n","export default __webpack_public_path__ + \"static/media/b3-logo-white.19776928.svg\";","import styled, { css } from \"styled-components\";\n\nexport const Container = styled.div<{ imgHeight?: boolean }>`\n display: flex;\n align-items: center;\n flex-direction: column;\n justify-content: center;\n width: 100%;\n height: 150px;\n padding: 32px;\n cursor: pointer;\n background: var(--velotax-background-color);\n border-radius: 6px;\n position: relative;\n border: 2px solid var(--velotax-ghost-white);\n position: relative;\n transition: 0.3s;\n\n :hover {\n opacity: 0.9;\n filter: contrast(0.9);\n }\n\n img {\n ${(props) =>\n props.imgHeight\n ? css`\n height: 100%;\n `\n : css`\n width: 100%;\n `}\n }\n\n .loading-animation {\n display: flex;\n align-items: center;\n flex-direction: column;\n justify-content: center;\n position: absolute;\n top: 0;\n left: 0;\n width: 100%;\n height: 100%;\n padding: 8px;\n border-radius: 6px;\n backdrop-filter: blur(4px);\n background: var(--ant-info-color-outline);\n span {\n font-size: 14px;\n line-height: 16px;\n font-weight: bold;\n text-align: center;\n }\n .anticon-spin {\n margin: 4px 8px 4px 0;\n font-size: 28px;\n color: var(--white);\n\n svg {\n margin: 0;\n }\n }\n }\n\n &.integrado {\n border: 2px solid var(--ant-success-color);\n }\n\n .integrado-check {\n bottom: 12px;\n right: 12px;\n font-size: 22px;\n position: absolute;\n }\n\n .click-here {\n margin: 16px 0 0;\n font-size: 11px !important;\n font-weight: 500;\n line-height: 20px;\n color: var(--velotax-font-color);\n opacity: 0.8;\n }\n\n a.b3-link {\n position: absolute;\n top: 0;\n left: 0;\n width: 100%;\n height: 100%;\n opacity: 0;\n }\n`;\n","import clsx from \"clsx\";\nimport React from \"react\";\nimport { useLocation } from \"react-use\";\nimport { Tooltip, Typography } from \"antd\";\nimport { CheckCircleTwoTone, LoadingOutlined } from \"@ant-design/icons\";\nimport { Container } from \"./styles\";\nimport { useB3Integration } from \"../../../contexts/B3IntegrationContext\";\n\ninterface IIntegrationButtonProps {\n alt: string;\n loading: boolean;\n integrado: boolean;\n setIntegrationModalVisible: Function;\n logo?: string;\n imgHeight?: boolean;\n handleClick?: () => void;\n b3Link?: string;\n isB3LinkRedirect?: boolean;\n}\n\nconst IntegrationButton: React.FC = ({\n alt,\n logo,\n b3Link,\n loading,\n integrado,\n imgHeight,\n isB3LinkRedirect,\n handleClick = () => {},\n setIntegrationModalVisible,\n}) => {\n const location = useLocation();\n const { timer } = useB3Integration();\n const isB3 = location.pathname === \"/warren/bolsa-integration\";\n\n const onClick = () => {\n if (isB3 || location.pathname === \"/warren/exterior-integration\") {\n handleClick();\n } else if (!loading) {\n setIntegrationModalVisible(true);\n }\n };\n return (\n \n \n {isB3LinkRedirect && b3Link && (\n \n .\n \n )}\n \n \n Clique aqui\n \n {!loading && integrado && (\n \n \n
\n )}\n {isB3 && loading && (\n \n \n Verificando autorização com a B3 ({timer} s) \n
\n )}\n {!isB3 && loading && (\n \n \n Isso pode demorar alguns minutos, aguarde \n
\n )}\n \n \n );\n};\n\nexport default IntegrationButton;\n","import { Button } from \"@mui/material\";\nimport { AiOutlineArrowDown } from \"react-icons/ai\";\nimport { B3AuthURL } from \"../../../utils\";\nimport b3Logo from \"../../../assets/b3-logo-white.svg\";\nimport { useB3Integration } from \"../../../contexts/B3IntegrationContext\";\nimport IntegrationButton from \"../../../components/IntegrationComponents/IntegrationButton\";\n\ninterface B3IntegrationButtonsProps {\n showIntegrationButton?: boolean;\n}\n\nexport const B3IntegrationButtons: React.FC = ({\n showIntegrationButton,\n}) => {\n const {\n openWithLink,\n b3Authorized,\n handleIntegrate,\n loadingIntegration,\n loadingAuthorization,\n checkingAuhtorization,\n } = useB3Integration();\n\n return (\n <>\n {showIntegrationButton && (\n <>\n {}}\n handleClick={() => handleIntegrate()}\n isB3LinkRedirect={openWithLink}\n loading={\n checkingAuhtorization ||\n loadingIntegration ||\n loadingAuthorization\n }\n />\n \n >\n )}\n handleIntegrate()}\n startIcon={ }\n disabled={loadingIntegration || loadingAuthorization}\n {...(openWithLink ? { href: B3AuthURL } : {})}\n >\n Conectar com a B3\n \n >\n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div``;\n\nexport const Content = styled.div``;\n","import { message } from \"antd\";\nimport { useRef, useState } from \"react\";\nimport Upload, { RcFile } from \"antd/lib/upload\";\nimport {\n UploadFile,\n UploadFileStatus,\n UploadProps,\n} from \"antd/lib/upload/interface\";\nimport apiCrypto from \"../../../services/apiCrypto\";\nimport { useBroker } from \"../../../contexts/BrokerContext\";\n\nconst isValidFile = (file: UploadFile) => file.type === \"application/pdf\";\nconst sameNameFile = (file: UploadFile, files: UploadFile[]) =>\n files.findIndex((item) => item.name === file.name) >= 0;\n\nexport const useNotasCorretagem = () => {\n const sameNameFiles = useRef([]);\n const [loadingSend, setLoadingSend] = useState(false);\n const [notas, setNotas] = useState[]>([]);\n const [showNotasCorretagemModal, setShowNotasCorretagemModal] =\n useState(false);\n\n const { getXtageIntegrationStatus } = useBroker();\n\n const beforeUploadNota = (file: RcFile) => {\n if (!isValidFile(file)) {\n message.error(\"Você só pode fazer upload de arquivos pdf\");\n return Upload.LIST_IGNORE;\n }\n if (sameNameFile(file, notas)) {\n sameNameFiles.current.push(file.name);\n message.error(\"Este arquivo já foi adicionado\");\n return Upload.LIST_IGNORE;\n }\n };\n\n const onChangeNotas = ({\n file,\n fileList,\n }: {\n file: UploadFile;\n fileList: UploadFile[];\n }) => {\n if (file && file.status === \"removed\") return;\n if (!isValidFile(file)) return;\n if (sameNameFiles.current.includes(file.name)) {\n sameNameFiles.current = sameNameFiles.current.filter(\n (name) => name !== file.name\n );\n return;\n }\n setNotas(\n fileList\n .filter((file) => file.status !== \"removed\")\n .map((file) => ({ ...file, status: \"done\" as UploadFileStatus }))\n );\n };\n\n const onRemoveNota = (file: UploadFile) => {\n setNotas((notas) => notas.filter((nota) => nota.uid !== file.uid));\n };\n\n const sendNotas = () => {\n const formData = new FormData();\n notas.forEach((nota) => {\n formData.append(\"files\", nota.originFileObj as File);\n });\n setLoadingSend(true);\n apiCrypto\n .post(\"/brokerage-notes/read\", formData)\n .then(() => {\n message.success(\"Informações extraídas com sucesso\");\n getXtageIntegrationStatus({\n redirect: true,\n });\n // navigate(\"/warren/cripto\");\n // navigate(\"/warren/cripto-historic\");\n })\n .catch((err) => {\n const errorMessage =\n err.reponse?.message?.error || \"Não foi possivel extrair informações\";\n message.error(errorMessage);\n })\n .finally(() => {\n setLoadingSend(false);\n });\n };\n\n const uploadProps: UploadProps = {\n fileList: notas,\n multiple: true,\n listType: \"text\",\n accept: \"application/pdf\",\n onRemove: onRemoveNota,\n onChange: onChangeNotas,\n beforeUpload: beforeUploadNota,\n customRequest: () => {},\n showUploadList: {\n showRemoveIcon: true,\n showPreviewIcon: false,\n showDownloadIcon: false,\n },\n };\n\n return {\n sendNotas,\n loadingSend,\n uploadProps,\n showNotasCorretagemModal,\n setShowNotasCorretagemModal,\n };\n};\n","import { Col, Row } from \"antd\";\nimport { Button } from \"@mui/material\";\nimport { useNavigate } from \"react-router\";\nimport { useNotasCorretagem } from \"./useNotasCorretagem\";\nimport { useBroker } from \"../../../contexts/BrokerContext\";\nimport UploadVelotax from \"../../../components/UploadVelotax\";\n\ninterface NotasCorretagemProps {}\n\nexport const NotasCorretagem: React.FC = () => {\n const navigate = useNavigate();\n const { xtageIntegration } = useBroker();\n const { sendNotas, loadingSend, uploadProps } = useNotasCorretagem();\n\n return (\n <>\n \n \n {xtageIntegration && (\n \n navigate(\"/warren/cripto\")}\n >\n Voltar\n \n \n )}\n \n sendNotas()}\n >\n Enviar\n \n \n
\n >\n );\n};\n","import { monthsExtended } from \"../../utils\";\n\nexport const monthList = Array.from(Array(12))\n .map((_, i) => ({\n month: i + 1,\n monthName: monthsExtended[i],\n valorRendimento: 0,\n impostoPagoExterior: 0,\n }))\n .reverse();\n\nexport const CalcAutonomosDarfItem = {\n header: true,\n sidebar: true,\n component: \"Rendimentos\",\n path: \"/velotax/autonomos-darf\",\n name: \"Calculadora\",\n title: \"\",\n features: [\n {\n apiUrl: \"/transaction\",\n },\n {\n apiUrl: \"/darf\",\n buttonLabel: \"Emitir DARF\",\n },\n {\n apiUrl: \"/darf/year\",\n },\n ],\n}","import { Button } from \"@mui/material\";\nimport { Typography, Spin, Table } from \"antd\";\nimport { AiOutlineDownload } from \"react-icons/ai\";\nimport { useCallback, useEffect, useState } from \"react\";\nimport { formatCurrency, formatDate, isMobile } from \"../../utils\";\nimport { Container } from \"../../pages/DividendosBolsa/styles\";\nimport { handleDownloadSheet } from \"../../utils/handleDownloadSheet\";\nimport { sendEmailSheet } from \"../../utils/sendEmailSheet\";\nimport { useAuth } from \"../../contexts/AuthContext\";\n\nexport const DividendosModal = ({ item, year, quotes }: any) => {\n const [total, setTotal] = useState({});\n const [object, setObject] = useState({});\n const [loading, setLoading] = useState(false);\n const dateInfo = {year: year, month: item?.month}\n const { user } = useAuth();\n\n const checkType = (el: any) => {\n if (el.type === \"DIVIDEND\" && el.operation === \"BUY\") {\n return \"valorRendimento\";\n } else return \"impostoPagoExterior\";\n };\n\n const getTotal = useCallback(async () => {\n let auxTotal: any = {\n totalRendimento: 0,\n totalRendimentoReal: 0,\n totalImposto: 0,\n totalImpostoReal: 0,\n };\n Object.keys(object).forEach((item: any) => {\n auxTotal = {\n ...auxTotal,\n totalRendimento:\n (auxTotal.totalRendimento ?? 0) + (object[item].valorRendimento ?? 0),\n totalRendimentoReal:\n (auxTotal.totalRendimentoReal ?? 0) +\n (object[item].valorRendimentoReal ?? 0),\n totalImposto:\n (auxTotal.totalImposto ?? 0) +\n (object[item].impostoPagoExterior ?? 0),\n totalImpostoReal:\n (auxTotal.totalImpostoReal ?? 0) +\n (object[item].impostoPagoExteriorReal ?? 0),\n };\n });\n\n setTotal(auxTotal);\n }, [object]);\n\n const groupBycode = useCallback(async () => {\n setLoading(true);\n const aux: any = {};\n for (const el of item.orders ?? []) {\n if (!aux[el.code]) {\n if (checkType(el) === \"valorRendimento\") {\n aux[el.code] = {\n valorRendimento: el.value,\n valorRendimentoReal: el.value * item.dolarQuote,\n date: formatDate(el.date),\n };\n } else {\n aux[el.code] = {\n impostoPagoExterior: el.value,\n impostoPagoExteriorReal: el.value * item.dolarQuote,\n date: formatDate(el.date),\n };\n }\n } else {\n if (checkType(el) === \"valorRendimento\") {\n aux[el.code] = {\n ...aux[el.code],\n valorRendimento: (aux[el.code]?.valorRendimento ?? 0) + el.value,\n valorRendimentoReal:\n (aux[el.code]?.valorRendimentoReal ?? 0) +\n el.value * item.dolarQuote,\n };\n } else {\n aux[el.code] = {\n ...aux[el.code],\n impostoPagoExterior:\n (aux[el.code]?.impostoPagoExterior ?? 0) + el.value,\n impostoPagoExteriorReal:\n (aux[el.code]?.impostoPagoExteriorReal ?? 0) +\n el.value * item.dolarQuote,\n };\n }\n }\n }\n\n setObject(aux);\n setLoading(false);\n }, [item.orders, item.dolarQuote]);\n\n useEffect(() => {\n groupBycode();\n }, [groupBycode, item.month, year]);\n\n useEffect(() => {\n getTotal();\n }, [getTotal, object]);\n\n return (\n \n {loading ? (\n \n ) : (\n <>\n {Object.keys(object)?.length > 0 && (\n }\n onClick={() =>\n {\n if (isMobile()) \n {\n sendEmailSheet('exterior', item?.orders, dateInfo, item?.dolarQuote, user, \"Dividendos\");\n }\n else \n {\n handleDownloadSheet(\n item?.buffer?.data,\n `Dividendos-${year}-${item.month}`\n )\n }\n }\n }\n >\n Baixar planilha\n \n )}\n ({\n ...object[key],\n ticker: key,\n }))}\n locale={{ emptyText: \"Nenhum dividendo recebido neste mês\" }}\n columns={[\n {\n align: \"left\",\n title: \"Ativo\",\n dataIndex: \"ticker\",\n render: (value) => (\n {value} \n ),\n },\n {\n align: \"left\",\n title: \"Data\",\n dataIndex: \"date\",\n },\n {\n width: \"30%\",\n align: \"left\",\n dataIndex: \"valorRendimento\",\n title: \"Valor em USD\",\n render: (value) => formatCurrency(value ?? 0, \"US$ \"),\n },\n {\n width: \"30%\",\n align: \"left\",\n dataIndex: \"valorRendimentoReal\",\n title: \"Valor em BRL\",\n render: (value) => formatCurrency(value ?? 0, \"R$ \"),\n },\n ]}\n />\n \n {Object.keys(object)\n .map((key) => ({\n ...object[key],\n ticker: key,\n }))\n ?.map((op: any, index: number) => (\n
\n
{op.ticker}
\n
Data: \n
{op.date} \n
Valor em USD: \n
\n {formatCurrency(op.valorRendimento ?? 0, \"US$ \")}\n \n
Valor em BRL: \n
\n {formatCurrency(op.valorRendimentoReal ?? 0, \"R$ \")}\n \n
\n ))}\n {!Object.keys(object).length && (\n
\n Nenhum dividendo recebido neste mês\n
\n )}\n
\n {Object.keys(object).length > 0 && (\n \n \n Total recebido no mês:\n \n \n {formatCurrency(total.totalRendimentoReal ?? 0)}\n \n
\n )}\n >\n )}\n \n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 32px;\n\n h1 {\n width: 700px;\n margin: 0 auto 1.5rem;\n font-weight: 400;\n font-size: 2rem;\n position: relative;\n line-height: 2.5rem;\n padding-bottom: 1rem;\n color: var(--velotax-font-color-light);\n\n :after {\n content: \"\";\n position: absolute;\n left: 0;\n bottom: 0;\n height: 2px;\n width: 100px;\n background-color: var(--ant-primary-color);\n }\n }\n\n span.ant-typography {\n opacity: 0.8;\n font-weight: 400;\n font-size: 0.85rem;\n line-height: 1.5rem;\n text-align: justify;\n color: var(--velotax-font-color-light);\n }\n\n h2.ant-typography {\n display: flex;\n align-items: center;\n column-gap: 8px;\n width: 700px;\n margin: 0 auto 2.5rem;\n font-size: 1rem;\n font-weight: 400;\n line-height: 1.5rem;\n color: var(--velotax-font-color-light);\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 0 0 32px;\n\n h1 {\n width: calc(100% - 48px);\n margin: 0 24px 1.5rem;\n padding: 24px 0 1rem;\n }\n\n h2.ant-typography {\n width: 100%;\n margin: 0 auto 1.5rem;\n padding: 0 24px;\n }\n }\n`;\n\nexport const FlexDiv = styled.div`\n display: flex;\n justify-content: center;\n flex-direction: column;\n`;\n\nexport const BolsaOperations = styled.div`\n margin-top: 20px;\n border-radius: 4px;\n color: var(--velotax-font-color-light);\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-background-color-ghost);\n\n .flex-actions {\n display: flex;\n justify-content: center;\n align-items: center;\n min-width: 170px;\n display: flex;\n }\n\n div.strong span {\n font-weight: 700 !important;\n }\n div.opacity {\n opacity: 0.8;\n }\n\n span.dividendos-header {\n background-color: var(--velotax-background-color-ghost) !important;\n font-weight: 700 !important;\n }\n div.header,\n div.sub-header,\n div.row {\n display: grid;\n grid-template-columns: 1fr 1fr 1fr 0.5fr;\n span {\n text-align: center;\n justify-content: center;\n }\n }\n\n table.table-month-operations {\n width: 100%;\n thead {\n height: auto;\n border-top: none;\n }\n tr {\n height: 48px;\n }\n td.table-vertical-head {\n background-color: var(--velotax-background-color-ghost);\n }\n td {\n min-width: 120px;\n text-align: center;\n }\n }\n\n div.sub-header,\n div.row {\n height: auto;\n min-height: 48px;\n border-top: none;\n & + div.sub-header {\n border-top: 1px solid var(--velotax-background-color-ghost);\n }\n & > span + span {\n background-color: var(--velotax-border-color);\n }\n & > span {\n display: flex;\n align-items: center;\n border-left: none;\n }\n &:last-of-type > span:last-of-type {\n border-radius: 0 0 4px 0;\n }\n }\n\n div.header {\n padding: 16px;\n border-radius: 4px 4px 0 0;\n background-color: var(--velotax-background-color-ghost);\n span {\n font-weight: 700;\n }\n button {\n display: none;\n }\n }\n\n div.sub-header {\n font-size: 14px;\n & > span {\n font-weight: 300;\n }\n }\n\n div.row {\n opacity: 0.85;\n font-size: 13px;\n border-top: 1px solid var(--velotax-background-color-ghost);\n }\n\n @media only screen and (max-device-width: 812px) {\n div.sub-header {\n font-size: 12px;\n }\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n margin: 0 auto;\n width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n &.notas {\n padding: 32px 0;\n background-color: transparent;\n svg {\n fill: var(--ant-primary-color);\n }\n }\n h1 {\n margin: 0 0 8px;\n font-size: 20px;\n color: var(--velotax-font-color);\n }\n h2 {\n margin: 0;\n font-size: 16px;\n font-weight: 400;\n color: var(--velotax-font-color);\n }\n .ant-list.ant-list-split {\n margin: 24px 0 0;\n padding: 0 16px;\n background-color: var(--velotax-ghost);\n border: 1px solid var(--velotax-ghost);\n border-radius: 8px;\n .ant-list-item-action > li {\n padding: 0;\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n .ant-list-item-action > li,\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color);\n }\n .ant-list-item-action-split {\n opacity: 0;\n }\n }\n .historic-month-status {\n font-weight: 500;\n padding: 4px 8px;\n border-radius: 4px;\n color: var(--black);\n display: flex;\n &.payed,\n &.regular {\n background-color: var(--ant-success-color);\n }\n &.pending {\n background-color: var(--ant-error-color);\n }\n &.not-payed {\n background-color: var(--velotax-font-color);\n }\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px 24px 32px;\n border-radius: 0;\n width: 100%;\n &.notas {\n padding: 32px 24px;\n }\n .ant-list.ant-list-split {\n .ant-list-item-action {\n display: grid;\n align-items: end;\n grid-template-rows: 1fr 0.75fr;\n grid-template-columns: 1fr 1fr;\n }\n }\n }\n`;\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 128px;\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n min-height: 100%;\n background-color: var(--velotax-background-color);\n }\n &.view,\n &.view-edit {\n padding: 0;\n }\n`;\n\nexport const Content = styled.div`\n padding: 32px;\n /* &:not(.view):not(.view-edit) {\n border: 1px solid var(--velotax-background-color-ghost);\n } */\n &.second.view,\n &.second.view-edit {\n margin-top: -24px;\n padding: 0 32px 32px;\n }\n &.second {\n padding: 32px;\n border: none;\n background-color: transparent;\n }\n margin: 0 auto;\n border-radius: 16px;\n background-color: var(--velotax-background-color);\n\n .no-plan-container {\n position: absolute;\n top: 0;\n left: 50%;\n width: 50%;\n z-index: 3;\n height: 100%;\n padding: 8px;\n display: flex;\n font-size: 16px;\n font-weight: 500;\n text-align: center;\n align-items: center;\n flex-direction: column;\n justify-content: center;\n background-color: #fff4;\n border-radius: 0 8px 8px 0;\n backdrop-filter: blur(4px);\n border: 1px solid var(--velotax-background-color-ghost);\n border-left: none;\n .ant-btn {\n margin-top: 8px;\n }\n }\n\n h3.date-title {\n font-size: 24px;\n font-weight: 700;\n }\n\n h3 {\n display: flex;\n align-items: center;\n font-size: 20px;\n column-gap: 16px;\n color: var(--velotax-font-color);\n margin-bottom: 0;\n }\n .month-chevron {\n width: 32px;\n height: 32px;\n cursor: pointer;\n &.empty {\n cursor: default;\n }\n }\n .desc-label {\n display: flex;\n align-items: center;\n svg {\n fill: var(--ant-primary-color);\n }\n }\n .desc-content {\n display: flex;\n align-items: flex-start;\n justify-content: center;\n column-gap: 8px;\n svg {\n fill: var(--ant-primary-color);\n }\n span {\n align-self: center;\n }\n button {\n min-width: 32px;\n }\n }\n .ml-40 {\n margin-left: 40px;\n }\n .add {\n min-width: 128px;\n margin: 0 0 16px auto;\n }\n .ant-descriptions-item-label {\n width: 50%;\n background-color: var(--velotax-ghost) !important;\n }\n .ant-descriptions-item-content {\n text-align: center;\n background-color: var(--velotax-background-color) !important;\n }\n .ant-list.ant-list-split {\n padding: 0 16px;\n background-color: var(--velotax-ghost) !important;\n border: 1px solid var(--velotax-background-color-ghost);\n border-radius: 8px;\n .ant-list-item-action > li {\n padding: 0;\n }\n }\n .ant-list-item-action > li,\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color);\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n .list-description {\n display: flex;\n flex-direction: column;\n }\n .min-darf-price {\n margin-top: 24px;\n display: flex;\n align-items: center;\n column-gap: 8px;\n }\n .text-center {\n display: block;\n text-align: center;\n }\n .ant-collapse-header {\n padding: 12px 0 !important;\n }\n .ant-collapse-content-box {\n padding: 16px 0 !important;\n }\n .ant-collapse\n > .ant-collapse-item\n > .ant-collapse-header\n .ant-collapse-arrow\n svg {\n transform: rotate(-90deg);\n }\n .ant-collapse-header-text {\n width: 100%;\n }\n .ant-collapse-item-disabled {\n & > .ant-collapse-header {\n cursor: default;\n .ant-collapse-arrow {\n opacity: 0;\n }\n }\n }\n .total-tax-header {\n width: calc(100% - 48px);\n display: flex;\n align-items: center;\n justify-content: space-between;\n &.original {\n h3:last-of-type {\n width: 40%;\n margin-bottom: 2px;\n }\n }\n h3 {\n margin: 0 0 6px;\n }\n }\n\n &.view .total-tax-header.impostos-devidos h3:first-of-type {\n max-width: 60%;\n font-size: 18px;\n margin-left: 4px;\n line-height: 22px;\n }\n\n h5 {\n margin: 0px;\n }\n\n @media only screen and (max-device-width: 812px) {\n width: 100%;\n border-radius: 0;\n padding: 24px 24px 48px;\n border: none !important;\n &.second {\n margin-top: -24px;\n padding: 0 24px 48px !important;\n background-color: var(--velotax-background-color);\n }\n h3 {\n font-size: 18px;\n column-gap: 8px;\n }\n .no-plan-container {\n font-size: 14px;\n line-height: 18px;\n }\n .total-tax-header {\n width: calc(100% - 20px);\n h3.text-center {\n span {\n font-size: 16px;\n }\n }\n }\n .anticon.anticon-right.ant-collapse-arrow {\n top: 24px;\n right: 0;\n }\n .desc-content {\n column-gap: 2px;\n flex-direction: row;\n &.is-editting {\n row-gap: 8px;\n flex-direction: column;\n button {\n width: 100%;\n }\n }\n &:not(.is-editting) {\n span {\n line-height: 32px;\n }\n button {\n margin-top: -2px;\n }\n }\n &.ml-40 {\n margin-left: 0;\n }\n }\n }\n`;\n\nexport const BolsaOperations = styled.div`\n display: grid;\n grid-template-columns: 2fr 1fr;\n border-radius: 8px;\n color: var(--velotax-font-color);\n background-color: var(--velotax-background-color);\n border: 1px solid var(--velotax-background-color-ghost);\n\n div.body-deducoes-tributarias {\n display: flex;\n flex-direction: column;\n padding: 0 8px;\n border-radius: 0 8px 8px 0;\n background-color: var(--velotax-background-color);\n border-left: 1px solid var(--velotax-background-color-ghost);\n span {\n height: 48px;\n display: flex;\n font-weight: 700;\n align-items: center;\n column-gap: 8px;\n }\n }\n\n div.header-deducoes-tributarias {\n display: flex;\n flex-direction: column;\n padding: 0 16px;\n border-radius: 8px 0 0 8px;\n background-color: var(--velotax-ghost);\n span {\n height: 48px;\n display: flex;\n font-weight: 700;\n align-items: center;\n column-gap: 8px;\n border-top: none;\n + span {\n border-top: 1px solid var(--velotax-background-color-ghost);\n }\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n grid-template-columns: 1.6fr 1fr;\n span {\n font-size: 12px;\n button {\n max-width: 24px;\n min-width: 24px;\n }\n }\n }\n`;\n","import clsx from \"clsx\";\nimport { FaRegEdit } from \"react-icons/fa\";\nimport NumberFormat from \"react-number-format\";\nimport { useLocation, useNavigate } from \"react-router-dom\";\nimport { PlusOutlined } from \"@ant-design/icons\";\nimport { InfoCircleOutlined } from \"@ant-design/icons\";\nimport { AiOutlineDelete, AiOutlineLock } from \"react-icons/ai\";\nimport {\n useState,\n useEffect,\n useCallback,\n useRef,\n useMemo,\n ReactNode,\n} from \"react\";\nimport {\n Typography,\n Row,\n Col,\n Space,\n Descriptions,\n Spin,\n InputNumber,\n Collapse,\n List,\n Skeleton,\n Tooltip,\n message,\n} from \"antd\";\nimport MsDarf from \"../../services/msDarf\";\nimport Button from \"../../components/Button\";\nimport { Page } from \"../../constants/brokers\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport { FormModal } from \"../../components/FormModal\";\nimport { apiPayment } from \"../../services/apiPayment\";\nimport { DrawerModal } from \"../../components/DrawerModal\";\nimport { PaymentModal } from \"../../components/PaymentModal\";\nimport { Content, BolsaOperations } from \"./styles\";\nimport DeleteConfirmationModal from \"../../components/DeleteConfirmationModal\";\nimport {\n ImpostoTotalRendimentosAutonomos,\n ModalAutonomos,\n ModalDependentes,\n ModalPensao,\n ModalPrevidencia,\n RendimentoFormItemRows,\n RendimentosImpostosDevidos,\n} from \"../../constants/rendimentos\";\nimport {\n formatCurrency,\n maskCpf,\n currencyToNumber,\n download,\n errorMessage,\n} from \"../../utils\";\nimport {\n minDarfPrice,\n maxDarfPrice,\n maxPixPayment,\n maxCreditCardPayment,\n} from \"../../constants/darf\";\nimport { IMonthStock, Iimpostos } from \"./interfaces\";\nimport api from \"../../services/apiExterior\";\n\ninterface DarfBolsaProps {\n item: Page;\n yearView: number;\n monthView: number;\n closeModal: () => void;\n view?: boolean;\n viewEdit?: boolean;\n}\n\nexport const AutonomosDarf: React.FC = ({\n view,\n viewEdit,\n yearView,\n monthView,\n closeModal,\n}) => {\n const navigate = useNavigate()\n const { state } = useLocation();\n const { user, showUserPlanModal, userPlanModal, hasPlan } = useAuth();\n\n const today = new Date();\n const queryYear = (state as any)?.year as number;\n const queryMonth = (state as any)?.month as number;\n const currentMonth = today.getMonth();\n const currentYear = today.getFullYear();\n const initialMonth = queryMonth\n ? queryMonth - 1\n : typeof monthView === \"number\"\n ? monthView\n : today.getMonth();\n const initialYear = queryYear\n ? queryYear\n : typeof yearView === \"number\"\n ? yearView\n : today.getFullYear();\n window.history.replaceState({}, document.title);\n\n const month = initialMonth;\n const year = initialYear;\n const isSaveItens = useRef(false);\n const [darf, setDarf] = useState<{ id: string }>();\n const [id, setId] = useState(\"\");\n const [loading, setLoading] = useState(false);\n const [itemIndex, setItemIndex] = useState(0);\n const [emitting, setEmitting] = useState(false);\n const [helpModal, setHelpModal] = useState();\n const [paymentData, setPaymentData] = useState();\n const [paymentModal, setPaymentModal] = useState(false);\n const [showDeleteModal, setShowDeleteModal] = useState(false);\n const [monthStock, setMonthStock] = useState([]);\n const [showRendimentoModal, setShowRendimentoModal] = useState(false);\n const [loadingDarfButton, setLoadingDarfButton] = useState(false);\n const [item, setItem] = useState({\n valorRendimento: 0,\n taxaCambio: 2.5,\n tipo: \"Dividendos no exterior\",\n });\n const [impostos, setImpostos] = useState({\n totalRendimentos: 0,\n baseTributaria: 0,\n aliquotaDevida: 0,\n impostoDevido: 0,\n impostoTotal: 0,\n totalDeducoes: 0,\n totalDependentes: 0,\n totalDespesas: 0,\n totalPensao: 0,\n totalPrevidencia: 0,\n impostoAcumulado: 0,\n juros: 0,\n multa: 0,\n gastosExterior: 0,\n payed: false,\n emitted: false,\n });\n\n const valorImpostoTotal =\n Number(\n Math.max(\n 0,\n impostos.impostoDevido +\n (impostos.impostoAcumulado || 0) -\n (impostos.gastosExterior || 0) +\n (impostos.multa || 0) +\n (impostos.juros || 0)\n )\n ) ?? 0;\n\n const handlePlanModal = () => {\n navigate('/planos')\n };\n\n const handleSaveMonth = () => {\n handleEventDarfButton(true);\n };\n\n const handleEventDarfButton = async (justSave?: boolean) => {\n setLoadingDarfButton(true);\n let retPlanInfo = {\n data: {\n active: false,\n type: \"\",\n },\n };\n try {\n retPlanInfo = await apiPayment.get(\"/user-plan/plan-info\");\n } catch (err) {\n console.error(err);\n }\n\n if (!retPlanInfo.data?.active && !retPlanInfo.data?.type) {\n handlePlanModal()\n setLoadingDarfButton(false);\n return\n }\n if (justSave) {\n generateDarf(true, true)\n setLoadingDarfButton(false);\n return\n }\n if (impostos.impostoDevido || 0 <= maxDarfPrice) {\n handleOpenPaymentModal()\n setLoadingDarfButton(false);\n return\n }\n\n // !retPlanInfo.data?.active && !retPlanInfo.data?.type\n // ? handlePlanModal()\n // : justSave\n // ? generateDarf(true, true)\n // : impostos.impostoDevido || 0 <= maxDarfPrice\n // ? handleOpenPaymentModal()\n // : handleOpenDarfModal();\n handleOpenDarfModal()\n setLoadingDarfButton(false);\n };\n\n const handleLabelButton: ReactNode = useMemo(() => {\n if ((impostos?.impostoDevido || 0) <= maxDarfPrice) return \"Pagar DARF\";\n else return \"Emitir DARF\";\n }, [impostos?.impostoDevido]);\n\n const ActionButtons = (\n \n {!view && (\n \n {!(month === currentMonth && year === currentYear) &&\n (valorImpostoTotal ?? 0) < minDarfPrice &&\n (!view\n ? monthStock.filter((m) => m.tipo !== \"Pensão Alimentícia\")\n : monthStock\n )?.length > 0 && (\n \n \n \n O valor do imposto devido é menor que{\" \"}\n {formatCurrency(minDarfPrice)}. Você não precisa emitir\n uma DARF para este mês. Clique aqui para regularizar este\n mês.\n >\n }>\n {\n handleRegularize(true);\n }}>\n Salvar sem emitir DARF\n \n \n \n )}\n {valorImpostoTotal >= minDarfPrice && (\n \n \n target.parentElement!}\n title={\n month === currentMonth && year === currentYear\n ? \"Aguarde o encerramento do mês para pagar seu DARF\"\n : \"\"\n }>\n handleEventDarfButton()}\n disabled={\n loading ||\n emitting ||\n (month === currentMonth && year === currentYear)\n }>\n {handleLabelButton}\n {\" \"}\n \n \n )}\n
\n )}\n {!(month === currentMonth && year === currentYear) && (\n \n \n \n Salvar apenas\n \n \n
\n )}\n \n );\n\n const buildCalcTaxBody = useCallback(() => {\n return {\n numeroDependentes: impostos.totalDependentes,\n gastosAutonomos: impostos.totalDespesas,\n gastosPrevidencia: impostos.totalPrevidencia,\n gastosPensao: impostos.totalPensao,\n gastosExterior: impostos.gastosExterior,\n };\n }, [\n impostos.totalDependentes,\n impostos.totalDespesas,\n impostos.totalPensao,\n impostos.totalPrevidencia,\n impostos.gastosExterior,\n ]);\n\n const calcTaxes = useCallback(async () => {\n setLoading(true);\n if (!view) {\n const deducoes = buildCalcTaxBody();\n const response = await api.post(\"/rendimentos/calc-tax\", {\n ...deducoes,\n rendimentos: monthStock,\n year,\n month: month + 1,\n impostoAcumulado: impostos.impostoAcumulado,\n gastosExterior: impostos.gastosExterior,\n });\n const { data } = response;\n setImpostos((currentValue) => {\n return {\n ...currentValue,\n aliquotaDevida: data.aliquotaDevida,\n baseTributaria: data.baseTributaria,\n impostoDevido: data.impostoDevido,\n impostoTotal: data.impostoTotal,\n juros: data.juros,\n multa: data.multa,\n totalDeducoes: data.totalDeducoes,\n totalRendimentos: data.totalRendimentos,\n };\n });\n }\n setLoading(false);\n }, [\n view,\n buildCalcTaxBody,\n monthStock,\n year,\n month,\n impostos.impostoAcumulado,\n impostos.gastosExterior,\n ]);\n\n const setPayData = useCallback(() => {\n const valueToCharge = Math.max(\n 0,\n (impostos?.impostoDevido || 0) +\n (impostos.impostoAcumulado || 0) -\n (impostos.gastosExterior || 0) +\n (impostos.juros || 0) +\n (impostos.multa || 0)\n );\n\n const dataToPayment = {\n cpf: user.user.cpf,\n name: user.user.name,\n valueCharge: valueToCharge,\n institution: \"velotax-rendimentos\",\n impostoDevido: impostos.impostoDevido,\n totalImpostoDevido: impostos.impostoTotal,\n impostoTotalFinal: valueToCharge,\n totalRendimentos: impostos.totalRendimentos,\n baseTributaria: impostos.baseTributaria,\n totalDependentes: impostos.totalDependentes,\n totalDespesas: impostos.totalDespesas,\n totalPrevidencia: impostos.totalPrevidencia,\n totalPensao: impostos.totalPensao,\n totalDeducoes: impostos.totalDeducoes,\n aliquotaDevida: impostos.aliquotaDevida,\n impostoAcumulado: impostos.impostoAcumulado,\n gastosExterior: impostos.gastosExterior,\n juros: impostos.juros,\n multa: impostos.multa,\n transactions: monthStock,\n year,\n month: month + 1,\n };\n\n setPaymentData(dataToPayment);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [impostos, month, year]);\n\n const getRendimentos = useCallback(async () => {\n if (!monthView && !yearView) return;\n setLoading(true);\n const response = await api.get(`/dividendos/rendimentos`, {\n params: { year, month: month + 1 },\n });\n const { data } = response;\n setId(data._id || \"\");\n setMonthStock(data.monthStock || []);\n const currentImpostos = {\n totalRendimentos: data?.totalRendimentos || 0,\n baseTributaria: data?.baseTributaria || 0,\n aliquotaDevida: data?.aliquotaDevida || 0,\n impostoDevido: data?.impostoDevido || 0,\n totalDeducoes: data?.totalDeducoes || 0,\n totalDependentes: data?.totalDependentes || 0,\n totalDespesas: data?.totalDespesas || 0,\n totalPensao: data?.totalPensao || 0,\n totalPrevidencia: data?.totalPrevidencia || 0,\n juros: data?.juros || 0,\n multa: data?.multa || 0,\n impostoTotal:\n (data.impostoDevido || 0) + (data.juros || 0) + (data.multa || 0),\n impostoAcumulado: data?.impostoAcumulado || 0,\n gastosExterior: data?.gastosExterior || 0,\n payed: data?.payed || false,\n emitted: data?.emitted || false,\n };\n setImpostos(currentImpostos);\n setLoading(false);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n const getRendimento = useCallback(async () => {\n setLoading(true);\n const response = await api.get(\"/darf/getById\", {\n params: { id: darf?.id },\n });\n const { data } = response;\n setMonthStock(data.transactions || []);\n const currentImpostos = {\n totalRendimentos: data.totalRendimentos || 0,\n baseTributaria: data.baseTributaria || 0,\n aliquotaDevida: data.aliquotaDevida || 0,\n impostoDevido: data.impostoDevido || 0,\n totalDeducoes: data.totalDeducoes || 0,\n totalDependentes: data.totalDependentes || 0,\n totalDespesas: data.totalDespesas || 0,\n totalPensao: data.totalPensao || 0,\n totalPrevidencia: data.totalPrevidencia || 0,\n juros: data.juros || 0,\n multa: data.multa || 0,\n impostoAcumulado: data.impostoAcumulado || 0,\n impostoTotal:\n (data.impostoDevido || 0) + (data.juros || 0) + (data.multa || 0),\n gastosExterior: data.gastosExterior || 0,\n payed: data.payed || false,\n emitted: data.emitted || false,\n };\n setImpostos(currentImpostos);\n setLoading(false);\n }, [darf]);\n\n useEffect(() => {\n setPayData();\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [setPayData, impostos]);\n\n useEffect(() => {\n if (view && darf) getRendimento();\n else getRendimentos();\n }, [getRendimentos, getRendimento, view, darf]);\n\n const buildNewImpostos = useCallback(() => {\n return {\n monthStock,\n year,\n month: month + 1,\n juros: impostos.juros,\n multa: impostos.multa,\n userCode: user.user.cpf,\n total: impostos.impostoTotal,\n impostoDevido: impostos.impostoDevido,\n totalRendimentos: impostos.totalRendimentos,\n baseTributaria: impostos.baseTributaria,\n aliquotaDevida: impostos.aliquotaDevida,\n totalDependentes: impostos.totalDependentes,\n totalDespesas: impostos.totalDespesas,\n totalPrevidencia: impostos.totalPrevidencia,\n totalPensao: impostos.totalPensao,\n totalDeducoes: impostos.totalDeducoes,\n };\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [impostos, year, month, user]);\n\n const saveItems = useCallback(\n async (rendimentos, variable?, value?) => {\n setLoading(true);\n const newImpostos = buildNewImpostos();\n if (id !== \"\") {\n await api.put(\"/dividendos/rendimentos/\", {\n _id: id,\n ...newImpostos,\n monthStock: rendimentos || monthStock,\n [variable]: value,\n });\n } else {\n const { data } = await api.post(\"/rendimentos\", {\n ...newImpostos,\n monthStock: rendimentos || monthStock,\n [variable]: value,\n });\n setId(data._id);\n }\n setLoading(false);\n getRendimentos();\n },\n // eslint-disable-next-line react-hooks/exhaustive-deps\n [id, buildNewImpostos, getRendimentos, year]\n );\n\n const generateDarf = async (pixValue: boolean, justSave?: boolean) => {\n setLoading(true);\n const { data } = await MsDarf.post(\"/darf\", {\n month: month + 1,\n year,\n cpf: user.user.cpf,\n name: user.user.name,\n paymentInfo: {\n clientInfo: {\n document: user.user.cpf,\n email: user.user.email,\n name: user.user.name,\n phone: \"\",\n },\n },\n insertPix: pixValue,\n valueCharge: Math.max(\n 0,\n impostos.impostoDevido +\n impostos.impostoAcumulado -\n impostos.gastosExterior +\n impostos.juros +\n impostos.multa\n ),\n institution: \"exterior-warren\",\n impostoDevido: impostos.impostoDevido,\n totalImpostoDevido: impostos.impostoDevido,\n totalRendimentos: impostos.totalRendimentos,\n baseTributaria: impostos.baseTributaria,\n totalDependentes: impostos.totalDependentes,\n totalDespesas: impostos.totalDespesas,\n totalPrevidencia: impostos.totalPrevidencia,\n totalPensao: impostos.totalPensao,\n totalDeducoes: impostos.totalDeducoes,\n aliquotaDevida: impostos.aliquotaDevida,\n impostoAcumulado: impostos.impostoAcumulado,\n juros: impostos.juros,\n multa: impostos.multa,\n transactions: monthStock,\n from: 18,\n gastosExterior: impostos.gastosExterior,\n justSave,\n });\n\n if (!justSave) {\n window.location.href = data.darfUrl;\n }\n setLoading(false);\n closeModal();\n // if (impostos.impostoTotal <= maxDarfPrice) {\n // handleOpenPaymentModal();\n // }\n };\n\n const handleRegularize = (regular: boolean) => {\n setLoading(true);\n setEmitting(true);\n api\n .post(\"transaction/regularize\", {\n year,\n month: month + 1,\n regular,\n })\n .then((res) => {\n if (res.data?.darfUrl) {\n download(res.data.darfUrl);\n }\n closeModal();\n message.success(\"Dados salvos\");\n })\n .catch(() => message.error(errorMessage))\n .catch((err) => console.log(\"err\", err))\n .finally(() => {\n setEmitting(false);\n setLoading(false);\n });\n };\n\n const handleCloseItemModal = useCallback(() => {\n setShowRendimentoModal(false);\n unsetItem();\n }, []);\n\n const addItem = async (data: IMonthStock) => {\n data.tipo = \"Dividendos no exterior\"\n // if (data.tipo === \"Exterior\") {\n // data.impostoPagoExterior = data?.impostoPagoExterior || 0;\n // }\n const rendimentos = [...monthStock, data];\n setMonthStock(rendimentos);\n saveItems(rendimentos);\n handleCloseItemModal();\n };\n\n const editItem = (data: IMonthStock) => {\n data.tipo = \"Dividendos no exterior\"\n // if (data.tipo === \"Exterior\") {\n // data.impostoPagoExterior = data?.impostoPagoExterior || 0;\n // }\n const aux = [...monthStock];\n aux.splice(itemIndex, 1, data);\n setMonthStock(aux);\n saveItems(aux);\n handleCloseItemModal();\n };\n\n const removeAsset = async () => {\n const aux = [...monthStock];\n aux.splice(itemIndex, 1);\n setMonthStock(aux);\n saveItems(aux);\n if (!isSaveItens.current) isSaveItens.current = true;\n };\n\n const handleAdd = (event: React.MouseEvent) => {\n event.stopPropagation();\n unsetItem();\n setShowRendimentoModal(true);\n };\n\n const handleEdit = (item: IMonthStock, index: number) => {\n setShowRendimentoModal(true);\n setItemIndex(index);\n setItem(item);\n };\n\n const handleRemove = (item: IMonthStock, index: number) => {\n setShowDeleteModal(true);\n setItemIndex(index);\n setItem(item);\n };\n\n const unsetItem = () => {\n setItem(null);\n };\n\n const handleCloseDeleteModal = () => {\n setShowDeleteModal(false);\n unsetItem();\n };\n\n const onCloseHelpModal = () => {\n setHelpModal(undefined);\n };\n\n const handleOpenDarfModal = (value?: boolean) => {\n const pixValue = value ?? false;\n generateDarf(pixValue);\n };\n\n const handleOpenPaymentModal = () => {\n setPaymentModal(true);\n };\n\n const handleClosePaymentModal = () => {\n setPaymentModal(false);\n };\n\n const changeInputState = (variable: string, value: string) => {\n const auxValue =\n variable !== \"totalDependentes\" ? currencyToNumber(value) : Number(value);\n setImpostos((currentValue) => {\n return {\n ...currentValue,\n [variable]: auxValue,\n };\n });\n saveItems(undefined, variable, auxValue);\n };\n\n return (\n \n
\n \n \n {!hasPlan && (\n
\n \n Contrate o plano premium para ver os impostos devidos\n \n }\n onClick={() => {\n try {\n handlePlanModal();\n closeModal();\n } catch (error) {\n console.warn(error);\n }\n }}>\n PREMIUM\n \n
\n )}\n
\n {RendimentosImpostosDevidos.map((description) => (\n \n {loading ? : description.render(impostos)}\n \n ))}\n \n
\n\n \n\n \n Rendimentos\n }>\n {hasPlan && !view ? (\n }>\n Adicionar\n \n ) : null}\n m.tipo !== \"Pensão Alimentícia\")\n : monthStock\n }\n renderItem={(item, index) => (\n handleEdit(item, index)}\n icon={ }\n />,\n handleRemove(item, index)}\n icon={ }\n />,\n ]\n }>\n \n Rendimento #{index + 1}}\n description={\n \n
Tipo: {item.tipo}
\n
\n Valor: {formatCurrency(item.valorRendimento || 0)}\n
\n {item.cpfPagador && (\n
\n CPF do pagador: {maskCpf(item.cpfPagador)}\n
\n )}\n {item.cpfBeneficiario && (\n
\n CPF do beneficiário:{\" \"}\n {maskCpf(item.cpfBeneficiario)}\n
\n )}\n {(item?.tipo === \"Exterior\" ||\n item?.tipo === \"Dividendos no exterior\") && (\n
\n Imposto Pago no Exterior:{\" \"}\n {formatCurrency(item.impostoPagoExterior ?? 0)}\n
\n )}\n {item?.tipo === \"Dividendos no exterior\" && (\n <>\n
Ativo: {item.ativo ?? \"\"}
\n >\n )}\n
\n }\n />\n \n \n )}\n />\n
\n \n\n {hasPlan && }\n\n {hasPlan && (\n <>\n \n \n Deduções Tributárias\n \n
\n \n \n \n setHelpModal(ModalDependentes)}\n />\n Dependentes\n \n\n \n setHelpModal(ModalAutonomos)}\n />\n Gastos Autônomos\n \n\n \n setHelpModal(ModalPrevidencia)}\n />\n Gastos Previdência\n \n\n \n setHelpModal(ModalPensao)}\n />\n Gastos Pensão\n \n
\n \n
\n \n {\n changeInputState(\"totalDependentes\", e);\n }}\n />\n \n \n \n changeInputState(\"totalDespesas\", e.target.value)\n }\n />\n \n \n \n changeInputState(\"totalPrevidencia\", e.target.value)\n }\n />\n \n \n \n changeInputState(\"totalPensao\", e.target.value)\n }\n />\n \n
\n
\n \n >\n )}\n\n {hasPlan && }\n\n {hasPlan && (\n <>\n \n Imposto total \n \n {loading ? (\n \n ) : (\n formatCurrency(valorImpostoTotal)\n )}\n \n
\n \n {ImpostoTotalRendimentosAutonomos.map((description: any) => (\n \n {loading ? (\n \n ) : description.Component ? (\n \n ) : (\n formatCurrency(\n Number(impostos[description.id as keyof Iimpostos] || 0)\n )\n )}\n \n ))}\n \n >\n )}\n\n {!view && hasPlan && {ActionButtons}
}\n \n \n\n
maxPixPayment}\n disableCreditCardOption={\n (paymentData?.valueCharge || 0) > maxCreditCardPayment\n }\n show={paymentModal}\n onCancel={handleClosePaymentModal}\n paymentData={paymentData}\n callDarf={(value: boolean) => handleOpenDarfModal(value)}\n maxPixValue={maxPixPayment}\n maxCCValue={maxCreditCardPayment}\n selectedYear={year}\n closeDarfModal={closeModal}\n />\n \n {helpModal?.content}\n \n \n Promise.resolve(removeAsset())}\n body=\"Você realmente quer deletar esse ativo?\"\n />\n \n );\n};\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 128px;\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n }\n`;\n\nexport const Content = styled.div`\n background-color: var(--velotax-background-color);\n border-radius: 16px;\n margin: 0 auto;\n padding: 32px;\n min-width: 500px;\n max-width: 800px;\n h1 {\n font-size: 24px;\n position: relative;\n margin-bottom: 32px;\n color: var(--velotax-font-color);\n &::before {\n content: \"\";\n position: absolute;\n width: 4px;\n top: 0;\n left: -32px;\n height: 100%;\n background-color: var(--ant-primary-color);\n }\n }\n @media only screen and (max-device-width: 812px) {\n border-radius: 0;\n min-width: 100%;\n padding: 16px;\n padding-bottom: 32px;\n h1 {\n &::before {\n display: none;\n }\n }\n }\n`;\n","import React, { Dispatch, SetStateAction, useEffect, useState } from \"react\";\nimport { Col, Divider, Form, Input, message, Modal, Row } from \"antd\";\nimport Button from \"../../Button\";\nimport apis from \"../../../services/apiExterior\";\nimport { useAuth } from \"../../../contexts/AuthContext\";\n// import passfolioLogo from \"../../../assets/passfolio-logo.png\";\nimport { validationFieldRequired } from \"../../../utils/formValidations\";\nimport { IntegrationData } from \"../interfaces\";\n\ninterface IPassfolioIntegrationModalProps {\n visible: boolean;\n dependentId?: string;\n setVisible: Dispatch>;\n setLoadingIntegration: Dispatch>;\n integrationData?: IntegrationData;\n}\n\nconst PassfolioIntegrationModal: React.FC = ({\n visible,\n setVisible,\n setLoadingIntegration,\n integrationData,\n}) => {\n const { user } = useAuth();\n const [form] = Form.useForm();\n const [integrationParams, setParams] = useState({\n key: \"\",\n userId: \"\",\n secret: \"\",\n token: \"\",\n });\n\n useEffect(() => {\n if (integrationData && visible) {\n form.setFieldsValue({\n key: integrationData.key,\n secret: integrationData.secret,\n userId: integrationData.userId,\n });\n }\n }, [form, integrationData, visible]);\n\n const onCancel = () => {\n setVisible(false);\n form.resetFields();\n };\n\n const onFinish = (values: any) => {\n setLoadingIntegration(true);\n apis\n .post(\n \"/passfolio/authenticate\",\n {\n ...values,\n },\n {\n headers: {\n \"x-api-provider\": \"exterior-velotax\",\n },\n }\n )\n .then((response) => {\n const { data } = response;\n const { token, authenticatorId } = data;\n const { key, secret } = values;\n\n setParams({ key, secret, userId: authenticatorId, token });\n })\n .catch((err) => {\n if (err.response?.data?.message) {\n message.error(\n err.response?.data?.message ||\n \"Algo deu errado, tente novamente mais tarde\"\n );\n }\n setLoadingIntegration(false);\n });\n };\n\n const onClick = (values: any) => {\n setLoadingIntegration(true);\n setVisible(false);\n apis\n .post(\n \"/passfolio/integrate\",\n {\n ...values,\n ...integrationParams,\n email: user.user.email,\n },\n {\n headers: {\n \"x-api-provider\": \"exterior-velotax\",\n },\n }\n )\n .catch((err) => {\n if (err.response?.data?.message) {\n message.error(\n err.response?.data?.message ||\n \"Algo deu errado, tente novamente mais tarde\"\n );\n }\n setParams({\n key: \"\",\n userId: \"\",\n secret: \"\",\n token: \"\",\n });\n setLoadingIntegration(false);\n });\n };\n\n return (\n \n \n Integrar com a Warren\n \n }\n >\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n Cancelar\n \n \n \n \n \n \n Logar\n \n {\n onClick(form.getFieldsValue());\n }}\n >\n Integrar\n \n \n \n
\n \n \n );\n};\n\nexport default PassfolioIntegrationModal;\n","export default __webpack_public_path__ + \"static/media/warren-logo.1de4cd65.svg\";","import { message } from \"antd\";\nimport React, { useEffect, useState } from \"react\";\nimport apis from \"../../../services/apiExterior\";\nimport IntegrationButton from \"../IntegrationButton\";\nimport { IntegrationStatus } from \"../ExteriorIntegration\";\n// import passfolioLogo from \"../../../assets/passfolio-logo.png\";\nimport PassfolioIntegrationModal from \"./PassfolioIntegrationModal\";\nimport WarrenLogo from '../../../assets/warren-logo.svg'\n\ninterface IGetEstadoIntegracao {\n success: (data: any) => void;\n error: (data: any) => void;\n}\n\ninterface IPassfolioIntegrationProps {\n dependentId?: string;\n integrationCallback: (type: IntegrationStatus) => void;\n}\n\nconst PassfolioIntegration: React.FC = ({\n dependentId,\n integrationCallback,\n}) => {\n const [loading, setLoading] = useState(true);\n const [integrationModalVisible, setIntegrationModalVisible] = useState(false);\n const [integrado, setIntegrado] = useState(false);\n const [integrationData, setIntegrationData] = useState();\n\n const getEstadoIntegracao = ({ success, error }: IGetEstadoIntegracao) =>\n apis\n .get(\"/passfolio/integration-status\", {\n headers: {\n \"x-api-provider\": \"exterior-velotax\",\n },\n })\n .then(({ data }) => {\n setIntegrationData(data);\n success(data);\n })\n .catch((err) => {\n error(err);\n });\n\n useEffect(() => {\n getEstadoIntegracao({\n success: (data) => {\n if (!data.integrated && data.providerExists) {\n setLoading(true);\n setIntegrado(false);\n integrationCallback(\"carregando\");\n } else if (data.integrated) {\n setLoading(false);\n setIntegrado(true);\n integrationCallback(\"integrado\");\n }\n },\n error: () => {\n integrationCallback(\"erro\");\n },\n });\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, []);\n\n useEffect(() => {\n let interval: NodeJS.Timeout | null = null;\n if (loading) {\n interval = setInterval(\n () => {\n integrationCallback(\"carregando\");\n getEstadoIntegracao({\n success: (data: any) => {\n if (!data.integrated && data.providerExists) {\n integrationCallback(\"carregando\");\n }\n\n if (!data.providerExists) {\n setLoading(false);\n setIntegrado(false);\n }\n\n if (data.integrated) {\n setLoading(false);\n setIntegrado(true);\n message.success(\"Integração feita com sucesso!\");\n integrationCallback(\"integrado\");\n }\n },\n error: (err: any) => {\n setLoading(false);\n integrationCallback(\"erro\");\n if (err.response?.data?.message) {\n return message.error(err.response?.data?.message);\n }\n return message.error(\n \"Algo deu errado na integração com a Foxbit, verifique suas credenciais\"\n );\n },\n });\n },\n\n 3000\n );\n } else {\n if (interval) {\n clearInterval(interval);\n }\n }\n return () => clearInterval(interval as unknown as NodeJS.Timeout);\n // eslint-disable-next-line react-hooks/exhaustive-deps\n }, [loading, integrado]);\n\n return (\n <>\n \n \n >\n );\n};\n\nexport default PassfolioIntegration;\n","import { Col, Row } from \"antd\";\nimport PassfolioIntegration from \"../PassfolioIntegration\";\n\ninterface IExteriorIntegrationProps {\n id?: string;\n data?: any;\n dependentId?: string;\n setData?: ({ id, value, customData, from }: any, patch?: any) => void;\n customActions?: { [id: string]: (data: any) => void };\n}\n\nexport type IntegrationStatus = \"carregando\" | \"integrado\" | \"erro\";\n\nconst ExteriorIntegration = (props: IExteriorIntegrationProps) => {\n const { id, dependentId, setData } = props;\n\n const handleIntegration = (type: IntegrationStatus) => {\n setData?.({ id, value: type === \"carregando\" }, false);\n };\n\n return (\n \n \n \n \n {/* \n \n */}\n
\n );\n};\n\nexport default ExteriorIntegration;\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n display: flex;\n align-items: center;\n flex-direction: column;\n padding: 64px 32px 128px;\n\n div.ant-typography {\n font-size: 0.875rem !important;\n }\n\n h1 {\n width: 700px;\n margin: 0 auto 1.5rem;\n font-weight: 400;\n font-size: 2rem;\n position: relative;\n line-height: 2.5rem;\n padding-bottom: 1rem;\n color: var(--velotax-font-color-light);\n\n :after {\n content: \"\";\n position: absolute;\n left: 0;\n bottom: 0;\n height: 2px;\n width: 100px;\n background-color: var(--ant-primary-color);\n }\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 0;\n\n h1 {\n width: calc(100% - 48px);\n width: calc(100% - 48px);\n padding: 24px 0 16px;\n margin: 0 24px 1.5rem;\n }\n }\n`;\nexport const Content = styled.div`\n padding: 32px;\n width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n\n .last-update {\n display: block;\n font-size: 11px;\n letter-spacing: 1px;\n margin-bottom: 16px;\n }\n div.ant-typography {\n font-size: 18px;\n }\n .ant-typography {\n color: var(--velotax-font-color);\n text-align: justify;\n }\n @media only screen and (max-device-width: 812px) {\n padding: 24px;\n border-radius: 0;\n width: 100%;\n h2 {\n &::before {\n left: -24px;\n }\n }\n }\n`;\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n padding: 64px 32px 32px;\n\n h1 {\n width: 700px;\n margin: 0 auto 1.5rem;\n font-weight: 400;\n font-size: 2rem;\n position: relative;\n line-height: 2.5rem;\n padding-bottom: 1rem;\n color: var(--velotax-font-color-light);\n\n :after {\n content: \"\";\n position: absolute;\n left: 0;\n bottom: 0;\n height: 2px;\n width: 100px;\n background-color: var(--ant-primary-color);\n }\n }\n\n div.ant-typography {\n margin: 0 auto 1.5rem;\n font-size: 1rem;\n width: 700px;\n line-height: 1.5rem;\n text-align: justify;\n color: var(--velotax-font-color);\n }\n\n @media only screen and (max-device-width: 812px) {\n padding: 24px 24px 0;\n\n h1,\n div.ant-typography {\n width: 100%;\n }\n }\n`;\n\nexport const Content = styled.div`\n .container-buttons {\n display: flex;\n width: 100%;\n justify-content: space-between;\n button {\n margin: 0px 0px 12px;\n :first-of-type {\n margin-right: 8px;\n }\n }\n }\n padding: 32px;\n margin: 0 auto;\n width: 700px;\n border-radius: 4px;\n background-color: var(--velotax-background-color);\n\n .add-button {\n display: flex;\n margin: 0 0 1.5rem auto;\n }\n\n .ant-list {\n padding: 0;\n border-radius: 4px;\n background-color: var(--velotax-background-color-ghost);\n .ant-list-item {\n padding: 16px;\n &.error-border {\n border: 1px solid var(--ant-error-color);\n }\n :first-child {\n border-radius: 4px 4px 0 0;\n }\n :last-child {\n border-radius: 0 0 4px 4px;\n }\n }\n .error-border + .error-border {\n border-top: none;\n }\n .ant-list-item-action > li {\n padding: 0;\n }\n }\n .ant-list-item-action > li,\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color-light);\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n\n .alert-icon {\n padding-top: 1px;\n fill: var(--ant-error-color) !important;\n }\n\n @media only screen and (max-device-width: 812px) {\n width: calc(100% + 48px);\n margin-left: -24px;\n padding: 24px 24px 64px;\n\n .container-buttons {\n flex-wrap: wrap;\n button {\n margin: 0 0 8px !important;\n width: 100%;\n }\n }\n }\n`;\n","import moment from \"moment\";\nimport { checkDateExpiration, checkDateLimit, isValidDate } from \"../../utils\";\nimport {\n validationFieldRequired,\n validationNumberFieldPositive,\n} from \"../../utils/formValidations\";\nimport {\n CurrencyFormItem,\n DateFormItem,\n NumberFormItem,\n TextFormItem,\n} from \"../../constants/formItems\";\n\nconst TransferenciaCustodiaFormItems = {\n id: TextFormItem({\n name: \"id\",\n label: \"Id\",\n hidden: true,\n }),\n codigo: (disabled: boolean) =>\n TextFormItem({\n name: \"codigo\",\n label: \"Código\",\n disabled,\n rules: validationFieldRequired,\n }),\n quantidade: (disabled: boolean) =>\n NumberFormItem({\n name: \"quantidade\",\n label: \"Quantidade\",\n disabled,\n rules: validationFieldRequired,\n decimalScale: 8,\n }),\n dataCompra: DateFormItem({\n name: \"dataCompra\",\n mask: \"99/99/9999\",\n label: \"Data de compra\",\n placeholder: \"DD/MM/YYYY\",\n rules: [\n {\n validator: (rule, value) =>\n !value || !isValidDate(value)\n ? Promise.reject(new Error(\"Informe uma data válida\"))\n : value && !checkDateLimit(\"2017-01-01\", value)\n ? Promise.reject(\n new Error(\"Informe uma data a partir de 01/01/2017\")\n )\n : value &&\n !checkDateExpiration(moment().add(1, \"day\").toString(), value)\n ? Promise.reject(\n new Error(\n `Informe uma data até ${moment().format(\"DD/MM/YYYY\")}`\n )\n )\n : Promise.resolve(),\n },\n ],\n }),\n valorCompraUSDOrigemBRA: CurrencyFormItem({\n prefix: \"US$ \",\n name: \"valorCompraUSDOrigemBRA\",\n label: \"Valor de compra (USD, Origem BRA)\",\n placeholder: \"US$ 0,00\",\n rules: validationNumberFieldPositive,\n }),\n valorCompraUSDOrigemEUA: CurrencyFormItem({\n prefix: \"US$ \",\n name: \"valorCompraUSDOrigemEUA\",\n label: \"Valor de compra (USD, Origem EUA)\",\n placeholder: \"US$ 0,00\",\n rules: validationNumberFieldPositive,\n }),\n};\n\nexport const TransferenciaCustodiaRows = (disabled: boolean) => [\n [TransferenciaCustodiaFormItems.id],\n [\n TransferenciaCustodiaFormItems.codigo(disabled),\n TransferenciaCustodiaFormItems.quantidade(disabled),\n ],\n [TransferenciaCustodiaFormItems.dataCompra],\n [TransferenciaCustodiaFormItems.valorCompraUSDOrigemBRA],\n [TransferenciaCustodiaFormItems.valorCompraUSDOrigemEUA],\n];\n","import styled from \"styled-components\";\n\nexport const Container = styled.div`\n .ant-list {\n padding: 0;\n border-radius: 4px;\n background-color: var(--velotax-background-color-ghost);\n .ant-list-item {\n padding: 16px;\n &.error-border {\n border: 1px solid var(--ant-error-color);\n }\n :first-child {\n border-radius: 4px 4px 0 0;\n }\n :last-child {\n border-radius: 0 0 4px 4px;\n }\n }\n .error-border + .error-border {\n border-top: none;\n }\n .ant-list-item-action > li {\n padding: 0;\n }\n }\n .ant-list-item-action > li,\n .ant-list-item-meta-title,\n .ant-list-item-meta-description {\n color: var(--velotax-font-color-light);\n }\n .ant-list-item-meta-description {\n opacity: 0.75;\n }\n\n .alert-icon {\n fill: var(--ant-error-color) !important;\n }\n\n .row-margin {\n margin-top: 24px;\n }\n`;\n","import { Button } from \"@mui/material\";\nimport { FaRegEdit } from \"react-icons/fa\";\nimport { Col, List, Row, Skeleton, Tooltip, Typography } from \"antd\";\nimport { Dispatch, SetStateAction, useEffect, useState } from \"react\";\nimport { AiOutlineDelete, AiOutlineInfoCircle } from \"react-icons/ai\";\nimport {\n LoadingOutlined,\n DownloadOutlined,\n PlusCircleOutlined,\n} from \"@ant-design/icons\";\nimport { Container } from \"./styles\";\nimport { formatCurrency } from \"../../../../utils\";\nimport AntButton from \"../../../../components/Button\";\nimport { FormModal } from \"../../../../components/FormModal\";\nimport { DoacoesHerancasRows } from \"../../../../constants/darfBolsa\";\nimport moment from \"moment-timezone\";\n\nexport interface DoacoesHerancasProps {\n changed: boolean;\n loadingPut: boolean;\n loadingGet: boolean;\n setLoadingPut: Dispatch>;\n setLoadingGet: Dispatch>;\n doacaoHerancaEdit: any;\n doacoesHerancas: any[];\n showAddPositionModal: boolean;\n downloadDoacoesHerancasExcel: () => Promise;\n setDoacaoHerancaEdit: Dispatch>;\n setShowAddPositionModal: Dispatch>;\n saveDoacoesHerancas: () => void;\n addDoacaoHeranca: (item: any) => void;\n editDoacaoHeranca: (item: any) => void;\n handleEditDoacaoHeranca: (item: any, index: number) => void;\n handleRemoveDoacaoHeranca: (item: any, index: number) => void;\n getOldTransactionsInit: () => void;\n}\n\nexport const DoacoesHerancas: React.FC = ({\n loadingGet,\n loadingPut,\n doacoesHerancas,\n downloadDoacoesHerancasExcel,\n setDoacaoHerancaEdit,\n setShowAddPositionModal,\n handleEditDoacaoHeranca,\n handleRemoveDoacaoHeranca,\n addDoacaoHeranca,\n editDoacaoHeranca,\n showAddPositionModal,\n doacaoHerancaEdit,\n}) => {\n const [downloading, setDownloading] = useState(false);\n const [orders, setOrders] = useState([]);\n\n useEffect(()=>{\n if (doacoesHerancas?.length > 0)\n {\n const ordersLocal: any[] = [];\n doacoesHerancas?.map((op) => {\n op?.orders.map((or: any)=> {\n or['dateView'] = moment(or?.date, [\"DD/MM/YYYY\",\"YYYY-MM-DD\",]).format(\"DD/MM/YYYY\");\n ordersLocal.push(or);\n })\n })\n setOrders(ordersLocal);\n }\n }, [doacoesHerancas])\n\n async function download() {\n setDownloading(true);\n const file = await downloadDoacoesHerancasExcel();\n if (file) {\n const blob = new Blob([file], { type: \"xlsx\" });\n const element = document.createElement(\"a\");\n element.href = URL.createObjectURL(blob);\n element.download = \"doacoes_herancas.xlsx\";\n document.body.appendChild(element);\n element.click();\n element.remove();\n }\n setDownloading(false);\n }\n\n return (\n \n \n \n \n Doações ou heranças\n \n \n \n \n {orders.length === 0 ? (\n
\n ) : (\n \n ) : (\n \n )\n }\n onClick={download}>\n BAIXAR EXCEL\n \n )}\n {\n setDoacaoHerancaEdit({});\n setShowAddPositionModal(true);\n }}\n startIcon={\n loadingPut || loadingGet ? (\n \n ) : (\n \n )\n }>\n Adicionar Doação/Herança\n \n
\n \n \n !oldPosition.lendingSide\n ) ?? []\n }\n renderItem={(item, index) => (\n \n \n }\n type=\"text\"\n />\n ,\n ]\n : []),\n }\n onClick={() => handleEditDoacaoHeranca(item, index)}\n />,\n }\n onClick={() => handleRemoveDoacaoHeranca(item, index)}\n />,\n ]}>\n \n {item.code}}\n description={\n <>\n \n
\n Data: {moment(item.date, [\n \"DD/MM/YYYY\",\n \"YYYY-MM-DD\",\n ]).format(\"DD/MM/YYYY\")} \n
\n
\n Quantidade: {item.quantity} \n
\n
\n Custo médio:{\" \"}\n {formatCurrency(item.price ?? 0)} \n
\n
\n >\n }\n />\n \n \n )}\n />\n \n
\n\n {\n setShowAddPositionModal(false);\n }}\n title={\n doacaoHerancaEdit?._id || doacaoHerancaEdit?.id\n ? \"Editar ativo\"\n : \"Adicionar ativo\"\n }\n itemToEdit={\n doacaoHerancaEdit.id || doacaoHerancaEdit.code\n ? doacaoHerancaEdit\n : undefined\n }\n />\n \n );\n};\n","import { Typography } from \"antd\";\nimport { DoacoesHerancaData, Question } from \"./interfaces\";\nimport {\n DoacoesHerancas,\n DoacoesHerancasProps,\n} from \"./Components/DoacoesHerancas\";\n\nexport const validateDoacoesHerancaData = (data: DoacoesHerancaData) => {\n return true;\n};\n\nexport const questions: Question[] = [\n {\n id: \"doacoesHerancas\",\n title: \"Complete seus dados de doações e heranças\",\n children: (\n \n Lembre-se de clicar em finalizar para salvar seus dados\n \n ),\n formItems: [\n {\n name: \"doacoesHerancas\",\n FormItem: ({ rest }) => (\n \n ),\n },\n ],\n },\n];\n","import styled from \"styled-components\";\n\nexport const BolsaManualContainer = styled.div`\n height: 100%;\n overflow-y: hidden;\n h2 {\n margin: 0 0 0 0 !important;\n }\n .beauty-scrollbar{\n margin-top: 64px;\n }\n`;\n","import { Darf } from \"../../pages/Darf\";\nimport { Bolsa } from \"../../pages/Bolsa\";\nimport { Crypto } from \"../../pages/Crypto\";\nimport { MeiDas } from \"../../pages/MeiDas\";\nimport { Historic } from \"../../pages/Historic\";\nimport { Exterior } from \"../../pages/Exterior\";\nimport { Settings } from \"../../pages/Settings\";\nimport { DarfBolsa } from \"../../pages/DarfBolsa\";\nimport { MeiReport } from \"../../pages/MeiReport\";\nimport { Regulation } from \"../../pages/Regulation\";\nimport { ReportBolsa } from \"../../pages/ReportBolsa\";\nimport { Integration } from \"../../pages/Integration\";\nimport { BolsaManual } from \"../../pages/BolsaManual\";\nimport { Rendimentos } from \"../../pages/Rendimentos\";\nimport { DarfExterior } from \"../../pages/DarfExterior\";\nimport { AnnualReport } from \"../../pages/AnnualReport\";\nimport { HistoricBolsa } from \"../../pages/HistoricBolsa\";\nimport { ReportExterior } from \"../../pages/ReportExterior\";\nimport { DividendosBolsa } from \"../../pages/DividendosBolsa\";\nimport { Identification } from \"../../pages/MeiIdentification\";\nimport { IntegrationBolsa } from \"../../pages/IntegrationBolsa\";\nimport { IntegrationCrypto } from \"../../pages/IntegrationCrypto\";\nimport { DividendosExterior } from \"../../pages/DividendosExterior\";\nimport { IntegrationExterior } from \"../../pages/IntegrationExterior\";\nimport { IntegrationExteriorXP } from \"../../pages/IntegrationExteriorXP\";\nimport { TransferenciaCustodia } from \"../../pages/TransferenciaCustodia\";\nimport { DoacoesHeranca } from \"../../pages/DoacoesHeranca\";\n\nexport const Pages: { [key: string]: any } = {\n Darf,\n Bolsa,\n Crypto,\n MeiDas,\n Exterior,\n Historic,\n Settings,\n DarfBolsa,\n MeiReport,\n Regulation,\n ReportBolsa,\n Integration,\n BolsaManual,\n DoacoesHeranca,\n Rendimentos,\n AnnualReport,\n DarfExterior,\n HistoricBolsa,\n ReportExterior,\n Identification,\n DividendosBolsa,\n IntegrationBolsa,\n IntegrationCrypto,\n DividendosExterior,\n IntegrationExterior,\n TransferenciaCustodia,\n IntegrationExteriorXP,\n};\n","import { Button } from \"@mui/material\";\nimport { useMemo, useState } from \"react\";\nimport { useNavigate } from \"react-router-dom\";\nimport { AiOutlineLock } from \"react-icons/ai\";\nimport { LoadingOutlined } from \"@ant-design/icons\";\nimport { Col, Modal, Row, Typography, message } from \"antd\";\nimport { Card } from \"../Exterior/Cards\";\nimport { itensBolsa } from \"./itensBolsa\";\nimport { HomeContainer } from \"../Home/styles\";\nimport AntButton from \"../../components/Button\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport { UserPlanEnum } from \"../../constants/plans\";\nimport { download, errorMessage } from \"../../utils\";\nimport api from \"../../services/apiBolsa\";\nimport { YearReport } from \"../../constants/rendimentos\";\n\nexport const Bolsa = () => {\n const navigate = useNavigate();\n const { user, hasPlan, showUserPlanModal, userPlanModal } = useAuth();\n const [loading, setLoading] = useState(false);\n const [infoModal, toggleModal] = useState(null);\n const [reportCount, setReportCount] = useState(0);\n const [downloading, setDownloading] = useState(false);\n const [reportIsReady, setReportIsReady] = useState(false);\n\n const userPlanInfoStatus = useMemo(\n () => user.user?.userPlanInfoVelotax,\n [user.user?.userPlanInfoVelotax]\n );\n\n const handlePlanModal = () => {\n navigate('/planos')\n };\n\n const handleAnualReport = async (id: string) => {\n const blockHandle =\n !userPlanInfoStatus.active &&\n !userPlanInfoStatus.type &&\n !(\n userPlanInfoStatus.type === UserPlanEnum.VELOTAX_MAIN_BASIC ||\n userPlanInfoStatus.type === UserPlanEnum.VELOTAX_MAIN_PRO ||\n userPlanInfoStatus.type === UserPlanEnum.VELOTAX_MAIN_CONCIERGE ||\n userPlanInfoStatus.type === UserPlanEnum.VELOTAX_MAIN_PRO_MONTH ||\n userPlanInfoStatus.type === UserPlanEnum.WARREN_BASIC ||\n userPlanInfoStatus.type === UserPlanEnum.WARREN_PRO ||\n userPlanInfoStatus.type === UserPlanEnum.WARREN_CONCIERGE ||\n userPlanInfoStatus.type === UserPlanEnum.WARREN_PRO_MONTH\n );\n if (blockHandle) {\n handlePlanModal();\n } else {\n toggleModal({ year: YearReport } as any);\n }\n };\n\n const downloadAction = (record: any) => {\n setDownloading(true);\n api\n .post(\"/darf/report/download/anual-report\", {\n from: \"warren\",\n id: record._id,\n month: record.month,\n year: record.year,\n })\n .then((res) => {\n if (res.data.fileUrl) {\n download(res.data.fileUrl);\n }\n if (res.data.file) {\n message.success(\"Relatório enviado para o seu e-mail\");\n // downloadPDF(res.data.file);\n }\n })\n .catch(({ response }) =>\n message.error(response?.data?.message ?? errorMessage)\n )\n .finally(() => {\n setDownloading(false);\n toggleModal(null);\n });\n };\n\n const checkProcessorAction = (record: any) => {\n setLoading(true);\n setReportIsReady(false);\n setReportCount(0);\n const reportCountInterval = setInterval(() => {\n setReportCount((n) => n + 1);\n }, 1500);\n let showToast = false;\n toggleModal(record);\n const itvl = setInterval(() => {\n api\n .get(`/darf/report/download/anual-report?year=${record.year}`)\n .then((res) => {\n const { data } = res;\n const { year, status } = data;\n const isProcessing = record.year === year && status === \"processing\";\n if (isProcessing && showToast) return;\n if (isProcessing && !showToast) {\n showToast = true;\n return message.success(\n `Estamos processando os dados para seu relatório anual de ${year}`\n );\n }\n clearInterval(reportCountInterval);\n const reportFinishCountInterval = setInterval(() => {\n setReportCount((n) => {\n if (n > 100) {\n clearInterval(reportFinishCountInterval);\n setLoading(false);\n setReportIsReady(true);\n clearInterval(itvl);\n }\n return n + 1;\n });\n }, 50);\n })\n .catch(({ response }) => {\n const { data, status } = response;\n if (status === 401)\n setTimeout(\n () => navigate(\"/warren/bolsa-integration\"),\n 2000\n );\n message.error(data?.message ?? errorMessage);\n setLoading(false);\n clearInterval(itvl);\n });\n }, 5000);\n };\n\n return (\n \n \n
\n
Bolsa de valores \n
\n Selecione a opção que deseja utilizar\n \n
\n {itensBolsa.slice(0, 3).map((item) => (\n
\n ) : (\n item.icon\n )\n }\n onClick={\n item.premium && (!hasPlan || user?.user?.userPlanInfoVelotax?.type?.includes('BASIC'))\n ? () => handlePlanModal()\n : item.id === \"anual-report\"\n ? () => handleAnualReport(item.id)\n : undefined\n }\n >\n {item.content}\n {item.description && (\n
{item.description}
\n )}\n {!hasPlan || item.premium && (!hasPlan.premium && !hasPlan.concierge)\n ? (\n
}\n onClick={() => {\n handlePlanModal();\n }}\n >\n PREMIUM\n \n )\n :\n null}\n \n ))}\n
\n
\n
\n {\n toggleModal(null);\n }}\n footer={\n \n \n }\n onClick={() => {\n if (!loading && !downloading) {\n if (reportIsReady) {\n downloadAction(infoModal);\n } else {\n checkProcessorAction({ year: 2022 } as any);\n }\n }\n }}\n >\n {loading && reportCount <= 100 ? `${reportCount}% ` : \"\"}\n {downloading\n ? \"Baixando informe\"\n : loading\n ? \"Gerando informe\"\n : reportIsReady\n ? \"Baixar informe\"\n : \"Gerar informe\"}\n \n \n \n {\n if (!loading && !downloading) {\n navigate(\"/warren/bolsa-historic\");\n }\n }}\n >\n Ir para calculadora de DARF\n \n \n
\n }\n >\n \n \n );\n};\n","import { Button } from \"@mui/material\";\nimport { useMemo, useState } from \"react\";\nimport { useNavigate } from \"react-router-dom\";\nimport { AiOutlineLock } from \"react-icons/ai\";\nimport { LoadingOutlined } from \"@ant-design/icons\";\nimport { Col, Modal, Row, Typography, message } from \"antd\";\nimport { Card } from \"../Exterior/Cards\";\nimport { itensCrypto } from \"./itensCrypto\";\nimport { HomeContainer } from \"../Home/styles\";\nimport AntButton from \"../../components/Button\";\nimport { useAuth } from \"../../contexts/AuthContext\";\nimport { UserPlanEnum } from \"../../constants/plans\";\nimport { ReportAlertTitle } from \"../ReportExterior\";\nimport { download, errorMessage } from \"../../utils\";\nimport api from \"../../services/apiBolsa\";\nimport { YearReport } from \"../../constants/rendimentos\";\n\nexport const Crypto = () => {\n const navigate = useNavigate();\n const { user, hasPlan, showUserPlanModal, userPlanModal } = useAuth();\n const [loading, setLoading] = useState(false);\n const [infoModal, toggleModal] = useState(null);\n const [reportCount, setReportCount] = useState(0);\n const [downloading, setDownloading] = useState(false);\n const [reportIsReady, setReportIsReady] = useState(false);\n\n const userPlanInfoStatus = useMemo(\n () => user.user?.userPlanInfoVelotax,\n [user.user?.userPlanInfoVelotax]\n );\n\n const handlePlanModal = () => {\n navigate('/planos')\n };\n\n const handleAnualReport = async (id: string) => {\n const blockHandle =\n !userPlanInfoStatus.active &&\n !userPlanInfoStatus.type &&\n !(\n userPlanInfoStatus.type === UserPlanEnum.VELOTAX_MAIN_BASIC ||\n userPlanInfoStatus.type === UserPlanEnum.VELOTAX_MAIN_PRO ||\n userPlanInfoStatus.type === UserPlanEnum.VELOTAX_MAIN_CONCIERGE ||\n userPlanInfoStatus.type === UserPlanEnum.VELOTAX_MAIN_PRO_MONTH ||\n userPlanInfoStatus.type === UserPlanEnum.WARREN_BASIC ||\n userPlanInfoStatus.type === UserPlanEnum.WARREN_PRO ||\n userPlanInfoStatus.type === UserPlanEnum.WARREN_CONCIERGE ||\n userPlanInfoStatus.type === UserPlanEnum.WARREN_PRO_MONTH\n );\n if (blockHandle) {\n handlePlanModal();\n } else {\n toggleModal({ year: YearReport } as any);\n }\n };\n\n const downloadAction = (record: any) => {\n setDownloading(true);\n api\n .post(\"/darf/report/download/anual-report\", {\n from: \"warren\",\n id: record._id,\n month: record.month,\n year: record.year,\n })\n .then((res) => {\n if (res.data.fileUrl) {\n download(res.data.fileUrl);\n }\n if (res.data.file) {\n message.success(\"Relatório enviado para o seu e-mail\");\n // downloadPDF(res.data.file);\n }\n })\n .catch(({ response }) =>\n message.error(response?.data?.message ?? errorMessage)\n )\n .finally(() => {\n setDownloading(false);\n toggleModal(null);\n });\n };\n\n const checkProcessorAction = (record: any) => {\n setLoading(true);\n setReportIsReady(false);\n setReportCount(0);\n const reportCountInterval = setInterval(() => {\n setReportCount((n) => n + 1);\n }, 1500);\n let showToast = false;\n toggleModal(record);\n const itvl = setInterval(() => {\n api\n .get(`/darf/report/download/anual-report?year=${record.year}`)\n .then((res) => {\n const { data } = res;\n const { year, status } = data;\n const isProcessing = record.year === year && status === \"processing\";\n if (isProcessing && showToast) return;\n if (isProcessing && !showToast) {\n showToast = true;\n return message.success(\n `Estamos processando os dados para seu relatório anual de ${year}`\n );\n }\n clearInterval(reportCountInterval);\n const reportFinishCountInterval = setInterval(() => {\n setReportCount((n) => {\n if (n > 100) {\n clearInterval(reportFinishCountInterval);\n setLoading(false);\n setReportIsReady(true);\n clearInterval(itvl);\n }\n return n + 1;\n });\n }, 50);\n })\n .catch(({ response }) => {\n const { data, status } = response;\n if (status === 401)\n setTimeout(\n () => navigate(\"/warren/bolsa-integration\"),\n 2000\n );\n message.error(data?.message ?? errorMessage);\n setLoading(false);\n clearInterval(itvl);\n });\n }, 5000);\n };\n\n return (\n \n \n
\n
Criptoativos \n
\n Selecione a opção que deseja utilizar\n \n
\n {itensCrypto.slice(0, 3).map((item) => (\n
\n ) : (\n item.icon\n )\n }\n onClick={\n item.premium && !hasPlan\n ? () => handlePlanModal()\n : item.id === \"anual-report\"\n ? () => handleAnualReport(item.id)\n : undefined\n }\n >\n {item.content}\n {item.description && (\n
{item.description}
\n )}\n {item.premium && !hasPlan && (\n
}\n onClick={() => {\n handlePlanModal();\n }}\n >\n PREMIUM\n \n )}\n \n ))}\n
\n
\n
\n {\n toggleModal(null);\n }}\n footer={\n \n \n }\n onClick={() => {\n if (!loading && !downloading) {\n if (reportIsReady) {\n downloadAction(infoModal);\n } else {\n checkProcessorAction({ year: YearReport } as any);\n }\n }\n }}\n >\n {loading && reportCount <= 100 ? `${reportCount}% ` : \"\"}\n {downloading\n ? \"Baixando relatório\"\n : loading\n ? \"Gerando relatório\"\n : reportIsReady\n ? \"Baixar relatório\"\n : \"Gerar relatório\"}\n \n \n \n {\n if (!loading && !downloading) {\n navigate(\"/warren/cripto-historic\");\n }\n }}\n >\n Ir para calculadora de DARF\n \n \n
\n }\n >\n \n Certifique-se de que todos os meses estejam com a situação \"Regular\"\n na Calculadora de DARF.\n \n \n \n );\n};\n","import { BsDownload } from \"react-icons/bs\";\nimport { BiHelpCircle } from \"react-icons/bi\";\nimport { useCallback, useEffect, useMemo, useState } from \"react\";\nimport { CheckCircleOutlined, CloseCircleOutlined } from \"@ant-design/icons\";\nimport {\n Col,\n List,\n message,\n Modal,\n Row,\n Select,\n Skeleton,\n Typography,\n} from \"antd\";\nimport apis from \"../../services/apis\";\nimport Button from \"../../components/Button\";\nimport { Page } from \"../../constants/brokers\";\nimport { PageInfo } from \"../../components/PageInfo\";\nimport { PlanModal } from \"../../components/PlanModal\";\nimport { PlanEnum, Plans } from \"../../constants/plans\";\nimport { useBroker } from \"../../contexts/BrokerContext\";\nimport { ConfirmModal, Container, Content } from \"./styles\";\nimport { StyledTag } from \"../Darf/Components/PayTags/styles\";\nimport { MEIAtividadePrincipalValue } from \"../../constants/identification\";\nimport {\n DasHelpModal,\n MeiDasStatusColorEnum,\n MeiDasStatusTextEnum,\n MeiReportDasStatusEnum,\n} from \"../../constants/das\";\nimport {\n download,\n errorMessage,\n formatDateHour,\n monthsExtended,\n} from \"../../utils\";\n\ninterface MeiDasResponse {\n year: number;\n month: number;\n path?: string;\n price?: number;\n payed?: boolean;\n issueDate?: string;\n requestDate?: string;\n inssBeneficiary?: boolean;\n status?: MeiReportDasStatusEnum;\n _id?: string;\n}\n\ninterface MeiDasProps {\n item: Page;\n}\n\nexport const MeiDas: React.FC = ({ item }) => {\n const initialYear = new Date().getFullYear();\n const {\n currentBroker,\n integration: { plans },\n currentPage,\n } = useBroker();\n\n const [loading, setLoading] = useState(false);\n const [year, setYear] = useState(initialYear);\n const [showModal, setShowModal] = useState(false);\n const [showHelpModal, setShowHelpModal] = useState(false);\n const [data, setData] = useState>();\n const [showPlanModal, setShowPlanModal] = useState(false);\n\n const { apiUrl } = item.settings;\n\n const yearOptions = Array.from(\n Array(new Date().getFullYear() - currentBroker.initialYear + 1)\n )\n .map((y, i) => ({\n value: currentBroker.initialYear + i,\n label: `${currentBroker.initialYear + i}`,\n }))\n .reverse();\n\n const dasMonths = useMemo(() => {\n const today = new Date();\n const currentMonth = today.getMonth();\n const currentYear = today.getFullYear();\n return Array.from(Array(year === currentYear ? currentMonth + 1 : 12)).map(\n (item, index) => ({\n year,\n month: index + 1,\n })\n );\n }, [year]);\n\n const mainActivity = useMemo(() => {\n const index = data?.findIndex((item) => item.price && item.price > 0);\n const keys = Object.keys(MEIAtividadePrincipalValue).map((key) =>\n Number(key)\n );\n return data &&\n data.length > 0 &&\n typeof index === \"number\" &&\n index >= 0 &&\n typeof data[index].price === \"number\" &&\n keys.includes(data[index].price!)\n ? MEIAtividadePrincipalValue[data[index].price!]\n : \"\";\n }, [data]);\n\n const getDasList = useCallback(() => {\n setLoading(true);\n (currentPage?.api || apis)\n .get(apiUrl, { params: { year } })\n .then((response) =>\n setData(\n dasMonths.map(\n (item) =>\n (response.data ?? []).find(\n (responseItem) => responseItem.month === item.month\n ) ?? item\n )\n )\n )\n .catch(() => message.error(errorMessage))\n .finally(() => setLoading(false));\n }, [apiUrl, year, dasMonths, currentPage]);\n\n const pay = useCallback(\n (payed: boolean, item: any) => {\n setLoading(true);\n (currentPage?.api || apis)\n .put(`${apiUrl}/${item._id}`, { payed })\n .then(() => {\n setData((data) =>\n (data || []).map((das) => ({\n ...das,\n payed: item._id === das._id ? payed : das.payed,\n }))\n );\n })\n .catch(() => message.error(errorMessage))\n .finally(() => setLoading(false));\n },\n [apiUrl, currentPage]\n );\n\n const downloadAction = (item: MeiDasResponse, index: number) => {\n if (item.path) {\n download(item.path);\n }\n };\n\n const handleEmit = (item: MeiDasResponse) => {\n setLoading(true);\n (currentPage?.api || apis)\n .post(apiUrl, {\n ...item,\n mainActivity,\n inssBeneficiary: !!item.inssBeneficiary,\n })\n .then(() => {\n getDasList();\n handleModal();\n })\n .catch(() => message.error(errorMessage))\n .finally(() => {\n setLoading(false);\n });\n };\n\n const checkIfPayed = (item: MeiDasResponse) => {\n if (plans?.name === PlanEnum.PRO_V1 && plans?.status === \"active\") {\n handleEmit(item);\n } else {\n setShowPlanModal(true);\n }\n };\n\n const inssAction = (item: MeiDasResponse) => {\n setData((data) =>\n data?.map((das) =>\n das.month === item.month\n ? {\n ...das,\n inssBeneficiary: !das.inssBeneficiary,\n }\n : das\n )\n );\n };\n\n const payAction = (value: boolean, record: any) => {\n pay(value, record);\n };\n\n const handleModal = () => {\n setShowModal((show) => !show);\n };\n\n const handleHelpModal = () => {\n setShowHelpModal((show) => !show);\n };\n\n const onChangeYear = (value: string) => {\n setYear(parseInt(value));\n };\n\n useEffect(() => {\n getDasList();\n }, [getDasList, year]);\n\n return (\n \n \n \n Lista de Guias (DAS) \n \n \n \n Mantenha o pagamento das suas guias mensais do MEI em dia.\n \n \n \n }\n />\n \n \n \n Obs.: Marque se você é Beneficiário do INSS antes de emitir a\n guia.\n \n \n
\n\n \n\n Selecione o ano: \n trigger.parentElement}\n />\n\n (\n \n Pago?\n payAction(!item.payed, item)}\n icon={\n item.payed ? (\n \n ) : (\n \n )\n }\n />\n >\n ),\n item.status && item.path && (\n downloadAction(item, index)}\n icon={