2
0

index.template.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577
  1. <?php
  2. /**
  3. * Simple Machines Forum (SMF)
  4. *
  5. * @package SMF
  6. * @author Simple Machines
  7. * @copyright 2014 Simple Machines and individual contributors
  8. * @license http://www.simplemachines.org/about/smf/license.php BSD
  9. *
  10. * @version 2.1 Alpha 1
  11. */
  12. /* This template is, perhaps, the most important template in the theme. It
  13. contains the main template layer that displays the header and footer of
  14. the forum, namely with main_above and main_below. It also contains the
  15. menu sub template, which appropriately displays the menu; the init sub
  16. template, which is there to set the theme up; (init can be missing.) and
  17. the linktree sub template, which sorts out the link tree.
  18. The init sub template should load any data and set any hardcoded options.
  19. The main_above sub template is what is shown above the main content, and
  20. should contain anything that should be shown up there.
  21. The main_below sub template, conversely, is shown after the main content.
  22. It should probably contain the copyright statement and some other things.
  23. The linktree sub template should display the link tree, using the data
  24. in the $context['linktree'] variable.
  25. The menu sub template should display all the relevant buttons the user
  26. wants and or needs.
  27. For more information on the templating system, please see the site at:
  28. http://www.simplemachines.org/
  29. */
  30. /**
  31. * Initialize the template... mainly little settings.
  32. */
  33. function template_init()
  34. {
  35. global $settings;
  36. /* $context, $options and $txt may be available for use, but may not be fully populated yet. */
  37. /* Use images from default theme when using templates from the default theme?
  38. if this is 'always', images from the default theme will be used.
  39. if this is 'defaults', images from the default theme will only be used with default templates.
  40. if this is 'never' or isn't set at all, images from the default theme will not be used. */
  41. $settings['use_default_images'] = 'never';
  42. // The version this template/theme is for. This should probably be the version of SMF it was created for.
  43. $settings['theme_version'] = '2.1';
  44. // Use plain buttons - as opposed to text buttons?
  45. $settings['use_buttons'] = true;
  46. // Show sticky and lock status separate from topic icons?
  47. $settings['separate_sticky_lock'] = true;
  48. // Set the following variable to true if this theme requires the optional theme strings file to be loaded.
  49. $settings['require_theme_strings'] = false;
  50. // Set the following variable to true is this theme wants to display the avatar of the user that posted the last post on the board index and message index
  51. $settings['avatars_on_indexes'] = false;
  52. }
  53. /**
  54. * The main sub template above the content.
  55. */
  56. function template_html_above()
  57. {
  58. global $context, $settings, $scripturl, $txt, $modSettings;
  59. // Show right to left and the character set for ease of translating.
  60. echo '<!DOCTYPE html>
  61. <html', $context['right_to_left'] ? ' dir="rtl"' : '', '>
  62. <head>';
  63. // The ?alp21 part of this link is just here to make sure browsers don't cache it wrongly.
  64. echo '
  65. <link rel="stylesheet" type="text/css" href="', $settings['theme_url'], '/css/index.css?alp21" />';
  66. // The most efficient way of writing multi themes is to use a master index.css plus variant.css files.
  67. if (!empty($context['theme_variant']))
  68. echo '
  69. <link rel="stylesheet" type="text/css" href="', $settings['theme_url'], '/css/index', $context['theme_variant'], '.css?alp21" />';
  70. // Save some database hits, if a width for multiple wrappers is set in admin.
  71. if (!empty($settings['forum_width']))
  72. echo '
  73. <style type="text/css">#wrapper, .frame {width: ', $settings['forum_width'], ';}</style>';
  74. // Quick and dirty testing of RTL horrors. Remove before production build.
  75. //echo '
  76. //<link rel="stylesheet" type="text/css" href="', $settings['theme_url'], '/css/rtl.css?alp21" />';
  77. // load in any css from mods or themes so they can overwrite if wanted
  78. template_css();
  79. // load in any javascript files from mods and themes
  80. template_javascript();
  81. // RTL languages require an additional stylesheet.
  82. if ($context['right_to_left'])
  83. {
  84. echo '
  85. <link rel="stylesheet" type="text/css" href="', $settings['theme_url'], '/css/rtl.css?alp21" />';
  86. if (!empty($context['theme_variant']))
  87. echo '
  88. <link rel="stylesheet" type="text/css" href="', $settings['theme_url'], '/css/rtl', $context['theme_variant'], '.css?alp21" />';
  89. }
  90. echo '
  91. <meta http-equiv="Content-Type" content="text/html; charset=', $context['character_set'], '" />
  92. <meta name="description" content="', !empty($context['meta_description']) ? $context['meta_description'] : $context['page_title_html_safe'], '" />', !empty($context['meta_keywords']) ? '
  93. <meta name="keywords" content="' . $context['meta_keywords'] . '" />' : '', '
  94. <title>', $context['page_title_html_safe'], '</title>';
  95. // Please don't index these Mr Robot.
  96. if (!empty($context['robot_no_index']))
  97. echo '
  98. <meta name="robots" content="noindex" />';
  99. // Present a canonical url for search engines to prevent duplicate content in their indices.
  100. if (!empty($context['canonical_url']))
  101. echo '
  102. <link rel="canonical" href="', $context['canonical_url'], '" />';
  103. // Show all the relative links, such as help, search, contents, and the like.
  104. echo '
  105. <link rel="help" href="', $scripturl, '?action=help" />
  106. <link rel="contents" href="', $scripturl, '" />', ($context['allow_search'] ? '
  107. <link rel="search" href="' . $scripturl . '?action=search" />' : '');
  108. // If RSS feeds are enabled, advertise the presence of one.
  109. if (!empty($modSettings['xmlnews_enable']) && (!empty($modSettings['allow_guestAccess']) || $context['user']['is_logged']))
  110. echo '
  111. <link rel="alternate" type="application/rss+xml" title="', $context['forum_name_html_safe'], ' - ', $txt['rss'], '" href="', $scripturl, '?type=rss2;action=.xml" />
  112. <link rel="alternate" type="application/rss+xml" title="', $context['forum_name_html_safe'], ' - ', $txt['atom'], '" href="', $scripturl, '?type=atom;action=.xml" />';
  113. // If we're viewing a topic, these should be the previous and next topics, respectively.
  114. if (!empty($context['links']['next']))
  115. {
  116. echo '
  117. <link rel="next" href="', $context['links']['next'], '" />';
  118. }
  119. if (!empty($context['links']['prev']))
  120. {
  121. echo '
  122. <link rel="prev" href="', $context['links']['prev'], '" />';
  123. }
  124. // If we're in a board, or a topic for that matter, the index will be the board's index.
  125. if (!empty($context['current_board']))
  126. echo '
  127. <link rel="index" href="', $scripturl, '?board=', $context['current_board'], '.0" />';
  128. // Output any remaining HTML headers. (from mods, maybe?)
  129. echo $context['html_headers'];
  130. echo '
  131. </head>
  132. <body id="', $context['browser_body_id'], '" class="action_', !empty($context['current_action']) ? $context['current_action'] : (!empty($context['current_board']) ?
  133. 'messageindex' : (!empty($context['current_topic']) ? 'display' : 'home')), !empty($context['current_board']) ? ' board_' . $context['current_board'] : '', '">';
  134. }
  135. function template_body_above()
  136. {
  137. global $context, $settings, $scripturl, $txt, $modSettings;
  138. // Wrapper div now echoes permanently for better layout options. h1 a is now target for "Go up" links.
  139. echo '
  140. <div id="top_section">
  141. <div class="frame">';
  142. // If the user is logged in, display some things that might be useful.
  143. if ($context['user']['is_logged'])
  144. {
  145. // Firstly, the user's menu
  146. echo '
  147. <ul class="floatleft" id="top_info">
  148. <li>
  149. <a href="', $scripturl, '?action=profile"', !empty($context['self_profile']) ? ' class="active"' : '', ' id="profile_menu_top" onclick="return false;">', $context['user']['name'], ' &#9660;</a>
  150. <div id="profile_menu" class="top_menu"></div>
  151. </li>';
  152. // Secondly, PMs if we're doing them
  153. if ($context['allow_pm'])
  154. {
  155. echo '
  156. <li>
  157. <a href="', $scripturl, '?action=pm"', !empty($context['self_pm']) ? ' class="active"' : '', ' id="pm_menu_top">', $txt['pm_short'], !empty($context['user']['unread_messages']) ? ' <span class="amt">' . $context['user']['unread_messages'] . '</span>' : '', '</a>
  158. <div id="pm_menu" class="top_menu"></div>
  159. </li>';
  160. }
  161. // Thirdly, alerts
  162. echo '
  163. <li>
  164. <a href="', $scripturl, '?action=alerts"', !empty($context['self_alerts']) ? ' class="active"' : '', ' id="alerts_menu_top">', $txt['alerts'], !empty($context['user']['alerts']) ? ' <span class="amt">' . $context['user']['alerts'] . '</span>' : '', '</a>
  165. <div id="alerts_menu" class="top_menu"></div>
  166. </li>';
  167. // And now we're done.
  168. echo '
  169. </ul>';
  170. }
  171. // Otherwise they're a guest. Ask them to either register or login.
  172. else
  173. echo '
  174. <ul class="floatleft">
  175. <li>', sprintf($txt[$context['can_register'] ? 'welcome_guest_register' : 'welcome_guest'], $txt['guest_title'], $scripturl . '?action=login'), '</li>
  176. </ul>';
  177. if ($context['allow_search'])
  178. {
  179. echo '
  180. <form id="search_form" class="floatright" action="', $scripturl, '?action=search2" method="post" accept-charset="', $context['character_set'], '">
  181. <input type="text" name="search" value="" class="input_text" />&nbsp;';
  182. // Using the quick search dropdown?
  183. if (!empty($modSettings['search_dropdown']))
  184. {
  185. $selected = !empty($context['current_topic']) ? 'current_topic' : (!empty($context['current_board']) ? 'current_board' : 'all');
  186. echo '
  187. <select name="search_selection">
  188. <option value="all"', ($selected == 'all' ? ' selected="selected"' : ''), '>', $txt['search_entireforum'], ' </option>';
  189. // Can't limit it to a specific topic if we are not in one
  190. if (!empty($context['current_topic']))
  191. echo '
  192. <option value="topic"', ($selected == 'current_topic' ? ' selected="selected"' : ''), '>', $txt['search_thistopic'], '</option>';
  193. // Can't limit it to a specific board if we are not in one
  194. if (!empty($context['current_board']))
  195. echo '
  196. <option value="board"', ($selected == 'current_board' ? ' selected="selected"' : ''), '>', $txt['search_thisbrd'], '</option>';
  197. echo '
  198. <option value="members"', ($selected == 'members' ? ' selected="selected"' : ''), '>', $txt['search_members'], ' </option>
  199. </select>';
  200. }
  201. // Search within current topic?
  202. if (!empty($context['current_topic']))
  203. echo '
  204. <input type="hidden" name="', (!empty($modSettings['search_dropdown']) ? 'sd_topic' : 'topic'), '" value="', $context['current_topic'], '" />';
  205. // If we're on a certain board, limit it to this board ;).
  206. elseif (!empty($context['current_board']))
  207. echo '
  208. <input type="hidden" name="', (!empty($modSettings['search_dropdown']) ? 'sd_brd[' : 'brd['), $context['current_board'], ']"', ' value="', $context['current_board'], '" />';
  209. echo '
  210. <input type="submit" name="search2" value="', $txt['search'], '" class="button_submit" />
  211. <input type="hidden" name="advanced" value="0" />
  212. </form>';
  213. }
  214. echo '
  215. </div>
  216. </div>';
  217. echo '
  218. <div id="header">
  219. <div class="frame">
  220. <h1 class="forumtitle">
  221. <a id="top" href="', $scripturl, '">', empty($context['header_logo_url_html_safe']) ? $context['forum_name'] : '<img src="' . $context['header_logo_url_html_safe'] . '" alt="' . $context['forum_name'] . '" />', '</a>
  222. </h1>';
  223. echo '
  224. ', empty($settings['site_slogan']) ? '<img id="smflogo" src="' . $settings['images_url'] . '/smflogo.png" alt="Simple Machines Forum" title="Simple Machines Forum" />' : '<div id="siteslogan" class="floatright">' . $settings['site_slogan'] . '</div>', '';
  225. echo'
  226. </div>
  227. </div>
  228. <div id="wrapper">
  229. <div id="upper_section">
  230. <div id="inner_section">
  231. <div id="inner_wrap">
  232. <div class="user">';
  233. // Otherwise they're a guest - this time ask them to either register or login - lazy bums...
  234. if (!empty($context['show_login_bar']))
  235. {
  236. echo '
  237. <script src="', $settings['default_theme_url'], '/scripts/sha1.js"></script>
  238. <form id="guest_form" action="', $scripturl, '?action=login2;quicklogin" method="post" accept-charset="', $context['character_set'], '" ', empty($context['disable_login_hashing']) ? ' onsubmit="hashLoginPassword(this, \'' . $context['session_id'] . '\', \'' . (!empty($context['login_token']) ? $context['login_token'] : '') . '\');"' : '', '>
  239. <input type="text" name="user" size="10" class="input_text" />
  240. <input type="password" name="passwrd" size="10" class="input_password" />
  241. <select name="cookielength">
  242. <option value="60">', $txt['one_hour'], '</option>
  243. <option value="1440">', $txt['one_day'], '</option>
  244. <option value="10080">', $txt['one_week'], '</option>
  245. <option value="43200">', $txt['one_month'], '</option>
  246. <option value="-1" selected="selected">', $txt['forever'], '</option>
  247. </select>
  248. <input type="submit" value="', $txt['login'], '" class="button_submit" />
  249. <div>', $txt['quick_login_dec'], '</div>';
  250. if (!empty($modSettings['enableOpenID']))
  251. echo '
  252. <br /><input type="text" name="openid_identifier" size="25" class="input_text openid_login" />';
  253. echo '
  254. <input type="hidden" name="hash_passwrd" value="" />
  255. <input type="hidden" name="', $context['session_var'], '" value="', $context['session_id'], '" />
  256. <input type="hidden" name="', $context['login_token_var'], '" value="', $context['login_token'], '" />
  257. </form>';
  258. }
  259. else
  260. {
  261. echo '
  262. ', $context['current_time'];
  263. }
  264. echo'
  265. </div>';
  266. // Show a random news item? (or you could pick one from news_lines...)
  267. if (!empty($settings['enable_news']) && !empty($context['random_news_line']))
  268. echo '
  269. <div class="news">
  270. <h2>', $txt['news'], ': </h2>
  271. <p>', $context['random_news_line'], '</p>
  272. </div>';
  273. echo '
  274. <hr class="clear" />
  275. </div>';
  276. // Show the menu here, according to the menu sub template, followed by the navigation tree.
  277. template_menu();
  278. theme_linktree();
  279. echo '
  280. </div>
  281. </div>';
  282. // The main content should go here.
  283. echo '
  284. <div id="content_section">
  285. <div id="main_content_section">';
  286. }
  287. function template_body_below()
  288. {
  289. global $context, $txt;
  290. echo '
  291. </div>
  292. </div>
  293. </div>';
  294. // Show the XHTML, RSS and WAP2 links, as well as the copyright.
  295. // Footer is now full-width by default. Frame inside it will match theme wrapper width automatically.
  296. echo '
  297. <div id="footer_section">
  298. <div class="frame">';
  299. // There is now a global "Go to top" link at the right.
  300. echo '
  301. <a href="#top_section" id="bot" class="go_up">', $txt['go_up'], '</a>
  302. <ul class="reset">
  303. <li class="copyright">', theme_copyright(), '</li>
  304. </ul>';
  305. // Show the load time?
  306. if ($context['show_load_time'])
  307. echo '
  308. <p>', sprintf($txt['page_created_full'], $context['load_time'], $context['load_queries']), '</p>';
  309. echo '
  310. </div>
  311. </div>';
  312. }
  313. function template_html_below()
  314. {
  315. // load in any javascipt that could be defered to the end of the page
  316. template_javascript(true);
  317. echo '
  318. </body>
  319. </html>';
  320. }
  321. /**
  322. * Show a linktree. This is that thing that shows "My Community | General Category | General Discussion"..
  323. * @param bool $force_show = false
  324. */
  325. function theme_linktree($force_show = false)
  326. {
  327. global $context, $settings, $shown_linktree, $scripturl, $txt;
  328. // If linktree is empty, just return - also allow an override.
  329. if (empty($context['linktree']) || (!empty($context['dont_default_linktree']) && !$force_show))
  330. return;
  331. echo '
  332. <div class="navigate_section">
  333. <ul>';
  334. if ($context['user']['is_logged'])
  335. echo '
  336. <li class="unread_links">
  337. <a href="', $scripturl, '?action=unread" title="', $txt['unread_since_visit'], '">', $txt['view_unread_category'], '</a>
  338. <a href="', $scripturl, '?action=unreadreplies" title="', $txt['show_unread_replies'], '">', $txt['unread_replies'], '</a>
  339. </li>';
  340. // Each tree item has a URL and name. Some may have extra_before and extra_after.
  341. foreach ($context['linktree'] as $link_num => $tree)
  342. {
  343. echo '
  344. <li', ($link_num == count($context['linktree']) - 1) ? ' class="last"' : '', '>';
  345. // Don't show a separator for the first one.
  346. // Better here. Always points to the next level when the linktree breaks to a second line.
  347. // Picked a better looking HTML entity, and added support for RTL plus a span for styling.
  348. if ($link_num != 0)
  349. echo '
  350. <span class="dividers">', $context['right_to_left'] ? ' &#9668; ' : ' &#9658; ', '</span>';
  351. // Show something before the link?
  352. if (isset($tree['extra_before']))
  353. echo $tree['extra_before'], ' ';
  354. // Show the link, including a URL if it should have one.
  355. echo $settings['linktree_link'] && isset($tree['url']) ? '
  356. <a href="' . $tree['url'] . '"><span>' . $tree['name'] . '</span></a>' : '<span>' . $tree['name'] . '</span>';
  357. // Show something after the link...?
  358. if (isset($tree['extra_after']))
  359. echo ' ', $tree['extra_after'];
  360. echo '
  361. </li>';
  362. }
  363. echo '
  364. </ul>
  365. </div>';
  366. $shown_linktree = true;
  367. }
  368. /**
  369. * Show the menu up top. Something like [home] [help] [profile] [logout]...
  370. */
  371. function template_menu()
  372. {
  373. global $context;
  374. echo '
  375. <div id="main_menu">
  376. <ul class="dropmenu" id="menu_nav">';
  377. // Note: Menu markup has been cleaned up to remove unnecessary spans and classes.
  378. foreach ($context['menu_buttons'] as $act => $button)
  379. {
  380. echo '
  381. <li id="button_', $act, '"', !empty($button['sub_buttons']) ? ' class="subsections"' :'', '>
  382. <a', $button['active_button'] ? ' class="active"' : '', ' href="', $button['href'], '"', isset($button['target']) ? ' target="' . $button['target'] . '"' : '', '>
  383. ', $button['title'], '
  384. </a>';
  385. if (!empty($button['sub_buttons']))
  386. {
  387. echo '
  388. <ul>';
  389. foreach ($button['sub_buttons'] as $childbutton)
  390. {
  391. echo '
  392. <li', !empty($childbutton['sub_buttons']) ? ' class="subsections"' :'', '>
  393. <a href="', $childbutton['href'], '"' , isset($childbutton['target']) ? ' target="' . $childbutton['target'] . '"' : '', '>
  394. ', $childbutton['title'], '
  395. </a>';
  396. // 3rd level menus :)
  397. if (!empty($childbutton['sub_buttons']))
  398. {
  399. echo '
  400. <ul>';
  401. foreach ($childbutton['sub_buttons'] as $grandchildbutton)
  402. echo '
  403. <li>
  404. <a href="', $grandchildbutton['href'], '"' , isset($grandchildbutton['target']) ? ' target="' . $grandchildbutton['target'] . '"' : '', '>
  405. ', $grandchildbutton['title'], '
  406. </a>
  407. </li>';
  408. echo '
  409. </ul>';
  410. }
  411. echo '
  412. </li>';
  413. }
  414. echo '
  415. </ul>';
  416. }
  417. echo '
  418. </li>';
  419. }
  420. echo '
  421. </ul>
  422. </div>';
  423. }
  424. /**
  425. * Generate a strip of buttons.
  426. * @param array $button_strip
  427. * @param string $direction = ''
  428. * @param array $strip_options = array()
  429. */
  430. function template_button_strip($button_strip, $direction = '', $strip_options = array())
  431. {
  432. global $context, $txt;
  433. if (!is_array($strip_options))
  434. $strip_options = array();
  435. // List the buttons in reverse order for RTL languages.
  436. if ($context['right_to_left'])
  437. $button_strip = array_reverse($button_strip, true);
  438. // Create the buttons...
  439. $buttons = array();
  440. foreach ($button_strip as $key => $value)
  441. {
  442. // @todo this check here doesn't make much sense now (from 2.1 on), it should be moved to where the button array is generated
  443. // Kept for backward compatibility
  444. if (!isset($value['test']) || !empty($context[$value['test']]))
  445. $buttons[] = '
  446. <li><a' . (isset($value['id']) ? ' id="button_strip_' . $value['id'] . '"' : '') . ' class="button_strip_' . $key . (!empty($value['active']) ? ' active' : '') . '" href="' . $value['url'] . '"' . (isset($value['custom']) ? ' ' . $value['custom'] : '') . '><span>' . $txt[$value['text']] . '</span></a></li>';
  447. }
  448. // No buttons? No button strip either.
  449. if (empty($buttons))
  450. return;
  451. echo '
  452. <div class="buttonlist', !empty($direction) ? ' float' . $direction : '', '"', (empty($buttons) ? ' style="display: none;"' : ''), (!empty($strip_options['id']) ? ' id="' . $strip_options['id'] . '"': ''), '>
  453. <ul>',
  454. implode('', $buttons), '
  455. </ul>
  456. </div>';
  457. }
  458. function template_maint_warning_above()
  459. {
  460. global $txt, $context, $scripturl;
  461. echo '
  462. <div class="errorbox" id="errors">
  463. <dl>
  464. <dt>
  465. <strong id="error_serious">', $txt['forum_in_maintainence'], '</strong>
  466. </dt>
  467. <dd class="error" id="error_list">
  468. ', sprintf($txt['maintenance_page'], $scripturl . '?action=admin;area=serversettings;' . $context['session_var'] . '=' . $context['session_id']), '
  469. </dd>
  470. </dl>
  471. </div>';
  472. }
  473. function template_maint_warning_below()
  474. {
  475. }
  476. ?>