path); db_query("UPDATE {print_page_counter} SET totalcount = totalcount + 1, timestamp = %d WHERE path = '%s'", time(), $nodepath); // If we affected 0 rows, this is the first time viewing the node. if (!db_affected_rows()) { // We must create a new row to store counters for the new node. db_query("INSERT INTO {print_page_counter} (path, totalcount, timestamp) VALUES ('%s', 1, %d)", $nodepath, time()); } } } /** * Select the print generator function based on the page type * * Depending on the type of node, this functions chooses the appropriate * generator function. * * @param $path * path of the original page * @param $cid * comment ID of the individual comment to be rendered * @param $format * format of the page being generated * @param $teaser * if set to TRUE, outputs only the node's teaser * @param $message * optional sender's message (used by the send e-mail module) * @return * array with the fields to be used in the template * @see _print_generate_node() * @see _print_generate_path() * @see _print_generate_book() */ function print_controller($path, $cid = NULL, $format = PRINT_HTML_FORMAT, $teaser = FALSE, $message = NULL) { if (empty($path)) { // If no path was provided, let's try to generate a page for the referer global $base_url; $ref = referer_uri(); $path = preg_replace("!^$base_url/!", '', $ref); if ($path === $ref) { $path = ''; } } if (!is_numeric($path)) { // Indirect call with print/alias // If there is a path alias with these arguments, generate a printer-friendly version for it $path = drupal_get_normal_path($path); $ret = preg_match('!^node/(.*)!i', $path, $matches); if ($ret == 1) { $path = $matches[1]; } } $parts = explode('/', $path); if (is_numeric($parts[0])) { $print = _print_generate_node($path, $cid, $format, $teaser, $message); } else { $ret = preg_match('!^book/export/html/(.*)!i', $path, $matches); if ($ret == 1) { // This is a book PF page link, handle trough the book handling functions $print = _print_generate_book($matches[1], $format, $teaser, $message); } else { // If no content node was found, handle the page printing with the 'printable' engine $print = _print_generate_path($path, $format, $teaser, $message); } } return $print; } /** * Generates a robots meta tag to tell them what they may index * * @return * string with the meta robots tag */ function _print_robots_meta_generator() { $print_robots_noindex = variable_get('print_robots_noindex', PRINT_ROBOTS_NOINDEX_DEFAULT); $print_robots_nofollow = variable_get('print_robots_nofollow', PRINT_ROBOTS_NOFOLLOW_DEFAULT); $print_robots_noarchive = variable_get('print_robots_noarchive', PRINT_ROBOTS_NOARCHIVE_DEFAULT); $robots_meta = array(); if (!empty($print_robots_noindex)) { $robots_meta[] = 'noindex'; } if (!empty($print_robots_nofollow)) { $robots_meta[] = 'nofollow'; } if (!empty($print_robots_noarchive)) { $robots_meta[] = 'noarchive'; } if (count($robots_meta) > 0) { $robots_meta = implode(', ', $robots_meta); $robots_meta = "\n"; } else { $robots_meta = ''; } return $robots_meta; } /** * Post-processor that fills the array for the template with common details * * @param $node * generated node with a printer-friendly node body * @param $message * optional sender's message (used by the send e-mail module) * @param $cid * id of current comment being generated (NULL when not generating * an individual comment) * @return * array with the fields to be used in the template */ function _print_var_generator($node, $message = NULL, $cid = NULL) { global $base_url, $language, $_print_urls; $path = empty($node->nid) ? $node->path : "node/$node->nid"; // print module settings $print_css = variable_get('print_css', PRINT_CSS_DEFAULT); $print_logo_options = variable_get('print_logo_options', PRINT_LOGO_OPTIONS_DEFAULT); $print_logo_url = variable_get('print_logo_url', PRINT_LOGO_URL_DEFAULT); $print_html_sendtoprinter = variable_get('print_html_sendtoprinter', PRINT_HTML_SENDTOPRINTER_DEFAULT); $print_sourceurl_enabled = variable_get('print_sourceurl_enabled', PRINT_SOURCEURL_ENABLED_DEFAULT); $print_sourceurl_forcenode = variable_get('print_sourceurl_forcenode', PRINT_SOURCEURL_FORCENODE_DEFAULT); $print_sourceurl_date = variable_get('print_sourceurl_date', PRINT_SOURCEURL_DATE_DEFAULT); $print_footer_options = variable_get('print_footer_options', PRINT_FOOTER_OPTIONS_DEFAULT); $print_footer_user = variable_get('print_footer_user', PRINT_FOOTER_USER_DEFAULT); $print['language'] = $language->language; $print['title'] = $node->title; $print['head'] = drupal_get_html_head(); $print['scripts'] = drupal_get_js(); $print['robots_meta'] = _print_robots_meta_generator(); $print['url'] = url($path, array('absolute' => TRUE)); $print['base_href'] = "\n"; $print['favicon'] = theme_get_setting('toggle_favicon') ? "\n" : ''; if (!empty($print_css)) { $replace_pairs = array('%b' => base_path(), '%t' => path_to_theme()); $user_css = strip_tags(strtr($print_css, $replace_pairs)); } else { drupal_add_css(drupal_get_path('module', 'print') .'/css/print.css'); } $drupal_css = drupal_add_css(); foreach ($drupal_css as $key => $types) { // Unset the theme's CSS $drupal_css[$key]['theme'] = array(); } // If we are sending a message via e-mail, the CSS must be embedded if (!empty($message)) { $style = ''; $css_files = array(); foreach ($drupal_css as $types) { foreach ($types as $values) { $css_files = array_merge($css_files, array_keys($values)); } } if (!empty($print_css)) { // Convert to a local path, by removing the base_path $pattern = '!^'. base_path() .'!'; $css_files[] = preg_replace($pattern, '', $user_css); } foreach ($css_files as $filename) { $res = file_exists($filename) ? file_get_contents($filename, TRUE) : FALSE; if ($res != FALSE) { $style .= $res; } } $print['css'] = "\n"; } else { $print['css'] = drupal_get_css($drupal_css); if (!empty($print_css)) { $print['css'] .= "\n"; } } $print['sendtoprinter'] = $print_html_sendtoprinter ? ' onload="window.print();"' : ''; switch ($print_logo_options) { case 0: // none $logo_url = 0; break; case 1: // theme's $logo_url = theme_get_setting('logo'); break; case 2: // user-specifed $logo_url = strip_tags($print_logo_url); break; } $print['logo'] = $logo_url ? "\n" : ''; switch ($print_footer_options) { case 0: // none $footer = ''; break; case 1: // theme's $footer = filter_xss_admin(variable_get('site_footer', FALSE)) ."\n". theme('blocks', 'footer'); $logo_url = theme_get_setting('logo'); break; case 2: // user-specifed $footer = $print_footer_user; break; } $print['footer_message'] = $footer; $published_site = variable_get('site_name', 0); if ($published_site) { $print_text_published = variable_get('print_text_published', t('Published on %site_name')); $published = t($print_text_published, array('%site_name' => $published_site)); $print['site_name'] = $published .' ('. l($base_url, $base_url) .')'; } else { $print['site_name'] = ''; } if ($print_sourceurl_enabled == 1) { /* Grab and format the src URL */ if (empty($print_sourceurl_forcenode)) { $url = $print['url']; } else { $url = $base_url .'/'. (((bool)variable_get('clean_url', '0')) ? '' : '?q=') . $path; } if (is_int($cid)) { $url .= '#comment-$cid'; } $retrieved_date = format_date(time(), 'small'); $print_text_retrieved = variable_get('print_text_retrieved', t('retrieved on %date')); $retrieved = t($print_text_retrieved, array('%date' => $retrieved_date)); $print['printdate'] = $print_sourceurl_date ? " ($retrieved)" : ''; $source_url = variable_get('print_text_source_url', t('Source URL')); $print['source_url'] = ''. $source_url . $print['printdate'] .': '. l($url, $url); } else { $print['source_url'] = ''; } if (isset($node->type)) { $node_type = $node->type; if (theme_get_setting("toggle_node_info_$node_type")) { $print_text_by = variable_get('print_text_by', t('By %author')); $by_author = ($node->name ? $node->name : variable_get('anonymous', t('Anonymous'))); $print['submitted'] = t($print_text_by, array('%author' => $by_author)); $print_text_created = variable_get('print_text_created', t('Created %date')); $created_datetime = format_date($node->created, 'small'); $print['created'] = t($print_text_created, array('%date' => $created_datetime)); } else { $print['submitted'] = ''; $print['created'] = ''; } $print['type'] = $node->type; } else { $print['submitted'] = ''; $print['created'] = ''; $print['type'] = ''; } menu_set_active_item($path); $breadcrumb = drupal_get_breadcrumb(); if (!empty($breadcrumb)) { $breadcrumb[] = menu_get_active_title(); $print['breadcrumb'] = implode(' > ', $breadcrumb); } else { $print['breadcrumb'] = ''; } // Display the collected links at the bottom of the page. Code once taken from Kjartan Mannes' project.module $print['pfp_links'] = ''; if (!empty($_print_urls)) { $urls = _print_friendly_urls(); $max = count($urls); $pfp_links = ''; if ($max) { for ($i = 0; $i < $max; $i++) { $pfp_links .= '['. ($i + 1) .'] '. $urls[$i] ."
\n"; } $links = variable_get('print_text_links', t('Links')); $print['pfp_links'] = "

$links:
$pfp_links

"; } } if (module_exists('taxonomy')) { $terms = taxonomy_link('taxonomy terms', $node); $print['taxonomy'] = theme('links', $terms); } $print['content'] = $node->body; $print['node'] = $node; $print['message'] = $message; return $print; } /** * Callback function for the preg_replace_callback for URL-capable patterns * * Manipulate URLs to make them absolute in the URLs list, and to add a * [n] footnote marker. * * @param $matches * array with the matched tag patterns, usually +text+ * @return * tag with re-written URL and when appropriate the [n] index to the * URL list */ function _print_rewrite_urls($matches) { global $base_url, $base_root, $_print_urls; // first, split the html into the different tag attributes $pattern = '!\s*(\w+\s*=\s*"(?:\\\"|[^"])*")\s*|\s*(\w+\s*=\s*\'(?:\\\\\'|[^\'])*\')\s*|\s*(\w+\s*=\s*\w+)\s*|\s+!'; $attribs = preg_split($pattern, $matches[1], -1, PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE); foreach ($attribs as $key => $value) { $attribs[$key] = preg_replace('!(\w)\s*=\s*(.*)!', '$1=$2', $value); } $size = count($attribs); for ($i=1; $i < $size; $i++) { // If the attribute is href or src, we may need to rewrite the URL in the value if (preg_match('!^(?:href|src)\s*?=(.*)!i', $attribs[$i], $urls) > 0) { $url = trim($urls[1], " \t\n\r\0\x0B\"'"); if (strpos($url, '://') || preg_match('!^mailto:.*?@.*?\..*?$!iu', $url)) { // URL is absolute, do nothing $newurl = urldecode($url); } else { if ($url[0] == '#') { // URL is an anchor tag if (!empty($_print_urls)) { $path = explode('/', $_GET['q']); unset($path[0]); $path = implode('/', $path); if (is_numeric($path)) { $path = "node/$path"; } // Printer-friendly URLs is on, so we need to make it absolute $newurl = url($path, array('fragment' => substr(urldecode($url), 1), 'absolute' => TRUE)); } // Because base href is the original page, change the link to // still be usable inside the print page $matches[1] = str_replace($url, $_GET['q'] . $url, $matches[1]); } else { // URL is relative, convert it into absolute URL if ($url[0] == '/') { // If it starts with '/' just append it to the server name $newurl = $base_root .'/'. trim(urldecode($url), '/'); } elseif (preg_match('!^(?:index.php)?\?q=!i', $url)) { // If it starts with ?q=, just prepend with the base URL $newurl = $base_url .'/'. trim(urldecode($url), '/'); } else { $newurl = url(trim(urldecode($url), '/'), array('absolute' => TRUE)); } $matches[1] = str_replace($url, $newurl, $matches[1]); } } } } $ret = '<'. $matches[1] .'>'; if (count($matches) == 4) { $ret .= $matches[2] . $matches[3]; if ((!empty($_print_urls)) && (isset($newurl))) { $ret .= ' ['. _print_friendly_urls(trim(stripslashes($newurl))) .']'; } } return $ret; } /** * Auxiliary function to store the Printer-friendly URLs list as static. * * @param $url * absolute URL to be inserted in the list * @return * list of URLs previously stored if $url is 0, or the current count * otherwise. */ function _print_friendly_urls($url = 0) { static $urls = array(); if ($url) { $url_idx = array_search($url, $urls); if ($url_idx !== FALSE) { return ($url_idx + 1); } else { $urls[] = $url; return count($urls); } } $ret = $urls; $urls = array(); return $ret; } /** * Choose most appropriate template * * Auxiliary function to resolve the most appropriate template trying to find * a content specific template in the theme or module dir before falling back * on a generic template also in those dirs. * * @param format * format of the PF page being rendered (html, pdf, etc.) * @param $type * name of the node type being rendered in a PF page * @return * filename of the most suitable template */ function _print_get_template($format = NULL, $type = NULL) { $filenames = array(); // First try to find a template defined both for the format and then the type if (!empty($format) && !empty($type)) { $filenames[] = "print_$format.node-$type.tpl.php"; } // Then only for the format if (!empty($format)) { $filenames[] = "print_$format.tpl.php"; } // If the node type is known, then try to find that type's template file if (!empty($type)) { $filenames[] = "print.node-$type.tpl.php"; } // Finally search for a generic template file $filenames[] = 'print.tpl.php'; foreach ($filenames as $value) { // First in the theme directory $file = drupal_get_path('theme', $GLOBALS['theme_key']) .'/'. $value; if (file_exists($file)) { return $file; } // Then in the module directory $file = drupal_get_path('module', 'print') .'/'. $value; if (file_exists($file)) { return $file; } } } /** * Check URL list settings for this node * * @param node * node object * @param $format * format of the page being generated * @return * TRUE if URL list should be displayed, FALSE otherwise */ function _print_url_list_enabled($node, $format = PRINT_HTML_FORMAT) { switch ($format) { case PRINT_HTML_FORMAT: $node_urllist = isset($node->print_display_urllist) ? $node->print_display_urllist : TRUE; $fmt = ''; break; case PRINT_MAIL_FORMAT: $node_urllist = isset($node->print_mail_display_urllist) ? $node->print_mail_display_urllist : TRUE; $fmt = $format .'_'; break; case PRINT_PDF_FORMAT: $node_urllist = isset($node->print_pdf_display_urllist) ? $node->print_pdf_display_urllist : TRUE; $fmt = $format .'_'; break; } if (!isset($node_urllist)) $node_urllist = TRUE; // Get value of Printer-friendly URLs setting return (variable_get('print_urls', PRINT_URLS_DEFAULT) && ($node_urllist) && variable_get('print_'. $fmt .'display_urllist_'. $node->type, PRINT_TYPE_URLLIST_DEFAULT)); } /** * Prepare a Printer-friendly-ready node body for content nodes * * @param $nid * node ID of the node to be rendered into a printer-friendly page * @param $cid * comment ID of the individual comment to be rendered * @param $format * format of the page being generated * @param $teaser * if set to TRUE, outputs only the node's teaser * @param $message * optional sender's message (used by the send e-mail module) * @return * filled array ready to be used in the template */ function _print_generate_node($nid, $cid = NULL, $format = PRINT_HTML_FORMAT, $teaser = FALSE, $message = NULL) { global $_print_urls; // We can take a node id $node = node_load(array('nid' => $nid)); if (!$node) { // Node not found drupal_not_found(); return FALSE; } elseif (!node_access('view', $node)) { // Access is denied drupal_access_denied(); return FALSE; } drupal_set_title($node->title); //alert other modules that we are generating a printer-friendly page, so they can choose to show/hide info $node->printing = TRUE; // Turn off Pagination by the Paging module unset($node->pages); unset($node->page_count); if ($cid === NULL) { // Adapted (simplified) version of node_view //Render the node content $node = node_build_content($node, $teaser, TRUE); // Disable fivestar widget output unset($node->content['fivestar_widget']); // Disable service links module output unset($node->content['service_links']); $content = drupal_render($node->content); // Disable the AdSense module ads $content = preg_replace('!
!sim', '', $content); if ($teaser) { $node->teaser = $content; unset($node->body); } else { $node->body = $content; unset($node->teaser); } //TODO the following was part of the fix for http://drupal.org/node/254863 //check if it is reproducible and find the exact condition which //triggered it //$node->body = html_entity_decode($node->body); } $print_comments = variable_get('print_comments', PRINT_COMMENTS_DEFAULT); if (function_exists('comment_render') && (($cid != NULL) || ($print_comments))) { //Print only the requested comment (or if $cid is NULL, all of them) $comments = comment_render($node, $cid); //Remove the comment forms $comments = preg_replace('!.*?!sim', '', $comments); //Remove the 'Post new comment' title $comments = preg_replace('!'. t('Post new comment') .'!', '', $comments); //Remove the comment title hyperlink $comments = preg_replace('!(.*?)(.*?)(.*?)!i', '$1$2$3', $comments); //Remove the comment author link $pattern = '!(<(?:span|div) class="submitted">.*?)(.*?)(.*?)!sim'; if (preg_match($pattern, $comments)) { $comments = preg_replace($pattern , '$1$2$3', $comments); } //Remove the comment links $comments = preg_replace('!\s*!sim', '', $comments); if ($cid != NULL) { // Single comment requested, output only the comment unset($node->body); } $node->body .= $comments; } node_invoke_nodeapi($node, 'alter', $teaser, TRUE); if ($teaser) { $node->body = $node->teaser; unset($node->teaser); } //Check URL list settings $_print_urls = _print_url_list_enabled($node, $format); // Convert the a href elements $pattern = '!<(a\s[^>]*?)>(.*?)()!is'; $node->body = preg_replace_callback($pattern, '_print_rewrite_urls', $node->body); init_theme(); $print = _print_var_generator($node, $message, $cid); return $print; } /** * Prepare a Printer-friendly-ready node body for non-content pages * * @param $path * path of the node to be rendered into a printer-friendly page * @param $format * format of the page being generated * @param $teaser * if set to TRUE, outputs only the node's teaser * @param $message * optional sender's message (used by the send e-mail module) * @return * filled array ready to be used in the template */ function _print_generate_path($path, $format = PRINT_HTML_FORMAT, $teaser = FALSE, $message = NULL) { global $_print_urls; $path = drupal_get_normal_path($path); menu_set_active_item($path); // Adapted from index.php. $node = new stdClass(); $node->body = menu_execute_active_handler($path); $node->title = drupal_get_title(); $node->path = $path; // It may happen that a drupal_not_found is called in the above call if (preg_match('/404 Not Found/', drupal_get_headers()) == 1) { return FALSE; } if (is_int($node->body)) { switch ($node->body) { case MENU_NOT_FOUND: drupal_not_found(); return FALSE; break; case MENU_ACCESS_DENIED: drupal_access_denied(); return FALSE; break; } } // Delete any links area $node->body = preg_replace('!\s*!sim', '', $node->body); //Check URL list settings $_print_urls = _print_url_list_enabled($node, $format); // Convert the a href elements $pattern = '!<(a\s[^>]*?)>(.*?)()!is'; $node->body = preg_replace_callback($pattern, '_print_rewrite_urls', $node->body); init_theme(); $print = _print_var_generator($node, $message); return $print; } /** * Prepare a Printer-friendly-ready node body for book pages * * @param $nid * node ID of the node to be rendered into a printer-friendly page * @param $format * format of the page being generated * @param $teaser * if set to TRUE, outputs only the node's teaser * @param $message * optional sender's message (used by the send e-mail module) * @return * filled array ready to be used in the template */ function _print_generate_book($nid, $format = PRINT_HTML_FORMAT, $teaser = FALSE, $message = NULL) { global $_print_urls; $node = node_load(array('nid' => $nid)); if (!$node) { // Node not found drupal_not_found(); return FALSE; } elseif (!node_access('view', $node) || (!user_access('access printer-friendly version'))) { // Access is denied drupal_access_denied(); return FALSE; } $tree = book_menu_subtree_data($node->book); $node->body = book_export_traverse($tree, 'book_node_export'); //Check URL list settings $_print_urls = _print_url_list_enabled($node, $format); // Convert the a href elements $pattern = '!<(a\s[^>]*?)>(.*?)()!is'; $node->body = preg_replace_callback($pattern, '_print_rewrite_urls', $node->body); init_theme(); $print = _print_var_generator($node, $message); // The title is already displayed by the book_recurse, so avoid duplication $print['title'] = ''; return $print; }