// http://stackoverflow.com/questions/9847580/how-to-detect-safari-chrome-ie-firefox-and-opera-browser var isOpera = !!window.opera || navigator.userAgent.indexOf(' OPR/') >= 0; var isFirefox = typeof InstallTrigger !== 'undefined'; // Firefox 1.0+ var isSafari = Object.prototype.toString.call(window.HTMLElement).indexOf('Constructor') > 0; var isChrome = !!window.chrome && !isOpera; // Chrome 1+ var isChromium = isChrome && navigator.userAgent.indexOf('Chromium') >= 0; var isIE = /*@cc_on!@*/false || !!document.documentMode; // At least IE6 function GetReportTemplate4(doc, invoice, layout, checkMath) { var client = invoice.client; var account = invoice.account; var currencyId = client.currency_id; if (invoice.image) { var left = layout.headerRight - invoice.imageWidth; doc.addImage(invoice.image, 'JPEG', left, 30, invoice.imageWidth, invoice.imageHeight); } /* table header */ doc.setDrawColor(200,200,200); doc.setFillColor(230,230,230); var left = layout.headerLeft - layout.tablePadding; var top = layout.headerTop + layout.rowHeight + 4; var width = layout.headerRight - layout.headerLeft + (2 * layout.tablePadding); var height = layout.rowHeight + 1; if (invoice.po_number) { top += layout.rowHeight; } if (invoice.due_date) { top += layout.rowHeight; } doc.rect(left, top, width, height, 'FD'); doc.setFontSize(10); doc.setFontType("normal"); displayAccount(doc, invoice, layout.marginLeft, layout.accountTop, layout); displayClient(doc, invoice, layout.marginLeft, layout.headerTop, layout); displayInvoice(doc, invoice, layout.headerLeft, layout.headerTop, layout, layout.headerRight); var headerY = layout.headerTop; var total = 0; doc.setDrawColor(200,200,200); doc.setFillColor(230,230,230); var left = layout.marginLeft - layout.tablePadding; var top = layout.tableTop - layout.tablePadding; var width = layout.headerRight - layout.marginLeft + (2 * layout.tablePadding); var height = layout.rowHeight + 2; doc.rect(left, top, width, height, 'FD'); displayInvoiceHeader(doc, invoice, layout); var y = displayInvoiceItems(doc, invoice, layout); doc.setFontSize(10); /* table footer */ /* doc.setDrawColor(200,200,200); doc.setLineWidth(1); doc.line(layout.marginLeft - layout.tablePadding, x, layout.lineTotalRight+layout.tablePadding, x); */ displayNotesAndTerms(doc, layout, invoice, y+20); y += displaySubtotals(doc, layout, invoice, y+20, 480) + 20; if (checkMath && NINJA.parseFloat(total).toFixed(4) != NINJA.parseFloat(invoice.amount).toFixed(4)) { var doc = new jsPDF('p', 'pt'); doc.setFont('Helvetica',''); doc.setFontSize(10); doc.text(100, 100, "An error occurred, please try again later."); onerror('Failed to generate PDF ' + total + ', ' + invoice.amount ); return doc; } doc.setDrawColor(200,200,200); doc.setFillColor(230,230,230); var left = layout.footerLeft - layout.tablePadding; var top = y - layout.tablePadding; var width = layout.headerRight - layout.footerLeft + (2 * layout.tablePadding); var height = layout.rowHeight + 2; doc.rect(left, top, width, height, 'FD'); doc.setFontType("bold"); doc.text(layout.footerLeft, y, 'Balance Due'); total = formatMoney(total - (invoice.amount - invoice.balance), currencyId); var totalX = layout.headerRight - (doc.getStringUnitWidth(total) * doc.internal.getFontSize()); doc.text(totalX, y, total); doc.setFontType("normal"); doc.text(layout.marginLeft, 790, "Created by InvoiceNinja.com"); return doc; } function generatePDF(invoice, checkMath) { invoice = calculateAmounts(invoice); report_id=invoice.invoice_design_id; doc= GetPdf(invoice,checkMath,report_id); return doc; } /* Handle converting variables in the invoices (ie, MONTH+1) */ function processVariables(str) { if (!str) return ''; var variables = ['MONTH','QUARTER','YEAR']; for (var i=0; i 1) { offset = match.split('+')[1]; } else if (match.split('-').length > 1) { offset = parseInt(match.split('-')[1]) * -1; } str = str.replace(match, getDatePart(variable, offset)); } } return str; } function getDatePart(part, offset) { offset = parseInt(offset); if (!offset) { offset = 0; } if (part == 'MONTH') { return getMonth(offset); } else if (part == 'QUARTER') { return getQuarter(offset); } else if (part == 'YEAR') { return getYear(offset); } } function getMonth(offset) { var today = new Date(); var months = [ "January", "February", "March", "April", "May", "June", "July", "August", "September", "October", "November", "December" ]; var month = today.getMonth(); month = parseInt(month) + offset; month = month % 12; if (month < 0) { month += 12; } return months[month]; } function getYear(offset) { var today = new Date(); var year = today.getFullYear(); return parseInt(year) + offset; } function getQuarter(offset) { var today = new Date(); var quarter = Math.floor((today.getMonth() + 3) / 3); quarter += offset; quarter = quarter % 4; if (quarter == 0) { quarter = 4; } return 'Q' + quarter; } /* Set the defaults for DataTables initialisation */ $.extend( true, $.fn.dataTable.defaults, { "sDom": "t<'row-fluid'<'span6'i><'span6'p>>", //"sDom": "<'row'<'span6'l><'span6'f>r>t<'row'<'span6'i><'span6'p>>", "sPaginationType": "bootstrap", //"bProcessing": true, //"iDisplayLength": 50, "bInfo": true, "oLanguage": { //"sLengthMenu": "_MENU_ records per page" "sLengthMenu": "_MENU_", "sSearch": "" } //"sScrollY": "500px", } ); /* Default class modification */ $.extend( $.fn.dataTableExt.oStdClasses, { "sWrapper": "dataTables_wrapper form-inline" } ); /* API method to get paging information */ $.fn.dataTableExt.oApi.fnPagingInfo = function ( oSettings ) { return { "iStart": oSettings._iDisplayStart, "iEnd": oSettings.fnDisplayEnd(), "iLength": oSettings._iDisplayLength, "iTotal": oSettings.fnRecordsTotal(), "iFilteredTotal": oSettings.fnRecordsDisplay(), "iPage": oSettings._iDisplayLength === -1 ? 0 : Math.ceil( oSettings._iDisplayStart / oSettings._iDisplayLength ), "iTotalPages": oSettings._iDisplayLength === -1 ? 0 : Math.ceil( oSettings.fnRecordsDisplay() / oSettings._iDisplayLength ) }; }; /* Bootstrap style pagination control */ $.extend( $.fn.dataTableExt.oPagination, { "bootstrap": { "fnInit": function( oSettings, nPaging, fnDraw ) { var oLang = oSettings.oLanguage.oPaginate; var fnClickHandler = function ( e ) { e.preventDefault(); if ( oSettings.oApi._fnPageChange(oSettings, e.data.action) ) { fnDraw( oSettings ); } }; $(nPaging).addClass('pagination').append( '' ); var els = $('a', nPaging); $(els[0]).bind( 'click.DT', { action: "previous" }, fnClickHandler ); $(els[1]).bind( 'click.DT', { action: "next" }, fnClickHandler ); }, "fnUpdate": function ( oSettings, fnDraw ) { var iListLength = 5; var oPaging = oSettings.oInstance.fnPagingInfo(); var an = oSettings.aanFeatures.p; var i, ien, j, sClass, iStart, iEnd, iHalf=Math.floor(iListLength/2); if ( oPaging.iTotalPages < iListLength) { iStart = 1; iEnd = oPaging.iTotalPages; } else if ( oPaging.iPage <= iHalf ) { iStart = 1; iEnd = iListLength; } else if ( oPaging.iPage >= (oPaging.iTotalPages-iHalf) ) { iStart = oPaging.iTotalPages - iListLength + 1; iEnd = oPaging.iTotalPages; } else { iStart = oPaging.iPage - iHalf + 1; iEnd = iStart + iListLength - 1; } for ( i=0, ien=an.length ; i'+j+'') .insertBefore( $('li:last', an[i])[0] ) .bind('click', function (e) { e.preventDefault(); oSettings._iDisplayStart = (parseInt($('a', this).text(),10)-1) * oPaging.iLength; fnDraw( oSettings ); } ); } // Add / remove disabled classes from the static elements if ( oPaging.iPage === 0 ) { $('li:first', an[i]).addClass('disabled'); } else { $('li:first', an[i]).removeClass('disabled'); } if ( oPaging.iPage === oPaging.iTotalPages-1 || oPaging.iTotalPages === 0 ) { $('li:last', an[i]).addClass('disabled'); } else { $('li:last', an[i]).removeClass('disabled'); } } } } } ); /* * TableTools Bootstrap compatibility * Required TableTools 2.1+ */ if ( $.fn.DataTable.TableTools ) { // Set the classes that TableTools uses to something suitable for Bootstrap $.extend( true, $.fn.DataTable.TableTools.classes, { "container": "DTTT btn-group", "buttons": { "normal": "btn", "disabled": "disabled" }, "collection": { "container": "DTTT_dropdown dropdown-menu", "buttons": { "normal": "", "disabled": "disabled" } }, "print": { "info": "DTTT_print_info modal" }, "select": { "row": "active" } } ); // Have the collection use a bootstrap compatible dropdown $.extend( true, $.fn.DataTable.TableTools.DEFAULTS.oTags, { "collection": { "container": "ul", "button": "li", "liner": "a" } } ); } /* $(document).ready(function() { $('#example').dataTable( { "sDom": "<'row'<'span6'l><'span6'f>r>t<'row'<'span6'i><'span6'p>>", "sPaginationType": "bootstrap", "oLanguage": { "sLengthMenu": "_MENU_ records per page" } } ); } ); */ function isStorageSupported() { try { return 'localStorage' in window && window['localStorage'] !== null; } catch (e) { return false; } } function isValidEmailAddress(emailAddress) { var pattern = new RegExp(/^((([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+(\.([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+)*)|((\x22)((((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(([\x01-\x08\x0b\x0c\x0e-\x1f\x7f]|\x21|[\x23-\x5b]|[\x5d-\x7e]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(\\([\x01-\x09\x0b\x0c\x0d-\x7f]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]))))*(((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(\x22)))@((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?$/i); return pattern.test(emailAddress); }; $(function() { $.ajaxSetup({ headers: { 'X-CSRF-Token': $('meta[name="csrf-token"]').attr('content') } }); }); function enableHoverClick($combobox, $entityId, url) { /* $combobox.mouseleave(function() { $combobox.css('text-decoration','none'); }).on('mouseenter', function(e) { setAsLink($combobox, $combobox.closest('.combobox-container').hasClass('combobox-selected')); }).on('focusout mouseleave', function(e) { setAsLink($combobox, false); }).on('click', function() { var clientId = $entityId.val(); if ($(combobox).closest('.combobox-container').hasClass('combobox-selected')) { if (parseInt(clientId) > 0) { window.open(url + '/' + clientId, '_blank'); } else { $('#myModal').modal('show'); } }; }); */ } function setAsLink($input, enable) { if (enable) { $input.css('text-decoration','underline'); $input.css('cursor','pointer'); } else { $input.css('text-decoration','none'); $input.css('cursor','text'); } } function setComboboxValue($combobox, id, name) { $combobox.find('input').val(id); $combobox.find('input.form-control').val(name); if (id && name) { $combobox.find('select').combobox('setSelected'); $combobox.find('.combobox-container').addClass('combobox-selected'); } else { $combobox.find('.combobox-container').removeClass('combobox-selected'); } } var BASE64_MARKER = ';base64,'; function convertDataURIToBinary(dataURI) { var base64Index = dataURI.indexOf(BASE64_MARKER) + BASE64_MARKER.length; var base64 = dataURI.substring(base64Index); var raw = window.atob(base64); var rawLength = raw.length; var array = new Uint8Array(new ArrayBuffer(rawLength)); for(i = 0; i < rawLength; i++) { array[i] = raw.charCodeAt(i); } return array; } ko.bindingHandlers.dropdown = { init: function (element, valueAccessor, allBindingsAccessor) { var options = allBindingsAccessor().dropdownOptions|| {}; var value = ko.utils.unwrapObservable(valueAccessor()); var id = (value && value.public_id) ? value.public_id() : (value && value.id) ? value.id() : value ? value : false; if (id) $(element).val(id); //console.log("combo-init: %s", id); $(element).combobox(options); /* ko.utils.registerEventHandler(element, "change", function () { console.log("change: %s", $(element).val()); var valueAccessor($(element).val()); //$(element).combobox('refresh'); }); */ }, update: function (element, valueAccessor) { var value = ko.utils.unwrapObservable(valueAccessor()); var id = (value && value.public_id) ? value.public_id() : (value && value.id) ? value.id() : value ? value : false; //console.log("combo-update: %s", id); if (id) { $(element).val(id); $(element).combobox('refresh'); } else { $(element).combobox('clearTarget'); $(element).combobox('clearElement'); } } }; ko.bindingHandlers.datePicker = { init: function (element, valueAccessor, allBindingsAccessor) { var value = ko.utils.unwrapObservable(valueAccessor()); if (value) $(element).datepicker('update', value); $(element).change(function() { var value = valueAccessor(); value($(element).val()); }) }, update: function (element, valueAccessor) { var value = ko.utils.unwrapObservable(valueAccessor()); if (value) $(element).datepicker('update', value); } }; function wordWrapText(value, width) { var doc = new jsPDF('p', 'pt'); doc.setFont('Helvetica',''); doc.setFontSize(10); var lines = value.split("\n"); for (var i = 0; i < lines.length; i++) { var numLines = doc.splitTextToSize(lines[i], width).length; if (numLines <= 1) continue; var j = 0; space = lines[i].length; while (j++ < lines[i].length) { if (lines[i].charAt(j) === " ") space = j; } lines[i + 1] = lines[i].substring(space + 1) + ' ' + (lines[i + 1] || ""); lines[i] = lines[i].substring(0, space); } var newValue = (lines.join("\n")).trim(); if (value == newValue) { return newValue; } else { return wordWrapText(newValue, width); } } function getClientDisplayName(client) { var contact = client.contacts[0]; if (client.name) { return client.name; } else if (contact.first_name || contact.last_name) { return contact.first_name + ' ' + contact.last_name; } else { return contact.email; } } function populateInvoiceComboboxes(clientId, invoiceId) { var clientMap = {}; var invoiceMap = {}; var invoicesForClientMap = {}; var $clientSelect = $('select#client'); for (var i=0; i 0 ? formatMoney(invoice.discount_amount, invoice.client.currency_id) : false}, {'Tax': invoice.tax_amount > 0 ? formatMoney(invoice.tax_amount, invoice.client.currency_id) : false}, {'Paid to Date': formatMoney(invoice.amount - invoice.balance, invoice.client.currency_id)} ]; return displayGrid(doc, invoice, data, 300, y, layout, true, 550, rightAlignTitleX) + 10; } function concatStrings() { var hasValue = false; var concatStr = ''; for (var i=0; i 0 && origY === layout.accountTop) { SetPdfColor('GrayText',doc); } var row = data[i]; if (!row) { continue; } if (hasheader && i === 0 && !rightAlignTitleX) { doc.setFontType('bold'); } if (typeof row === 'object') { for (var key in row) { if (row.hasOwnProperty(key)) { var value = row[key] ? row[key] + '' : false; } } if (!value) { continue; } var marginLeft; if (rightAlignX) { marginLeft = rightAlignX - (doc.getStringUnitWidth(value) * doc.internal.getFontSize()); } else { marginLeft = x + 80; } doc.text(marginLeft, y, value); if (rightAlignTitleX && i === 0) { doc.setFontType('bold'); } else { doc.setFontType('normal'); } if (rightAlignTitleX) { marginLeft = rightAlignTitleX - (doc.getStringUnitWidth(key) * doc.internal.getFontSize()); } else { marginLeft = x; } doc.text(marginLeft, y, key); } else { doc.text(x, y, row); } numLines++; y += layout.rowHeight; } return numLines * layout.rowHeight; } function displayNotesAndTerms(doc, layout, invoice, y) { doc.setFontType("normal"); var origY = y; if (invoice.public_notes) { doc.text(layout.marginLeft, y, invoice.public_notes); y += 16 + (doc.splitTextToSize(invoice.public_notes, 300).length * doc.internal.getFontSize()); } if (invoice.terms) { doc.setFontType("bold"); doc.text(layout.marginLeft, y, "Terms"); y += 16; doc.setFontType("normal"); doc.text(layout.marginLeft, y, invoice.terms); y += 16 + (doc.splitTextToSize(invoice.terms, 300).length * doc.internal.getFontSize()); } return y - origY; } function calculateAmounts(invoice) { var total = 0; var hasTaxes = false; for (var i=0; i 0) || (item.tax_rate && parseFloat(item.tax_rate) > 0)) { hasTaxes = true; } } invoice.subtotal_amount = total; if (invoice.discount > 0) { var discount = total * (invoice.discount/100); total -= discount; } var tax = 0; if (invoice.tax && parseFloat(invoice.tax.rate)) { tax = parseFloat(invoice.tax.rate); } else if (invoice.tax_rate && parseFloat(invoice.tax_rate)) { tax = parseFloat(invoice.tax_rate); } if (tax) { var tax = total * (tax/100); total = parseFloat(total) + parseFloat(tax); } invoice.balance_amount = total - (invoice.amount - invoice.balance); invoice.tax_amount = tax; invoice.discount_amount = discount; invoice.has_taxes = hasTaxes; return invoice; } function getInvoiceTaxRate(invoice) { var tax = 0; if (invoice.tax && parseFloat(invoice.tax.rate)) { tax = parseFloat(invoice.tax.rate); } else if (invoice.tax_rate && parseFloat(invoice.tax_rate)) { tax = parseFloat(invoice.tax_rate); } return tax; } function displayInvoiceHeader(doc, invoice, layout) { var costX = layout.unitCostRight - (doc.getStringUnitWidth('Unit Cost') * doc.internal.getFontSize()); var qtyX = layout.qtyRight - (doc.getStringUnitWidth('Quantity') * doc.internal.getFontSize()); var taxX = layout.taxRight - (doc.getStringUnitWidth('Tax') * doc.internal.getFontSize()); var totalX = layout.lineTotalRight - (doc.getStringUnitWidth('Line Total') * doc.internal.getFontSize()); doc.text(layout.marginLeft, layout.tableTop, 'Item'); doc.text(layout.descriptionLeft, layout.tableTop, 'Description'); doc.text(costX, layout.tableTop, 'Unit Cost'); doc.text(qtyX, layout.tableTop, 'Quantity'); doc.text(totalX, layout.tableTop, 'Line Total'); if (invoice.has_taxes) { doc.text(taxX, layout.tableTop, 'Tax'); } } function displayInvoiceItems(doc, invoice, layout) { doc.setFontType("normal"); var line = 1; var total = 0; var shownItem = false; var currencyId = invoice && invoice.client ? invoice.client.currency_id : 1; var tableTop = layout.tableTop; doc.setFontSize(8); for (var i=0; i 770) { line = 0; tableTop = layout.accountTop + layout.tablePadding; y = tableTop; top = y - layout.tablePadding; newTop = top + (numLines * layout.tableRowHeight); doc.addPage(); } var left = layout.marginLeft - layout.tablePadding; var width = layout.marginRight + layout.tablePadding; var cost = formatMoney(item.cost, currencyId, true); var qty = NINJA.parseFloat(item.qty) ? NINJA.parseFloat(item.qty) + '' : ''; var notes = item.notes; var productKey = item.product_key; var tax = 0; if (item.tax && parseFloat(item.tax.rate)) { tax = parseFloat(item.tax.rate); } else if (item.tax_rate && parseFloat(item.tax_rate)) { tax = parseFloat(item.tax_rate); } // show at most one blank line if (shownItem && (!cost || cost == '0.00') && !qty && !notes && !productKey) { continue; } shownItem = true; // process date variables notes = processVariables(notes); productKey = processVariables(productKey); var lineTotal = NINJA.parseFloat(item.cost) * NINJA.parseFloat(item.qty); if (tax) { lineTotal += lineTotal * tax / 100; } if (lineTotal) { total += lineTotal; } lineTotal = formatMoney(lineTotal, currencyId, true); var costX = layout.unitCostRight - (doc.getStringUnitWidth(cost) * doc.internal.getFontSize()); var qtyX = layout.qtyRight - (doc.getStringUnitWidth(qty) * doc.internal.getFontSize()); var taxX = layout.taxRight - (doc.getStringUnitWidth(tax+'%') * doc.internal.getFontSize()); var totalX = layout.lineTotalRight - (doc.getStringUnitWidth(lineTotal) * doc.internal.getFontSize()); if (i==0) y -= 4; line += numLines; if (invoice.invoice_design_id == 1) { if (i%2 == 0) { doc.setDrawColor(255,255,255); doc.setFillColor(245,245,245); doc.rect(left, top, width-left, newTop-top, 'FD'); doc.setLineWidth(0.3); doc.setDrawColor(200,200,200); doc.line(left, top, width, top); doc.line(left, newTop, width, newTop); } } else if (invoice.invoice_design_id == 2) { if (i%2 == 0) { left = 0; width = 1000; doc.setDrawColor(255,255,255); doc.setFillColor(235,235,235); doc.rect(left, top, width-left, newTop-top, 'FD'); } } else { doc.setLineWidth(0.3); doc.setDrawColor(150,150,150); doc.line(left, newTop, width, newTop); } y += 4; if (invoice.invoice_design_id == 1) { SetPdfColor('LightBlue', doc); } else if (invoice.invoice_design_id == 2) { SetPdfColor('SomeGreen', doc); } else if (invoice.invoice_design_id == 3) { doc.setFontType('bold'); } else { SetPdfColor('Black', doc); } doc.text(layout.marginLeft, y, productKey); SetPdfColor('Black', doc); doc.setFontType('normal'); doc.text(layout.descriptionLeft, y, notes); doc.text(costX, y, cost); doc.text(qtyX, y, qty); doc.text(totalX, y, lineTotal); if (tax) { doc.text(taxX, y, tax+'%'); } } y = tableTop + (line * layout.tableRowHeight) + (3 * layout.tablePadding); var cutoff = 700; if (invoice.terms) { cutoff -= 50; } if (invoice.public_notes) { cutoff -= 50; } if (y > cutoff) { doc.addPage(); return layout.marginLeft; } return y; }