您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
Help nonresident alien fill in the 1099-B stock transactions by importing transactions from local CSV file
// ==UserScript== // @name Glacier Tax Prep Stock Transactions Helper // @namespace http://tampermonkey.net/ // @version 1.4 // @description Help nonresident alien fill in the 1099-B stock transactions by importing transactions from local CSV file // @author Sibei Chen, JiahaoShan // @match https://www.glaciertax.com/IRSForm/StockTransaction* // @icon  // @require https://greasyfork.org/scripts/370520-super-gm-set-and-get/code/Super%20GM%20set%20and%20get.js?version=614650 // @grant GM_getValue // @grant GM_setValue // @license MIT // ==/UserScript== var transactions = null; var pid = null; (function () { 'use strict'; loadLayout(); var fileInput = document.getElementById('fileInput'); var payorId = document.getElementById("payorId"); var button = document.getElementById("process"); payorId.addEventListener("change", _ => { if (!validatePayorId()) { fileInput.disabled = true; button.disabled = true; } else { fileInput.disabled = false; pid = payorId.value; } }); fileInput.addEventListener("change", event => { if (fileInput.files.length != 0) { button.disabled = false; } else { button.disabled = true; } }); button.addEventListener("click", _ => { importFile(fileInput.files[0]); }); if (GM_SuperValue.get("status", "") == "importing") { fillTransaction(); } else { GM_SuperValue.set("transactions", 0); GM_SuperValue.set("payorId", 0); GM_SuperValue.set("index", 0); } })(); function loadLayout() { var css = document.createElement("style"); css.innerHTML = ".myInput {margin-left: 20px;}"; document.body.appendChild(css); var availablePayorIds = getPayorIds(); var options = `<option value='' disabled selected>Select Payor ID (EIN)</option>\n`; for (let id of availablePayorIds) { options += `<option value='${id}'>${id}</option>\n`; } var importerHTML = ` <div id='helper'> <h2>Glacier Tax Prep Form 1099-B Stock Transactions Importer</h2> <fieldset> <legend>Instruction</legend> <ol> <li> Add 1099-B forms in the previous page. If you get Consolidated Form 1099 PDF from Robinhood, <a href='https://github.com/joshfraser/robinhood-to-csv' target='_blank'>you may export it to .csv file</a>. </li> <li>Select a desired Payor ID (EIN) in the dropdown list.</li> <li> Choose the corresponding local .csv file. Make sure your .csv file follow the format strictly as below: </br> <code> name,acquired,sold,proceeds,cost</br> ALPHABET INC CLASS A COMMON STOCK,10/19/2016,11/03/2016,779.07,824.52</br> TESLA MOTORS INC,12/02/2016,12/06/2016,1468.19,1460.64 </code> </li> <li>The "Start" button will be enabled once valid EIN has been selected and csv file has been provided.</li> </ol> <ul> <li> Disclaimer: This greasyfork user script is NOT an official tool from Glacier Tax Prep. </li> <li> Please submit an <a href='https://github.com/SibeiC/Glacier-tax-1099-B-Stock-Transactions-Helper/issues/new' target="_blank">issue</a> on <a href='https://github.com/SibeiC/Glacier-tax-1099-B-Stock-Transactions-Helper' target="_blank">GitHub</a> for bug reports and feature requests. </li> </ul> </fieldset> <fieldset> <legend>Select Payor ID (EIN) and Choose .csv File</legend> <span class='myInput'> <label for="payorId">Payor ID (EIN): </label> <select name="payorId" id="payorId"> ${options} </select> </span> <span class='myInput'><input id='fileInput' type='file' id='input' accept='text/*,.csv' disabled></span> <span class='myInput'><button id='process' disabled>Start</button></span> <span class='myInput' style='display:none;color:white;font-size:1.2em;background-color:red;' id='errorMessage'></span> </fieldset> </div> `; $("#main").prepend(importerHTML); } function getPayorIds() { var payorIds = []; var trs = $('tbody').find('tr'); for (var i = 0; i < trs.length; ++i) { if (trs[i].childElementCount == 4) { if ($(trs[i]).find("td:nth-child(2)")[0].innerText == "Total Proceeds (Box 2) 1099-B") { payorIds.push($(trs[i]).find("td:nth-child(1)")[0].innerText); } } } return payorIds; } function validatePayorId() { return getPayorIds().includes(payorId.value); } function importFile(file) { var textType = /.*\.csv/; if (file.name.match(textType)) { var reader = new FileReader(); reader.onload = _ => { var lines = processData(reader.result); if (!lines) { fileInput.value = ""; return; } transactions = lines; hideErrorMessage(); GM_SuperValue.set("transactions", transactions); GM_SuperValue.set("payorId", pid); GM_SuperValue.set("index", 0); GM_SuperValue.set("status", "importing") fillTransaction(); } reader.readAsText(file); } else { $("#process").hide(); showErrorMessage("ERROR: only csv with header 'name,acquired,sold,proceeds,cost' is supported!"); } } function getTotalProceeds() { if (!validatePayorId()) return -1; var trs = $('tbody').find('tr'); for (var i = 0; i < trs.length; ++i) { var isTarget = $(trs[i]).find("td")[0].innerText == payorId.value; if (isTarget) { return $(trs[i]).find("td:nth-child(3)").text().trim(); } } } function validateDate(dateString) { return dateString.match(/^(\d{1,2})\/(\d{1,2})\/(\d{4})$/) ? true : false; } function validateInputFile(lines) { var inputTotalProceeds = 0; for (var i = 0; i < lines.length; i++) { if (!lines[i][0] || lines[i][0] == "") { showErrorMessage(getErrorMessage(i, "empty name")); return false; } if (!lines[i][1] || !validateDate(lines[i][1])) { showErrorMessage(getErrorMessage(i, "invalid acquired date")); return false; } if (!lines[i][2] || !validateDate(lines[i][2])) { showErrorMessage(getErrorMessage(i, "invalid sold date")); return false; } if (!lines[i][3] || isNaN(lines[i][3]) || parseFloat(lines[i][3]) < 0) { showErrorMessage(getErrorMessage(i, "invalid proceeds number")); return false; } if (!lines[i][4] || isNaN(lines[i][4]) || parseFloat(lines[i][4]) < 0) { showErrorMessage(getErrorMessage(i, "invalid cost number")); return false; } var acquiredDate = new Date(lines[i][1]); var soldDate = new Date(lines[i][2]); if (acquiredDate.getTime() > soldDate.getTime()) { showErrorMessage(getErrorMessage(i, "invalid acquired date or sold date")); return false; } inputTotalProceeds += parseFloat(lines[i][3]); } var inputTotal = inputTotalProceeds.toFixed(2); var totalProceeds = parseFloat(getTotalProceeds()).toFixed(2); if (totalProceeds == -1.00) { return false; } else if (inputTotal != totalProceeds) { alert("Total proceeds in the input file (" + inputTotal + ") doesn't match the total proceeds in the 1099 (" + totalProceeds + "). You may proceed, but it may cause errors."); return true; } return true; } function processData(allText) { var allTextLines = allText.split(/\r\n|\n/); var headers = allTextLines[0].split(','); if (headers.length != 5) { showErrorMessage('ERROR: Invalid number of headers'); return false; } if (headers[0] != 'name' || headers[1] != 'acquired' || headers[2] != 'sold' || headers[3] != 'proceeds' || headers['4'] != 'cost') { showErrorMessage("ERROR: only csv with header 'name,acquired,sold,proceeds,cost' is supported!"); return false; } var lines = []; for (var i = 1; i < allTextLines.length; i++) { var data = allTextLines[i].split(','); if (data.length == headers.length) { var tarr = []; for (var j = 0; j < headers.length; j++) { if (headers[j] == 'acquired') { var acquiredData = data[j].split(' '); if (acquiredData && acquiredData.length > 1) { data[j] = acquiredData[acquiredData.length - 1]; } } tarr.push(data[j]); } lines.push(tarr); } } return validateInputFile(lines) ? lines : false; } function hideErrorMessage() { $("#errorMessage").hide(); } function showErrorMessage(message) { $("#errorMessage").css("background-color", "red"); $("#errorMessage").show().html(message); } function getErrorMessage(lineNumber, message) { return "ERROR: The " + (lineNumber + 1) + getOrdinalIndicator(lineNumber + 1) + " line has " + message; } // GM_SuperValue.set("transactions", transactions); // GM_SuperValue.set("payorId", pid); // GM_SuperValue.set("index", 0); // GM_SuperValue.set("status", "importing") function fillTransaction() { var transactions = GM_SuperValue.get("transactions"); var pid = GM_SuperValue.get("payorId"); var index = GM_SuperValue.get("index"); if (index === undefined) { index = 0; } var filling = $("#Name").length ? true : false; if (filling) { var transaction = transactions[index]; GM_SuperValue.set("index", index + 1); console.log(index); console.log(transaction); if (index + 1 >= transactions.length) { GM_SuperValue.set("status", "done"); location.reload(false); } $("#Name").val(transaction[0]); if ($("#PurchasedDateString").length != 0) { $("#PurchasedDateString").val(transaction[1]); } $("#SoldDateString").val(transaction[2]); $("#SalesPrice").val(transaction[3]); $("#PurchasePrice").val(transaction[4]); document.forms[0].submit(); } else { var trs = $('tbody').find('tr'); for (var i = 0; i < trs.length; ++i) { var isTarget = $(trs[i]).find("td")[0].innerText == pid; if (isTarget && index < transactions.length) { $(trs[i]).find("td:nth-child(4) a:nth-child(1)")[0].click() } } } }