Glacier Tax Prep Stock Transactions Helper

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.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()
            }
        }
    }
}