123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335 |
- /*
- * printThis v1.14.1
- * @desc Printing plug-in for jQuery
- * @author Jason Day
- *
- * Resources (based on):
- * - jPrintArea: http://plugins.jquery.com/project/jPrintArea
- * - jqPrint: https://github.com/permanenttourist/jquery.jqprint
- * - Ben Nadal: http://www.bennadel.com/blog/1591-Ask-Ben-Print-Part-Of-A-Web-Page-With-jQuery.htm
- *
- * Licensed under the MIT licence:
- * http://www.opensource.org/licenses/mit-license.php
- *
- * (c) Jason Day 2015-2018
- *
- * Usage:
- *
- * $("#mySelector").printThis({
- * debug: false, // show the iframe for debugging
- * importCSS: true, // import parent page css
- * importStyle: false, // import style tags
- * printContainer: true, // grab outer container as well as the contents of the selector
- * loadCSS: "path/to/my.css", // path to additional css file - use an array [] for multiple
- * pageTitle: "", // add title to print page
- * removeInline: false, // remove all inline styles from print elements
- * removeInlineSelector: "body *", // custom selectors to filter inline styles. removeInline must be true
- * printDelay: 333, // variable print delay
- * header: null, // prefix to html
- * footer: null, // postfix to html
- * base: false, // preserve the BASE tag, or accept a string for the URL
- * formValues: true, // preserve input/form values
- * canvas: false, // copy canvas elements
- * doctypeString: '...', // enter a different doctype for older markup
- * removeScripts: false, // remove script tags from print content
- * copyTagClasses: false // copy classes from the html & body tag
- * beforePrintEvent: null, // callback function for printEvent in iframe
- * beforePrint: null, // function called before iframe is filled
- * afterPrint: null // function called before iframe is removed
- * });
- *
- * Notes:
- * - the loadCSS will load additional CSS (with or without @media print) into the iframe, adjusting layout
- */
- ;
- (function($) {
- function appendContent($el, content) {
- if (!content) return;
- // Simple test for a jQuery element
- $el.append(content.jquery ? content.clone() : content);
- }
- function appendBody($body, $element, opt) {
- // Clone for safety and convenience
- // Calls clone(withDataAndEvents = true) to copy form values.
- var $content = $element.clone(opt.formValues);
- if (opt.formValues) {
- // Copy original select and textarea values to their cloned counterpart
- // Makes up for inability to clone select and textarea values with clone(true)
- copyValues($element, $content, 'select, textarea');
- }
- if (opt.removeScripts) {
- $content.find('script').remove();
- }
- if (opt.printContainer) {
- // grab $.selector as container
- $content.appendTo($body);
- } else {
- // otherwise just print interior elements of container
- $content.each(function() {
- $(this).children().appendTo($body)
- });
- }
- }
- // Copies values from origin to clone for passed in elementSelector
- function copyValues(origin, clone, elementSelector) {
- var $originalElements = origin.find(elementSelector);
- clone.find(elementSelector).each(function(index, item) {
- $(item).val($originalElements.eq(index).val());
- });
- }
- var opt;
- $.fn.printThis = function(options) {
- opt = $.extend({}, $.fn.printThis.defaults, options);
- var $element = this instanceof jQuery ? this : $(this);
- var strFrameName = "printThis-" + (new Date()).getTime();
- if (window.location.hostname !== document.domain && navigator.userAgent.match(/msie/i)) {
- // Ugly IE hacks due to IE not inheriting document.domain from parent
- // checks if document.domain is set by comparing the host name against document.domain
- var iframeSrc = "javascript:document.write(\"<head><script>document.domain=\\\"" + document.domain + "\\\";</s" + "cript></head><body></body>\")";
- var printI = document.createElement('iframe');
- printI.name = "printIframe";
- printI.id = strFrameName;
- printI.className = "MSIE";
- document.body.appendChild(printI);
- printI.src = iframeSrc;
- } else {
- // other browsers inherit document.domain, and IE works if document.domain is not explicitly set
- var $frame = $("<iframe id='" + strFrameName + "' name='printIframe' />");
- $frame.appendTo("body");
- }
- var $iframe = $("#" + strFrameName);
- // show frame if in debug mode
- if (!opt.debug) $iframe.css({
- position: "absolute",
- width: "0px",
- height: "0px",
- left: "-600px",
- top: "-600px"
- });
- // before print callback
- if (typeof opt.beforePrint === "function") {
- opt.beforePrint();
- }
- // $iframe.ready() and $iframe.load were inconsistent between browsers
- setTimeout(function() {
- // Add doctype to fix the style difference between printing and render
- function setDocType($iframe, doctype){
- var win, doc;
- win = $iframe.get(0);
- win = win.contentWindow || win.contentDocument || win;
- doc = win.document || win.contentDocument || win;
- doc.open();
- doc.write(doctype);
- doc.close();
- }
- if (opt.doctypeString){
- setDocType($iframe, opt.doctypeString);
- }
- var $doc = $iframe.contents(),
- $head = $doc.find("head"),
- $body = $doc.find("body"),
- $base = $('base'),
- baseURL;
- // add base tag to ensure elements use the parent domain
- if (opt.base === true && $base.length > 0) {
- // take the base tag from the original page
- baseURL = $base.attr('href');
- } else if (typeof opt.base === 'string') {
- // An exact base string is provided
- baseURL = opt.base;
- } else {
- // Use the page URL as the base
- baseURL = document.location.protocol + '//' + document.location.host;
- }
- $head.append('<base href="' + baseURL + '">');
- // import page stylesheets
- if (opt.importCSS) $("link[rel=stylesheet]").each(function() {
- var href = $(this).attr("href");
- if (href) {
- var media = $(this).attr("media") || "all";
- $head.append("<link type='text/css' rel='stylesheet' href='" + href + "' media='" + media + "'>");
- }
- });
- // import style tags
- if (opt.importStyle) $("style").each(function() {
- $head.append(this.outerHTML);
- });
- // add title of the page
- if (opt.pageTitle) $head.append("<title>" + opt.pageTitle + "</title>");
- // import additional stylesheet(s)
- if (opt.loadCSS) {
- if ($.isArray(opt.loadCSS)) {
- jQuery.each(opt.loadCSS, function(index, value) {
- $head.append("<link type='text/css' rel='stylesheet' href='" + this + "'>");
- });
- } else {
- $head.append("<link type='text/css' rel='stylesheet' href='" + opt.loadCSS + "'>");
- }
- }
- var pageHtml = $('html')[0];
- // CSS VAR in html tag when dynamic apply e.g. document.documentElement.style.setProperty("--foo", bar);
- $doc.find('html').prop('style', pageHtml.style.cssText);
- // copy 'root' tag classes
- var tag = opt.copyTagClasses;
- if (tag) {
- tag = tag === true ? 'bh' : tag;
- if (tag.indexOf('b') !== -1) {
- $body.addClass($('body')[0].className);
- }
- if (tag.indexOf('h') !== -1) {
- $doc.find('html').addClass(pageHtml.className);
- }
- }
- // print header
- appendContent($body, opt.header);
- if (opt.canvas) {
- // add canvas data-ids for easy access after cloning.
- var canvasId = 0;
- // .addBack('canvas') adds the top-level element if it is a canvas.
- $element.find('canvas').addBack('canvas').each(function(){
- $(this).attr('data-printthis', canvasId++);
- });
- }
- appendBody($body, $element, opt);
- if (opt.canvas) {
- // Re-draw new canvases by referencing the originals
- $body.find('canvas').each(function(){
- var cid = $(this).data('printthis'),
- $src = $('[data-printthis="' + cid + '"]');
- this.getContext('2d').drawImage($src[0], 0, 0);
- // Remove the markup from the original
- if ($.isFunction($.fn.removeAttr)) {
- $src.removeAttr('data-printthis');
- } else {
- $.each($src, function(i, el) {
- el.removeAttribute('data-printthis')
- });
- }
- });
- }
- // remove inline styles
- if (opt.removeInline) {
- // Ensure there is a selector, even if it's been mistakenly removed
- var selector = opt.removeInlineSelector || '*';
- // $.removeAttr available jQuery 1.7+
- if ($.isFunction($.removeAttr)) {
- $body.find(selector).removeAttr("style");
- } else {
- $body.find(selector).attr("style", "");
- }
- }
- // print "footer"
- appendContent($body, opt.footer);
- // attach event handler function to beforePrint event
- function attachOnBeforePrintEvent($iframe, beforePrintHandler) {
- var win = $iframe.get(0);
- win = win.contentWindow || win.contentDocument || win;
- if (typeof beforePrintHandler === "function") {
- if ('matchMedia' in win) {
- win.matchMedia('print').addListener(function(mql) {
- if(mql.matches) beforePrintHandler();
- });
- } else {
- win.onbeforeprint = beforePrintHandler;
- }
- }
- }
- attachOnBeforePrintEvent($iframe, opt.beforePrint);
- setTimeout(function() {
- if ($iframe.hasClass("MSIE")) {
- // check if the iframe was created with the ugly hack
- // and perform another ugly hack out of neccessity
- window.frames["printIframe"].focus();
- $head.append("<script> window.print(); </s" + "cript>");
- } else {
- // proper method
- if (document.queryCommandSupported("print")) {
- $iframe[0].contentWindow.document.execCommand("print", false, null);
- } else {
- $iframe[0].contentWindow.focus();
- $iframe[0].contentWindow.print();
- }
- }
- // remove iframe after print
- if (!opt.debug) {
- setTimeout(function() {
- $iframe.remove();
- }, 1000);
- }
- // after print callback
- if (typeof opt.afterPrint === "function") {
- opt.afterPrint();
- }
- }, opt.printDelay);
- }, 333);
- };
- // defaults
- $.fn.printThis.defaults = {
- debug: false, // show the iframe for debugging
- importCSS: true, // import parent page css
- importStyle: false, // import style tags
- printContainer: true, // print outer container/$.selector
- loadCSS: "", // path to additional css file - use an array [] for multiple
- pageTitle: "", // add title to print page
- removeInline: false, // remove inline styles from print elements
- removeInlineSelector: "*", // custom selectors to filter inline styles. removeInline must be true
- printDelay: 333, // variable print delay
- header: null, // prefix to html
- footer: null, // postfix to html
- base: false, // preserve the BASE tag or accept a string for the URL
- formValues: true, // preserve input/form values
- canvas: false, // copy canvas content
- doctypeString: '<!DOCTYPE html>', // enter a different doctype for older markup
- removeScripts: false, // remove script tags from print content
- copyTagClasses: false, // copy classes from the html & body tag
- beforePrintEvent: null, // callback function for printEvent in iframe
- beforePrint: null, // function called before iframe is filled
- afterPrint: null // function called before iframe is removed
- };
- })(jQuery);
|