// ==UserScript==
// @name Glacier Tax Prep Stock Transactions Helper
// @namespace http://tampermonkey.net/
// @version 1.3
// @description Help nonresident alien fill in the 1099-B stock transactions by importing transactions from local CSV file
// @author Tony 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/TonyCSB/Glacier-tax-1099-B-Stock-Transactions-Helper/issues/new' target="_blank">issue</a> on <a href='https://github.com/TonyCSB/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()
}
}
}
}