// 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 generatePDF(invoice, checkMath) { console.log ('DESIGN:'+invoice.invoice_design_id); 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) { if (!width) width = 200; 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) { 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); } total = formatMoney(total - (invoice.amount - invoice.balance), currencyId); var balance = formatMoney(total, currencyId); var doc = new jsPDF('p', 'pt'); //set default style for report doc.setFont('Helvetica',''); doc.setFontSize(7); //---------------------------------------------------------------------------------------------------- //Print header on document //for now we will put static header //but later this could be changed to more flexible solution if (invoice.image) { var left = headerRight - invoice.imageWidth; doc.addImage(invoice.image, 'JPEG', left, 30, invoice.imageWidth, invoice.imageHeight); } if (invoice.imageLogo1) { pageHeight=820; var left = headerRight ; y=pageHeight-invoice.imageLogoHeight1; var left = headerRight - invoice.imageLogoWidth1; doc.addImage(invoice.imageLogo1, 'JPEG', left, y, invoice.imageLogoWidth1, invoice.imageLogoHeight1); } var invoiceNumberX = headerRight - (doc.getStringUnitWidth(invoiceNumber, false) * doc.internal.getFontSize()); var invoiceDateX = headerRight - (doc.getStringUnitWidth(invoiceDate) * doc.internal.getFontSize()); var dueDateX = headerRight - (doc.getStringUnitWidth(dueDate) * doc.internal.getFontSize()); var poNumberX = headerRight - (doc.getStringUnitWidth(invoice.po_number) * doc.internal.getFontSize()); // doc.setFontType("normal"); var y = accountTop; var left = marginLeft; doc.setFontSize(7); SetPdfColor('LightBlue',doc); if (account.name) { y += rowHeight; doc.text(left, y, account.name); } SetPdfColor('GrayText',doc); doc.setFontSize(6); //TODO:NOT AVAILEABLE FROM DATAMOEL //account.email='email N/A'; if (account.email) { y += rowHeight; doc.text(left, y, account.email); } else { //console.log('account.email NOT DEFINED !'); } //TODO:NOT AVAILEABLE FROM DATAMOEL //account.phone='phone N/A'; if (account.phone) { y += rowHeight; doc.text(left, y, account.phone); } else { //console.log('account.phone NOT DEFINED !'); } var HeaderMarginThirdColumn=70;//should be dynamic and dependent on 1st image and 2nd column width var y = accountTop; var left = marginLeft+HeaderMarginThirdColumn; if (account.address1) { y += rowHeight; doc.text(left, y, account.address1); } if (account.address2) { y += rowHeight; doc.text(left, y, account.address2); } if (account.city || account.state || account.postal_code) { y += rowHeight; doc.text(left, y, account.city + ', ' + account.state + ' ' + account.postal_code); } if (account.country) { y += rowHeight; doc.text(left, y, account.country.name); } //-----------------------------Publish Client Details block-------------------------------------------- var y = accountTop; var left = marginLeft; var headerY = headerTop; SetPdfColor('LightBlue',doc); doc.setFontSize(8); //doc.setFontType("bold"); doc.text(50, headerTop, 'Invoice'); SetPdfColor('GrayLogo',doc); //set black color y=130; doc.line(30, y, 560, y); // horizontal line var line1=headerTop+16; var line2=headerTop+16*2; var line21=headerTop+16*1.6; var line22=headerTop+16*2.2; var line3=headerTop+16*3; var line31=headerTop+16*3.6; var marginLeft1=50; var marginLeft2=120; var marginLeft3=180; SetPdfColor('Black',doc); //set black color doc.setFontSize(6); doc.text(marginLeft1, line1, 'Invoice Number'); doc.text(marginLeft1, line2, 'Invoice date'); doc.text(marginLeft1, line3, 'Amount Due'); // invoiceNumber='12345' // invoiceDate='12345' //invoiceAmount='12345' doc.setFontType("bold"); doc.text(marginLeft2, line1, invoiceNumber); doc.setFontType("normal"); doc.text(marginLeft2, line2, invoiceDate); SetPdfColor('LightBlue',doc); //set black color doc.text(marginLeft2, line3, balance); ClientCompanyName=client.name; ClientCompanyEmail='';//client.email;//'22222222'; ClientCompanyPhone=client.work_phone; ClientCompanyAddress1=client.address1; ClientCompanyAddress2=client.address2+' '+client.postal_code; SetPdfColor('Black',doc); //set black color doc.setFontType("bold"); doc.text(marginLeft3, line1, ClientCompanyName); doc.setFontType("normal"); if(client) { ClientCompanyName=getClientDisplayName(client); ClientCompanyPhone=client.work_phone; ClientCompanyEmail=client.contacts[0].email; } doc.text(marginLeft3, line21, ClientCompanyAddress1); doc.text(marginLeft3, line22, ClientCompanyAddress2); doc.text(marginLeft3, line3, ClientCompanyEmail); doc.text(marginLeft3, line31, ClientCompanyPhone); SetPdfColor('GrayLogo',doc); //set black color y=195; doc.line(30, y, 560, y); // horizontal line //--------------------------------Publishing Table-------------------------------------------------- GlobalY=y+30; SetPdfColor('Black',doc); doc.setFontSize(7); var hasTaxes = false; for (var i=0; i 0) || (item.tax_rate && parseFloat(item.tax_rate) > 0)) { hasTaxes = true; break; } } if (hasTaxes) { descriptionLeft -= 20; unitCostRight -= 40; qtyRight -= 40; } var costX = unitCostRight - (doc.getStringUnitWidth('Unit Cost') * doc.internal.getFontSize()); var qtyX = qtyRight - (doc.getStringUnitWidth('Quantity') * doc.internal.getFontSize()); var taxX = taxRight - (doc.getStringUnitWidth('Tax') * doc.internal.getFontSize()); var totalX = lineTotalRight - (doc.getStringUnitWidth('Line Total') * doc.internal.getFontSize()); tableTop=GlobalY;//redefine this to dynamic value doc.setFontSize(9); doc.text(tableLeft, tableTop, 'Item'); doc.text(descriptionLeft, tableTop, 'Description'); doc.text(costX, tableTop, 'Unit Cost'); doc.text(qtyX, tableTop, 'Quantity'); doc.text(totalX, tableTop, 'Line Total'); if (hasTaxes) { doc.text(taxX, tableTop, 'Tax'); } doc.setFontSize(7); /* line items */ //doc.setFontType("normal"); var line = 1; var total = 0; var shownItem = false; GlobalY=GlobalY+14; //padding from top var FontSize=7; doc.setFontSize(FontSize); var MaxLinesPerPage=40; for (var i=0; i MaxLinesPerPage) { line = 0; tableTop = 40; GlobalY=tableTop; var MaxLinesPerPage=70; doc.addPage(); if (invoice.imageLogo1) { pageHeight=820; var left = headerRight ; y=pageHeight-invoice.imageLogoHeight1; var left = headerRight - invoice.imageLogoWidth1; doc.addImage(invoice.imageLogo1, 'JPEG', left, y, invoice.imageLogoWidth1, invoice.imageLogoHeight1); } } } //-------------------------------Publishing Document balance------------------------------------------ // var dueDateX = headerRight - (doc.getStringUnitWidth(dueDate) * doc.internal.getFontSize()); x += 16+50; doc.setFontType("bold"); MsgRightAlign=400; Msg='Total'; var TmpMsgX = MsgRightAlign-(doc.getStringUnitWidth(Msg) * doc.internal.getFontSize()); doc.text(TmpMsgX, x, Msg); doc.setFontType("normal"); AmountText = formatMoney(total , currencyId); headerLeft=headerRight+400; var AmountX = headerLeft - (doc.getStringUnitWidth(AmountText) * doc.internal.getFontSize()); doc.text(AmountX, x, AmountText); x += doc.internal.getFontSize()*2; //doc.text(footerLeft, x, ''); Msg='Amount Payed'; var TmpMsgX = MsgRightAlign-(doc.getStringUnitWidth(Msg) * doc.internal.getFontSize()); doc.text(TmpMsgX, x, Msg); AmountText = formatMoney(paid_to_date , currencyId); headerLeft=headerRight+400; var AmountX = headerLeft - (doc.getStringUnitWidth(AmountText) * doc.internal.getFontSize()); doc.text(AmountX, x, AmountText); doc.setFontSize(10); x += doc.internal.getFontSize()*4; //doc.text(footerLeft, x, ''); Msg='Amount Due'; var TmpMsgX = MsgRightAlign-(doc.getStringUnitWidth(Msg) * doc.internal.getFontSize()); doc.text(TmpMsgX, x, Msg); SetPdfColor('LightBlue',doc); AmountText = formatMoney(balance , currencyId); headerLeft=headerRight+400; var AmountX = headerLeft - (doc.getStringUnitWidth(AmountText) * doc.internal.getFontSize()); doc.text(AmountX, x, AmountText); return doc; } function SetPdfColor(color,doc) { if (color=='LightBlue') { return doc.setTextColor(41,156, 194); } if (color=='Black') { return doc.setTextColor(0,0,0);//select color black } if (color=='GrayLogo') { return doc.setTextColor(207,209, 210);//select color Custom Report GRAY } if (color=='GrayText') { return doc.setTextColor(161,160,160);//select color Custom Report GRAY Colour } alert('color is not defined'); return false; }