Completed
Push — develop ( de7659...415144 )
by Daniel
09:44
created

forum_topics::show_topics()   B

Complexity

Conditions 2
Paths 2

Size

Total Lines 29
Code Lines 21

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 23
CRAP Score 2

Importance

Changes 1
Bugs 0 Features 1
Metric Value
c 1
b 0
f 1
dl 0
loc 29
ccs 23
cts 23
cp 1
rs 8.8571
cc 2
eloc 21
nc 2
nop 2
crap 2
1
<?php
2
/**
3
 *
4
 * @package sitemaker
5
 * @copyright (c) 2013 Daniel A. (blitze)
6
 * @license http://opensource.org/licenses/gpl-2.0.php GNU General Public License v2
7
 *
8
 */
9
10
namespace blitze\sitemaker\blocks;
11
12
use blitze\sitemaker\services\blocks\driver\block;
13
use Nickvergessen\TrimMessage\TrimMessage;
14
15
/**
16
 * Forum Topics Block
17
 */
18
class forum_topics extends block
19
{
20
	/** @var \phpbb\auth\auth */
21
	protected $auth;
22
23
	/** @var \phpbb\content_visibility */
24
	protected $content_visibility;
25
26
	/** @var \phpbb\language\language */
27
	protected $translator;
28
29
	/** @var \phpbb\user */
30
	protected $user;
31
32
	/** @var \blitze\sitemaker\services\date_range */
33
	protected $date_range;
34
35
	/** @var \blitze\sitemaker\services\forum\data */
36
	protected $forum_data;
37
38
	/** @var \blitze\sitemaker\services\forum\options */
39
	protected $forum_options;
40
41
	/** @var string */
42
	protected $phpbb_root_path;
43
44
	/** @var string */
45
	protected $php_ext;
46
47
	/** @var array */
48
	private $fields = array();
49
50
	/** @var array */
51
	private $settings = array();
52
53
	/** @var array */
54
	private $topic_tracking_info = array();
55
56
	const FORUMS_ORDER_FIRST_POST = 0;
57
	const FORUMS_ORDER_LAST_POST = 1;
58
	const FORUMS_ORDER_LAST_READ = 2;
59
60
	/**
61
	 * Constructor
62
	 *
63
	 * @param \phpbb\auth\auth							$auth				Permission object
64
	 * @param \phpbb\content_visibility					content_visibility	Content visibility object
65
	 * @param \phpbb\language\language					$translator			Language object
66
	 * @param \phpbb\user								$user				User object
67
	 * @param \blitze\sitemaker\services\date_range		$date_range			Date Range Object
68
	 * @param \blitze\sitemaker\services\forum\data		$forum_data			Forum Data object
69
	 * @param \blitze\sitemaker\services\forum\options	$forum_options		Forum Data object
70
	 * @param string									$phpbb_root_path	Path to the phpbb includes directory.
71
	 * @param string									$php_ext			php file extension
72
	 */
73 6
	public function __construct(\phpbb\auth\auth $auth, \phpbb\content_visibility $content_visibility, \phpbb\language\language $translator, \phpbb\user $user, \blitze\sitemaker\services\date_range $date_range, \blitze\sitemaker\services\forum\data $forum_data, \blitze\sitemaker\services\forum\options $forum_options, $phpbb_root_path, $php_ext)
74
	{
75 6
		$this->auth = $auth;
76 6
		$this->content_visibility = $content_visibility;
77 6
		$this->translator = $translator;
78 6
		$this->user = $user;
79 6
		$this->date_range = $date_range;
80 6
		$this->forum_data = $forum_data;
81 6
		$this->forum_options = $forum_options;
82 6
		$this->phpbb_root_path = $phpbb_root_path;
83 6
		$this->php_ext = $php_ext;
84 6
	}
85
86
	/**
87
	 * {@inheritdoc}
88
	 */
89 1
	public function get_config(array $settings)
90
	{
91 1
		$forum_options = $this->forum_options->get_all();
92 1
		$topic_type_options = $this->get_topic_type_options();
93 1
		$preview_options = $this->get_preview_options();
94 1
		$range_options = $this->get_range_options();
95 1
		$sort_options = $this->get_sorting_options();
96 1
		$template_options = $this->get_view_options();
97
98
		return array(
99 1
			'legend1'		=> 'SETTINGS',
100 1
			'forum_ids'			=> array('lang' => 'SELECT_FORUMS', 'validate' => 'string', 'type' => 'multi_select', 'options' => $forum_options, 'default' => array(), 'explain' => false),
101 1
			'topic_type'		=> array('lang' => 'TOPIC_TYPE', 'validate' => 'string', 'type' => 'checkbox', 'options' => $topic_type_options, 'default' => array(), 'explain' => false),
102 1
			'max_topics'		=> array('lang' => 'MAX_TOPICS', 'validate' => 'int:0:20', 'type' => 'number:0:20', 'maxlength' => 2, 'explain' => false, 'default' => 5),
103 1
			'date_range'		=> array('lang' => 'LIMIT_POST_TIME', 'validate' => 'string', 'type' => 'select', 'options' => $range_options, 'default' => '', 'explain' => false),
104 1
			'order_by'			=> array('lang' => 'ORDER_BY', 'validate' => 'string', 'type' => 'select', 'options' => $sort_options, 'default' => self::FORUMS_ORDER_LAST_POST, 'explain' => false),
105
106 1
			'legend2'		=> 'DISPLAY',
107 1
			'enable_tracking'	=> array('lang' => 'ENABLE_TOPIC_TRACKING', 'validate' => 'bool', 'type' => 'radio:yes_no', 'explain' => false, 'default' => false),
108 1
			'topic_title_limit'	=> array('lang' => 'TOPIC_TITLE_LIMIT', 'validate' => 'int:0:255', 'type' => 'number:0:255', 'maxlength' => 3, 'explain' => false, 'default' => 25),
109 1
			'template'			=> array('lang' => 'TEMPLATE', 'validate' => 'string', 'type' => 'select', 'options' => $template_options, 'default' => 'titles', 'explain' => false),
110 1
			'display_preview'	=> array('lang' => 'DISPLAY_PREVIEW', 'validate' => 'string', 'type' => 'select', 'options' => $preview_options, 'default' => '', 'explain' => false),
111 1
			'preview_max_chars'	=> array('lang' => 'PREVIEW_MAX_CHARS', 'validate' => 'int:0:255', 'type' => 'number:0:255', 'maxlength' => 3, 'explain' => false, 'default' => 125),
112 1
		);
113
	}
114
115
	/**
116
	 * {@inheritdoc}
117
	 */
118 5
	public function display(array $bdata, $edit_mode = false)
119
	{
120 5
		$this->settings = $bdata['settings'];
121
122 5
		$topic_data = $this->get_topic_data();
123
124 5
		$content = '';
125 5
		if (sizeof($topic_data))
126 5
		{
127 5
			$content = $this->get_block_content($topic_data);
128 5
		}
129
130
		return array(
131 5
			'title'		=> $this->get_block_title(),
132 5
			'content'	=> $content,
133 5
		);
134
	}
135
136
	/**
137
	 * @param array $topic_data
138
	 * @return string
139
	 */
140 5
	protected function get_block_content(array $topic_data)
141
	{
142 5
		$this->set_display_fields();
143
144 5
		$view = 'S_' . strtoupper($this->settings['template']);
145 5
		$post_data = $this->get_post_data($topic_data);
146 5
		$topic_data = array_values($topic_data);
147
148 5
		$this->show_topics($topic_data, $post_data);
149 5
		unset($topic_data, $post_data);
150
151 5
		$this->ptemplate->assign_vars(array(
152 5
			$view				=> true,
153 5
			'S_IS_BOT'			=> $this->user->data['is_bot'],
154 5
			'LAST_POST_IMG'		=> $this->user->img('icon_topic_latest'),
155 5
			'NEWEST_POST_IMG'	=> $this->user->img('icon_topic_newest'),
156 5
		));
157
158 5
		return $this->ptemplate->render_view('blitze/sitemaker', 'blocks/forum_topics.html', 'forum_topics_block');
159
	}
160
161
	/**
162
	 * @param array $topic_data
163
	 * @param array $post_data
164
	 */
165 5
	protected function show_topics(array &$topic_data, array &$post_data)
166
	{
167 5
		for ($i = 0, $size = sizeof($topic_data); $i < $size; $i++)
168
		{
169 5
			$row =& $topic_data[$i];
170 5
			$forum_id = $row['forum_id'];
171 5
			$topic_id = $row['topic_id'];
172
173
			$tpl_ary = array(
174 5
				'FORUM_TITLE'		=> $row['forum_name'],
175 5
				'TOPIC_TITLE'		=> truncate_string(censor_text($row['topic_title']), $this->settings['topic_title_limit'], 255, false, '...'),
176 5
				'TOPIC_AUTHOR'		=> get_username_string('full', $row[$this->fields['user_id']], $row[$this->fields['username']], $row[$this->fields['user_colour']]),
177 5
				'TOPIC_PREVIEW'		=> $this->get_preview(array_pop($post_data[$topic_id])),
178 5
				'TOPIC_POST_TIME'	=> $this->user->format_date($row[$this->fields['time']]),
179 5
				'ATTACH_ICON_IMG'	=> $this->get_attachment_icon($forum_id, $row['topic_attachment']),
180 5
				'REPLIES'			=> $this->content_visibility->get_count('topic_posts', $row, $forum_id) - 1,
181 5
				'VIEWS'				=> $row['topic_views'],
182 5
				'S_UNREAD_TOPIC'	=> $this->is_unread_topic($forum_id, $topic_id, $row['topic_last_post_time']),
183
184 5
				'U_VIEWTOPIC'		=> append_sid($this->phpbb_root_path . 'viewtopic.' . $this->php_ext, "f=$forum_id&amp;t=$topic_id"),
185 5
				'U_VIEWFORUM'		=> append_sid($this->phpbb_root_path . 'viewforum.' . $this->php_ext, "f=$forum_id"),
186 5
				'U_NEW_POST'		=> append_sid($this->phpbb_root_path . 'viewtopic.' . $this->php_ext, "f=$forum_id&amp;t=$topic_id&amp;view=unread") . '#unread',
187 5
				'U_LAST_POST'		=> append_sid($this->phpbb_root_path . 'viewtopic.' . $this->php_ext, "f=$forum_id&amp;t=$topic_id&amp;p=" . $row['topic_last_post_id']) . '#p' . $row['topic_last_post_id'],
188 5
			);
189
190 5
			$this->ptemplate->assign_block_vars('topicrow', $tpl_ary);
191 5
			unset($topic_data[$i], $post_data[$topic_id]);
192 5
		}
193 5
	}
194
195
	/**
196
	 * @return string
197
	 */
198 5
	protected function get_block_title()
199
	{
200
		$types = array(
201 5
			POST_GLOBAL		=> 'FORUM_GLOBAL_ANNOUNCEMENTS',
202 5
			POST_ANNOUNCE	=> 'FORUM_ANNOUNCEMENTS',
203 5
			POST_STICKY		=> 'FORUM_STICKY_POSTS',
204 5
			POST_NORMAL		=> 'FORUM_RECENT_TOPICS',
205 5
		);
206
207
		// if more than one topic type is selected, we default to RECENT_TOPICS
208 5
		$topic_type = join(',', $this->settings['topic_type']);
209
210 5
		return ($this->settings['order_by'] !== self::FORUMS_ORDER_LAST_READ) ? (isset($types[$topic_type]) ? $types[$topic_type] : 'FORUM_RECENT_TOPICS') : 'TOPICS_LAST_READ';
211
	}
212
213
	/**
214
	 * @param array $row
215
	 * @return string
216
	 */
217 5
	protected function get_preview(array $row)
218
	{
219 5
		$preview = '';
220 5
		if ($this->settings['display_preview'])
221 5
		{
222 3
			$trim = new TrimMessage($row['post_text'], $row['bbcode_uid'], $this->settings['preview_max_chars']);
223 3
			$row['post_text'] = $trim->message();
224 3
			unset($trim);
225
226 3
			$parse_flags = ($row['bbcode_bitfield'] ? OPTION_FLAG_BBCODE : 0) | OPTION_FLAG_SMILIES;
227 3
			$preview = generate_text_for_display($row['post_text'], $row['bbcode_uid'], $row['bbcode_bitfield'], $parse_flags, true);
228 3
		}
229
230 5
		return $preview;
231
	}
232
233
	/**
234
	 * @return array
235
	 */
236 5
	private function get_topic_data()
237
	{
238
		$sort_order = array(
239 5
			self::FORUMS_ORDER_FIRST_POST		=> 't.topic_time',
240 5
			self::FORUMS_ORDER_LAST_POST		=> 't.topic_last_post_time',
241 5
			self::FORUMS_ORDER_LAST_READ		=> 't.topic_last_view_time'
242 5
		);
243
244 5
		$range_info = $this->date_range->get($this->settings['date_range']);
245
246 5
		$this->forum_data->query($this->settings['enable_tracking'])
247 5
			->fetch_forum($this->settings['forum_ids'])
248 5
			->fetch_topic_type($this->settings['topic_type'])
249 5
			->fetch_date_range($range_info['start'], $range_info['stop'])
250 5
			->set_sorting($sort_order[$this->settings['order_by']])
251 5
			->build();
252
253 5
		$topic_data = $this->forum_data->get_topic_data($this->settings['max_topics']);
254 5
		$this->topic_tracking_info = $this->forum_data->get_topic_tracking_info();
255
256 5
		return $topic_data;
257
	}
258
259
	/**
260
	 * @param array $topic_data
261
	 * @return array
262
	 */
263 5
	private function get_post_data(array $topic_data)
264
	{
265 5
		if ($this->settings['display_preview'])
266 5
		{
267 3
			$post_data = $this->forum_data->get_post_data($this->settings['display_preview']);
268 3
		}
269
		else
270
		{
271 2
			$post_data = array_fill_keys(array_keys($topic_data), array(array('post_text' => '', 'bbcode_uid' => '', 'bbcode_bitfield' => '')));
272
		}
273
274 5
		return $post_data;
275
	}
276
277
	/**
278
	 *
279
	 */
280 5
	private function set_display_fields()
281
	{
282 5
		if ($this->settings['display_preview'] == 'last')
283 5
		{
284 1
			$this->fields['time'] = 'topic_last_post_time';
285 1
			$this->fields['user_id'] = 'topic_last_poster_id';
286 1
			$this->fields['username'] = 'topic_last_poster_name';
287 1
			$this->fields['user_colour'] = 'topic_last_poster_colour';
288
289 1
			$this->ptemplate->assign_var('L_POST_BY_AUTHOR', $this->translator->lang('LAST_POST_BY_AUTHOR'));
290 1
		}
291
		else
292
		{
293 4
			$this->fields['time'] = 'topic_time';
294 4
			$this->fields['user_id'] = 'topic_poster';
295 4
			$this->fields['username'] = 'topic_first_poster_name';
296 4
			$this->fields['user_colour'] = 'topic_first_poster_colour';
297
		}
298 5
	}
299
300
	/**
301
	 * @param int $forum_id
302
	 * @param int $topic_attachment
303
	 * @return string
304
	 */
305 5
	private function get_attachment_icon($forum_id, $topic_attachment)
306
	{
307 5
		return ($this->user_can_view_attachments($forum_id) && $topic_attachment) ? $this->user->img('icon_topic_attach', $this->translator->lang('TOTAL_ATTACHMENTS')) : '';
308
	}
309
310
	/**
311
	 * @param int $forum_id
312
	 * @return bool
313
	 */
314 5
	private function user_can_view_attachments($forum_id)
315
	{
316 5
		return ($this->auth->acl_get('u_download') && $this->auth->acl_get('f_download', $forum_id)) ? true : false;
317
	}
318
319
	/**
320
	 * @param int $forum_id
321
	 * @param int $topic_id
322
	 * @param int $topic_last_post_time
323
	 * @return bool
324
	 */
325 5
	private function is_unread_topic($forum_id, $topic_id, $topic_last_post_time)
326
	{
327 5
		return (isset($this->topic_tracking_info[$forum_id][$topic_id]) && $topic_last_post_time > $this->topic_tracking_info[$forum_id][$topic_id]) ? true : false;
328
	}
329
330
	/**
331
	 * @return array
332
	 */
333 1
	private function get_topic_type_options()
334
	{
335
		return array(
336 1
			POST_NORMAL     => 'POST_NORMAL',
337 1
			POST_STICKY     => 'POST_STICKY',
338 1
			POST_ANNOUNCE   => 'POST_ANNOUNCEMENT',
339 1
			POST_GLOBAL     => 'POST_GLOBAL',
340 1
		);
341
	}
342
343
	/**
344
	 * @return array
345
	 */
346 1
	private function get_preview_options()
347
	{
348
		return array(
349 1
			''      => 'NO',
350 1
			'first' => 'SHOW_FIRST_POST',
351 1
			'last'  => 'SHOW_LAST_POST',
352 1
		);
353
	}
354
355
	/**
356
	 * @return array
357
	 */
358 1
	private function get_range_options()
359
	{
360
		return array(
361 1
			''      => 'ALL_TIME',
362 1
			'today' => 'TODAY',
363 1
			'week'  => 'THIS_WEEK',
364 1
			'month' => 'THIS_MONTH',
365 1
			'year'  => 'THIS_YEAR',
366 1
		);
367
	}
368
369
	/**
370
	 * @return array
371
	 */
372 1
	private function get_sorting_options()
373
	{
374
		return array(
375 1
			self::FORUMS_ORDER_FIRST_POST => 'FIRST_POST_TIME',
376 1
			self::FORUMS_ORDER_LAST_POST  => 'LAST_POST_TIME',
377 1
			self::FORUMS_ORDER_LAST_READ  => 'LAST_READ_TIME',
378 1
		);
379
	}
380
381
	/**
382
	 * @return array
383
	 */
384 1
	private function get_view_options()
385
	{
386
		return array(
387 1
			'titles'    => 'TITLES',
388 1
			'mini'      => 'MINI',
389 1
			'context'   => 'CONTEXT',
390 1
		);
391
	}
392
}
393