]> git.wh0rd.org Git - tt-rss.git/blob - classes/feeds.php
Merge branch 'fix-target-blank-vulnerability' into 'master'
[tt-rss.git] / classes / feeds.php
1 <?php
2 require_once "colors.php";
3
4 class Feeds extends Handler_Protected {
5
6     private $params;
7
8     function csrf_ignore($method) {
9                 $csrf_ignored = array("index", "feedbrowser", "quickaddfeed", "search");
10
11                 return array_search($method, $csrf_ignored) !== false;
12         }
13
14         private function format_headline_subtoolbar($feed_site_url, $feed_title,
15                         $feed_id, $is_cat, $search,
16                         $view_mode, $error, $feed_last_updated) {
17
18                 $catchup_sel_link = "catchupSelection()";
19
20                 $archive_sel_link = "archiveSelection()";
21                 $delete_sel_link = "deleteSelection()";
22
23                 $sel_all_link = "selectArticles('all')";
24                 $sel_unread_link = "selectArticles('unread')";
25                 $sel_none_link = "selectArticles('none')";
26                 $sel_inv_link = "selectArticles('invert')";
27
28                 $tog_unread_link = "selectionToggleUnread()";
29                 $tog_marked_link = "selectionToggleMarked()";
30                 $tog_published_link = "selectionTogglePublished()";
31
32                 $set_score_link = "setSelectionScore()";
33
34                 if ($is_cat) $cat_q = "&is_cat=$is_cat";
35
36                 if ($search) {
37                         $search_q = "&q=$search";
38                 } else {
39                         $search_q = "";
40                 }
41
42                 $reply .= "<span class=\"holder\">";
43
44                 $rss_link = htmlspecialchars(get_self_url_prefix() .
45                         "/public.php?op=rss&id=$feed_id$cat_q$search_q");
46
47                 // right part
48
49                 $error_class = $error ? "error" : "";
50
51                 $reply .= "<span class='r'>
52                         <a href=\"#\"
53                                 title=\"".__("View as RSS feed")."\"
54                                 onclick=\"displayDlg('".__("View as RSS")."','generatedFeed', '$feed_id:$is_cat:$rss_link')\">
55                                 <img class=\"noborder\" src=\"images/pub_set.png\"></a>";
56
57
58 #               $reply .= "<span>";
59                 $reply .= "<span id='feed_title' class='$error_class'>";
60
61                 if ($feed_site_url) {
62                         $last_updated = T_sprintf("Last updated: %s",
63                                 $feed_last_updated);
64
65                         $target = "target=\"_blank\"";
66                         $reply .= "<a title=\"$last_updated\" $target href=\"$feed_site_url\">".
67                                 truncate_string($feed_title, 30)."</a>";
68
69                         if ($error) {
70                                 $error = htmlspecialchars($error);
71                                 $reply .= "&nbsp;<img title=\"$error\" src='images/error.png' alt='error' class=\"noborder\">";
72                         }
73
74                 } else {
75                         $reply .= $feed_title;
76                 }
77
78                 $reply .= "</span>";
79
80                 $reply .= "</span>";
81
82 #               $reply .= "</span>";
83
84                 // left part
85
86                 $reply .= "<span class=\"main\">";
87                 $reply .= "<span id='selected_prompt'></span>";
88
89                 /*$reply .= "<span class=\"sel_links\">
90                         <a href=\"#\" onclick=\"$sel_all_link\">".__('All')."</a>,
91                         <a href=\"#\" onclick=\"$sel_unread_link\">".__('Unread')."</a>,
92                         <a href=\"#\" onclick=\"$sel_inv_link\">".__('Invert')."</a>,
93                         <a href=\"#\" onclick=\"$sel_none_link\">".__('None')."</a></li>";
94
95                 $reply .= "</span> "; */
96
97                 $reply .= "<select dojoType=\"dijit.form.Select\"
98                         onchange=\"headlineActionsChange(this)\">";
99
100                 $reply .= "<option value=\"0\" disabled='1'>".__('Select...')."</option>";
101
102                 $reply .= "<option value=\"$sel_all_link\">".__('All')."</option>";
103                 $reply .= "<option value=\"$sel_unread_link\">".__('Unread')."</option>";
104                 $reply .= "<option value=\"$sel_inv_link\">".__('Invert')."</option>";
105                 $reply .= "<option value=\"$sel_none_link\">".__('None')."</option>";
106
107                 $reply .= "<option value=\"0\" disabled=\"1\">".__('Selection toggle:')."</option>";
108
109                 $reply .= "<option value=\"$tog_unread_link\">".__('Unread')."</option>
110                         <option value=\"$tog_marked_link\">".__('Starred')."</option>
111                         <option value=\"$tog_published_link\">".__('Published')."</option>";
112
113                 $reply .= "<option value=\"0\" disabled=\"1\">".__('Selection:')."</option>";
114
115                 $reply .= "<option value=\"$catchup_sel_link\">".__('Mark as read')."</option>";
116                 $reply .= "<option value=\"$set_score_link\">".__('Set score')."</option>";
117
118                 if ($feed_id != "0") {
119                         $reply .= "<option value=\"$archive_sel_link\">".__('Archive')."</option>";
120                 } else {
121                         $reply .= "<option value=\"$archive_sel_link\">".__('Move back')."</option>";
122                         $reply .= "<option value=\"$delete_sel_link\">".__('Delete')."</option>";
123
124                 }
125
126                 if (PluginHost::getInstance()->get_plugin("mail")) {
127                         $reply .= "<option value=\"emailArticle(false)\">".__('Forward by email').
128                                 "</option>";
129                 }
130
131                 if (PluginHost::getInstance()->get_plugin("mailto")) {
132                         $reply .= "<option value=\"mailtoArticle(false)\">".__('Forward by email').
133                                 "</option>";
134                 }
135
136                 $reply .= "<option value=\"0\" disabled=\"1\">".__('Feed:')."</option>";
137
138                 //$reply .= "<option value=\"catchupPage()\">".__('Mark as read')."</option>";
139
140                 $reply .= "<option value=\"displayDlg('".__("View as RSS")."','generatedFeed', '$feed_id:$is_cat:$rss_link')\">".__('View as RSS')."</option>";
141
142                 $reply .= "</select>";
143
144                 //$reply .= "</h2";
145
146                 foreach (PluginHost::getInstance()->get_hooks(PluginHost::HOOK_HEADLINE_TOOLBAR_BUTTON) as $p) {
147                          $reply .= $p->hook_headline_toolbar_button($feed_id, $is_cat);
148                 }
149
150                 $reply .= "</span></span>";
151
152                 return $reply;
153         }
154
155         private function format_headlines_list($feed, $method, $view_mode, $limit, $cat_view,
156                                         $next_unread_feed, $offset, $vgr_last_feed = false,
157                                         $override_order = false, $include_children = false, $check_first_id = false,
158                                         $skip_first_id_check = false) {
159
160                 $disable_cache = false;
161
162                 $reply = array();
163
164                 $rgba_cache = array();
165
166                 $timing_info = microtime(true);
167
168                 $topmost_article_ids = array();
169
170                 if (!$offset) $offset = 0;
171                 if ($method == "undefined") $method = "";
172
173                 $method_split = explode(":", $method);
174
175                 if ($method == "ForceUpdate" && $feed > 0 && is_numeric($feed)) {
176                         // Update the feed if required with some basic flood control
177
178                         $any_needs_curl = false;
179
180                         if (ini_get("open_basedir")) {
181                                 $pluginhost = PluginHost::getInstance();
182                                 foreach ($pluginhost->get_plugins() as $plugin) {
183                                         $flags = $plugin->flags();
184
185                                         if (isset($flags["needs_curl"]) && $flags["needs_curl"]) {
186                                                 $any_needs_curl = true;
187                                                 break;
188                                         }
189                                 }
190                         }
191
192                         //if ($_REQUEST["debug"]) print "<!-- any_needs_curl: $any_needs_curl -->";
193
194                         if (!$any_needs_curl) {
195
196                                 $result = $this->dbh->query(
197                                                 "SELECT cache_images," . SUBSTRING_FOR_DATE . "(last_updated,1,19) AS last_updated
198                                                 FROM ttrss_feeds WHERE id = '$feed'");
199
200                                 if ($this->dbh->num_rows($result) != 0) {
201                                         $last_updated = strtotime($this->dbh->fetch_result($result, 0, "last_updated"));
202                                         $cache_images = sql_bool_to_bool($this->dbh->fetch_result($result, 0, "cache_images"));
203
204                                         if (!$cache_images && time() - $last_updated > 120) {
205                                                 include "rssfuncs.php";
206                                                 update_rss_feed($feed, true, true);
207                                         } else {
208                                                 $this->dbh->query("UPDATE ttrss_feeds SET last_updated = '1970-01-01', last_update_started = '1970-01-01'
209                                                                 WHERE id = '$feed'");
210                                         }
211                                 }
212                         } else {
213                                 $this->dbh->query("UPDATE ttrss_feeds SET last_updated = '1970-01-01', last_update_started = '1970-01-01'
214                                                                 WHERE id = '$feed'");
215                         }
216                 }
217
218                 if ($method_split[0] == "MarkAllReadGR")  {
219                         catchup_feed($method_split[1], false);
220                 }
221
222                 // FIXME: might break tag display?
223
224                 if (is_numeric($feed) && $feed > 0 && !$cat_view) {
225                         $result = $this->dbh->query(
226                                 "SELECT id FROM ttrss_feeds WHERE id = '$feed' LIMIT 1");
227
228                         if ($this->dbh->num_rows($result) == 0) {
229                                 $reply['content'] = "<div align='center'>".__('Feed not found.')."</div>";
230                         }
231                 }
232
233                 @$search = $this->dbh->escape_string($_REQUEST["query"]);
234                 @$search_language = $this->dbh->escape_string($_REQUEST["search_language"]); // PGSQL only
235
236                 if ($search) {
237                         $disable_cache = true;
238                 }
239
240                 if ($_REQUEST["debug"]) $timing_info = print_checkpoint("H0", $timing_info);
241
242
243                 if (!$cat_view && is_numeric($feed) && $feed < PLUGIN_FEED_BASE_INDEX && $feed > LABEL_BASE_INDEX) {
244                         $handler = PluginHost::getInstance()->get_feed_handler(
245                                 PluginHost::feed_to_pfeed_id($feed));
246
247                         if ($handler) {
248                                 $options = array(
249                                         "limit" => $limit,
250                                         "view_mode" => $view_mode,
251                                         "cat_view" => $cat_view,
252                                         "search" => $search,
253                                         "override_order" => $override_order,
254                                         "offset" => $offset,
255                                         "owner_uid" => $_SESSION["uid"],
256                                         "filter" => false,
257                                         "since_id" => 0,
258                                         "include_children" => $include_children);
259
260                                 $qfh_ret = $handler->get_headlines(PluginHost::feed_to_pfeed_id($feed),
261                                         $options);
262                         }
263
264                 } else {
265                         /*$qfh_ret = queryFeedHeadlines($feed, $limit, $view_mode, $cat_view,
266                                 $search, false, $override_order, $offset, 0,
267                                 false, 0, $include_children, $topid);*/
268
269                         //function queryFeedHeadlines($feed, $limit,
270                         // $view_mode, $cat_view, $search, $search_mode,
271                         // $override_order = false, $offset = 0, $owner_uid = 0, $filter = false, $since_id = 0, $include_children = false,
272                         // $ignore_vfeed_group = false, $override_strategy = false, $override_vfeed = false, $start_ts = false, $check_top_id = false) {
273
274                         $params = array(
275                                 "feed" => $feed,
276                                 "limit" => $limit,
277                                 "view_mode" => $view_mode,
278                                 "cat_view" => $cat_view,
279                                 "search" => $search,
280                                 "search_language" => $search_language,
281                                 "override_order" => $override_order,
282                                 "offset" => $offset,
283                                 "include_children" => $include_children,
284                                 "check_first_id" => $check_first_id,
285                                 "skip_first_id_check" => $skip_first_id_check
286                         );
287
288                         $qfh_ret = queryFeedHeadlines($params);
289                 }
290
291                 $vfeed_group_enabled = get_pref("VFEED_GROUP_BY_FEED") && $feed != -6;
292
293                 if ($_REQUEST["debug"]) $timing_info = print_checkpoint("H1", $timing_info);
294
295                 $result = $qfh_ret[0];
296                 $feed_title = $qfh_ret[1];
297                 $feed_site_url = $qfh_ret[2];
298                 $last_error = $qfh_ret[3];
299                 $last_updated = strpos($qfh_ret[4], '1970-') === FALSE ?
300                         make_local_datetime($qfh_ret[4], false) : __("Never");
301                 $highlight_words = $qfh_ret[5];
302                 $reply['first_id'] = $qfh_ret[6];
303
304                 $vgroup_last_feed = $vgr_last_feed;
305
306                 $reply['toolbar'] = $this->format_headline_subtoolbar($feed_site_url,
307                         $feed_title,
308                         $feed, $cat_view, $search, $view_mode,
309                         $last_error, $last_updated);
310
311                 $headlines_count = is_numeric($result) ? 0 : $this->dbh->num_rows($result);
312
313                 if ($offset == 0) {
314                         foreach (PluginHost::getInstance()->get_hooks(PluginHost::HOOK_HEADLINES_BEFORE) as $p) {
315                                  $reply['content'] .= $p->hook_headlines_before($feed, $cat_view, $qfh_ret);
316                         }
317                 }
318
319                 $reply['content'] = '';
320
321                 if ($headlines_count > 0) {
322
323                         $lnum = $offset;
324
325                         $num_unread = 0;
326                         $cur_feed_title = '';
327
328                         if ($_REQUEST["debug"]) $timing_info = print_checkpoint("PS", $timing_info);
329
330                         $expand_cdm = get_pref('CDM_EXPANDED');
331
332                         while ($line = $this->dbh->fetch_assoc($result)) {
333
334                                 $line["content_preview"] =  "&mdash; " . truncate_string(strip_tags($line["content"]), 250);
335
336                                 foreach (PluginHost::getInstance()->get_hooks(PluginHost::HOOK_QUERY_HEADLINES) as $p) {
337                                         $line = $p->hook_query_headlines($line, 250, false);
338                                 }
339
340                                 if (get_pref('SHOW_CONTENT_PREVIEW')) {
341                                         $content_preview =  $line["content_preview"];
342                                 }
343
344                                 $id = $line["id"];
345                                 $feed_id = $line["feed_id"];
346                                 $label_cache = $line["label_cache"];
347                                 $labels = false;
348
349                                 if ($label_cache) {
350                                         $label_cache = json_decode($label_cache, true);
351
352                                         if ($label_cache) {
353                                                 if ($label_cache["no-labels"] == 1)
354                                                         $labels = array();
355                                                 else
356                                                         $labels = $label_cache;
357                                         }
358                                 }
359
360                                 if (!is_array($labels)) $labels = get_article_labels($id);
361
362                                 $labels_str = "<span class=\"HLLCTR-$id\">";
363                                 $labels_str .= format_article_labels($labels, $id);
364                                 $labels_str .= "</span>";
365
366                                 if (count($topmost_article_ids) < 3) {
367                                         array_push($topmost_article_ids, $id);
368                                 }
369
370                                 $class = "";
371
372                                 if (sql_bool_to_bool($line["unread"])) {
373                                         $class .= " Unread";
374                                         ++$num_unread;
375                                 }
376
377                                 if (sql_bool_to_bool($line["marked"])) {
378                                         $marked_pic = "<img
379                                                 src=\"images/mark_set.png\"
380                                                 class=\"markedPic\" alt=\"Unstar article\"
381                                                 onclick='toggleMark($id)'>";
382                                         $class .= " marked";
383                                 } else {
384                                         $marked_pic = "<img
385                                                 src=\"images/mark_unset.png\"
386                                                 class=\"markedPic\" alt=\"Star article\"
387                                                 onclick='toggleMark($id)'>";
388                                 }
389
390                                 if (sql_bool_to_bool($line["published"])) {
391                                         $published_pic = "<img src=\"images/pub_set.png\"
392                                                 class=\"pubPic\"
393                                                         alt=\"Unpublish article\" onclick='togglePub($id)'>";
394                                         $class .= " published";
395                                 } else {
396                                         $published_pic = "<img src=\"images/pub_unset.png\"
397                                                 class=\"pubPic\"
398                                                 alt=\"Publish article\" onclick='togglePub($id)'>";
399                                 }
400
401 #                               $content_link = "<a target=\"_blank\" rel=\"noopener noreferrer\" href=\"".$line["link"]."\">" .
402 #                                       $line["title"] . "</a>";
403
404 #                               $content_link = "<a
405 #                                       href=\"" . htmlspecialchars($line["link"]) . "\"
406 #                                       onclick=\"view($id,$feed_id);\">" .
407 #                                       $line["title"] . "</a>";
408
409 #                               $content_link = "<a href=\"javascript:viewContentUrl('".$line["link"]."');\">" .
410 #                                       $line["title"] . "</a>";
411
412                                 $updated_fmt = make_local_datetime($line["updated"], false, false, false, true);
413                                 $date_entered_fmt = T_sprintf("Imported at %s",
414                                         make_local_datetime($line["date_entered"], false));
415
416                                 $score = $line["score"];
417
418                                 $score_pic = "images/" . get_score_pic($score);
419
420 /*                              $score_title = __("(Click to change)");
421                                 $score_pic = "<img class='hlScorePic' src=\"images/$score_pic\"
422                                         onclick=\"adjustArticleScore($id, $score)\" title=\"$score $score_title\">"; */
423
424                                 $score_pic = "<img class='hlScorePic' score='$score' onclick='changeScore($id, this)' src=\"$score_pic\"
425                                         title=\"$score\">";
426
427                                 if ($score > 500) {
428                                         $hlc_suffix = "high";
429                                 } else if ($score < -100) {
430                                         $hlc_suffix = "low";
431                                 } else {
432                                         $hlc_suffix = "";
433                                 }
434
435                                 $entry_author = $line["author"];
436
437                                 if ($entry_author) {
438                                         $entry_author = " &mdash; $entry_author";
439                                 }
440
441                                 $has_feed_icon = feed_has_icon($feed_id);
442
443                                 if ($has_feed_icon) {
444                                         $feed_icon_img = "<img class=\"tinyFeedIcon\" src=\"".ICONS_URL."/$feed_id.ico\" alt=\"\">";
445                                 } else {
446                                         $feed_icon_img = "<img class=\"tinyFeedIcon\" src=\"images/pub_set.png\" alt=\"\">";
447                                 }
448
449                                 $entry_site_url = $line["site_url"];
450
451                                 //setting feed headline background color, needs to change text color based on dark/light
452                                 $fav_color = $line['favicon_avg_color'];
453
454                                 require_once "colors.php";
455
456                                 if ($fav_color && $fav_color != 'fail') {
457                                         if (!isset($rgba_cache[$feed_id])) {
458                                                 $rgba_cache[$feed_id] = join(",", _color_unpack($fav_color));
459                                         }
460                                 }
461
462                                 if (!get_pref('COMBINED_DISPLAY_MODE')) {
463
464                                         if ($vfeed_group_enabled) {
465                                                 if ($feed_id != $vgroup_last_feed && $line["feed_title"]) {
466
467                                                         $cur_feed_title = $line["feed_title"];
468                                                         $vgroup_last_feed = $feed_id;
469
470                                                         $cur_feed_title = htmlspecialchars($cur_feed_title);
471
472                                                         $vf_catchup_link = "<a class='catchup' onclick='catchupFeedInGroup($feed_id);' href='#'>".__('mark feed as read')."</a>";
473
474                                                         $reply['content'] .= "<div data-feed-id='$feed_id' id='FTITLE-$feed_id' class='cdmFeedTitle'>".
475                                                                 "<div style='float : right'>$feed_icon_img</div>".
476                                                                 "<a class='title' href=\"#\" onclick=\"viewfeed({feed:$feed_id})\">".
477                                                                 $line["feed_title"]."</a>
478                                                                 $vf_catchup_link</div>";
479
480
481                                                 }
482                                         }
483
484                                         $mouseover_attrs = "onmouseover='postMouseIn(event, $id)'
485                                                 onmouseout='postMouseOut($id)'";
486
487                                         $reply['content'] .= "<div class='hl $class' data-orig-feed-id='$feed_id' data-article-id='$id' id='RROW-$id' $mouseover_attrs>";
488
489                                         $reply['content'] .= "<div class='hlLeft'>";
490
491                                         $reply['content'] .= "<input dojoType=\"dijit.form.CheckBox\"
492                                                         type=\"checkbox\" onclick=\"toggleSelectRow2(this)\"
493                                                         class='rchk'>";
494
495                                         $reply['content'] .= "$marked_pic";
496                                         $reply['content'] .= "$published_pic";
497
498                                         $reply['content'] .= "</div>";
499
500                                         $reply['content'] .= "<div onclick='return hlClicked(event, $id)'
501                                                 class=\"hlTitle\"><span class='hlContent $hlc_suffix'>";
502                                         $reply['content'] .= "<a id=\"RTITLE-$id\" class=\"title $hlc_suffix\"
503                                                 href=\"" . htmlspecialchars($line["link"]) . "\"
504                                                 onclick=\"\">" .
505                                                 truncate_string($line["title"], 200);
506
507                                         if (get_pref('SHOW_CONTENT_PREVIEW')) {
508                                                         $reply['content'] .= "<span class=\"contentPreview\">" . $line["content_preview"] . "</span>";
509                                         }
510
511                                         $reply['content'] .= "</a></span>";
512
513                                         $reply['content'] .= $labels_str;
514
515                                         $reply['content'] .= "</div>";
516
517                                         if (!$vfeed_group_enabled) {
518                                                 if (@$line["feed_title"]) {
519                                                         $rgba = @$rgba_cache[$feed_id];
520
521                                                         $reply['content'] .= "<span class=\"hlFeed\"><a style=\"background : rgba($rgba, 0.3)\" href=\"#\" onclick=\"viewfeed({feed:$feed_id})\">".
522                                                                 truncate_string($line["feed_title"],30)."</a></span>";
523                                                 }
524                                         }
525
526
527                                         $reply['content'] .= "<span class=\"hlUpdated\">";
528
529                                         $reply['content'] .= "<div title='$date_entered_fmt'>$updated_fmt</div>
530                                                 </span>";
531
532                                         $reply['content'] .= "<div class=\"hlRight\">";
533
534                                         $reply['content'] .= $score_pic;
535
536                                         if ($line["feed_title"] && !$vfeed_group_enabled) {
537
538                                                 $reply['content'] .= "<span onclick=\"viewfeed({feed:$feed_id})\"
539                                                         style=\"cursor : pointer\"
540                                                         title=\"".htmlspecialchars($line['feed_title'])."\">
541                                                         $feed_icon_img</span>";
542                                         }
543
544                                         $reply['content'] .= "</div>";
545                                         $reply['content'] .= "</div>";
546
547                                 } else {
548
549                                         if ($line["tag_cache"])
550                                                 $tags = explode(",", $line["tag_cache"]);
551                                         else
552                                                 $tags = false;
553
554                                         $line["content"] = sanitize($line["content"],
555                                                         sql_bool_to_bool($line['hide_images']), false, $entry_site_url, $highlight_words, $line["id"]);
556
557                                         foreach (PluginHost::getInstance()->get_hooks(PluginHost::HOOK_RENDER_ARTICLE_CDM) as $p) {
558                                                 $line = $p->hook_render_article_cdm($line);
559                                         }
560
561                                         if ($vfeed_group_enabled && $line["feed_title"]) {
562                                                 if ($feed_id != $vgroup_last_feed) {
563
564                                                         $cur_feed_title = $line["feed_title"];
565                                                         $vgroup_last_feed = $feed_id;
566
567                                                         $cur_feed_title = htmlspecialchars($cur_feed_title);
568
569                                                         $vf_catchup_link = "<a class='catchup' onclick='catchupFeedInGroup($feed_id);' href='#'>".__('mark feed as read')."</a>";
570
571                                                         $has_feed_icon = feed_has_icon($feed_id);
572
573                                                         if ($has_feed_icon) {
574                                                                 $feed_icon_img = "<img class=\"tinyFeedIcon\" src=\"".ICONS_URL."/$feed_id.ico\" alt=\"\">";
575                                                         } else {
576                                                                 //$feed_icon_img = "<img class=\"tinyFeedIcon\" src=\"images/blank_icon.gif\" alt=\"\">";
577                                                         }
578
579                                                         $reply['content'] .= "<div data-feed-id='$feed_id' id='FTITLE-$feed_id' class='cdmFeedTitle'>".
580                                                                 "<div style=\"float : right\">$feed_icon_img</div>".
581                                                                 "<a href=\"#\" class='title' onclick=\"viewfeed({feed:$feed_id})\">".
582                                                                 $line["feed_title"]."</a> $vf_catchup_link</div>";
583
584                                                 }
585                                         }
586
587                                         $mouseover_attrs = "onmouseover='postMouseIn(event, $id)'
588                                                 onmouseout='postMouseOut($id)'";
589
590                                         $expanded_class = $expand_cdm ? "expanded" : "expandable";
591
592                                         $reply['content'] .= "<div class=\"cdm $hlc_suffix $expanded_class $class\"
593                                                 id=\"RROW-$id\" data-article-id='$id' data-orig-feed-id='$feed_id' $mouseover_attrs>";
594
595                                         $reply['content'] .= "<div class=\"cdmHeader\">";
596                                         $reply['content'] .= "<div style=\"vertical-align : middle\">";
597
598                                         $reply['content'] .= "<input dojoType=\"dijit.form.CheckBox\"
599                                                         type=\"checkbox\" onclick=\"toggleSelectRow2(this, false, true)\"
600                                                         class='rchk'>";
601
602                                         $reply['content'] .= "$marked_pic";
603                                         $reply['content'] .= "$published_pic";
604
605                                         $reply['content'] .= "</div>";
606
607                                         if ($highlight_words && count($highlight_words > 0)) {
608                                                 foreach ($highlight_words as $word) {
609                                                         $line["title"] = preg_replace("/(\Q$word\E)/i",
610                                                                 "<span class=\"highlight\">$1</span>", $line["title"]);
611                                                 }
612                                         }
613
614                                         $reply['content'] .= "<span id=\"RTITLE-$id\"
615                                                 onclick=\"return cdmClicked(event, $id);\"
616                                                 class=\"titleWrap $hlc_suffix\">
617                                                 <a class=\"title $hlc_suffix\"
618                                                 title=\"".htmlspecialchars($line["title"])."\"
619                                                 target=\"_blank\" rel=\"noopener noreferrer\" href=\"".
620                                                 htmlspecialchars($line["link"])."\">".
621                                                 $line["title"] .
622                                                 "</a> <span class=\"author\">$entry_author</span>";
623
624                                         $reply['content'] .= $labels_str;
625
626                                         $reply['content'] .= "<span class='collapseBtn' style='display : none'>
627                                                 <img src=\"images/collapse.png\" onclick=\"cdmCollapseArticle(event, $id)\"
628                                                 title=\"".__("Collapse article")."\"/></span>";
629
630                                         if (!$expand_cdm)
631                                                 $content_hidden = "style=\"display : none\"";
632                                         else
633                                                 $excerpt_hidden = "style=\"display : none\"";
634
635                                         $reply['content'] .= "<span $excerpt_hidden id=\"CEXC-$id\" class=\"cdmExcerpt\">" . $content_preview . "</span>";
636
637                                         $reply['content'] .= "</span>";
638
639                                         if (!$vfeed_group_enabled) {
640                                                 if (@$line["feed_title"]) {
641                                                         $rgba = @$rgba_cache[$feed_id];
642
643                                                         $reply['content'] .= "<div class=\"hlFeed\">
644                                                                 <a href=\"#\" style=\"background-color: rgba($rgba,0.3)\"
645                                                                 onclick=\"viewfeed({feed:$feed_id})\">".
646                                                                 truncate_string($line["feed_title"],30)."</a>
647                                                         </div>";
648                                                 }
649                                         }
650
651                                         $reply['content'] .= "<span class='updated' title='$date_entered_fmt'>
652                                                 $updated_fmt</span>";
653
654                                         $reply['content'] .= "<div class='scoreWrap' style=\"vertical-align : middle\">";
655                                         $reply['content'] .= "$score_pic";
656
657                                         if (!get_pref("VFEED_GROUP_BY_FEED") && $line["feed_title"]) {
658                                                 $reply['content'] .= "<span style=\"cursor : pointer\"
659                                                         title=\"".htmlspecialchars($line["feed_title"])."\"
660                                                         onclick=\"viewfeed({feed:$feed_id})\">$feed_icon_img</span>";
661                                         }
662                                         $reply['content'] .= "</div>";
663
664                                         $reply['content'] .= "</div>";
665
666                                         $reply['content'] .= "<div class=\"cdmContent\" $content_hidden
667                                                 onclick=\"return cdmClicked(event, $id);\"
668                                                 id=\"CICD-$id\">";
669
670                                         $reply['content'] .= "<div id=\"POSTNOTE-$id\">";
671                                         if ($line['note']) {
672                                                 $reply['content'] .= format_article_note($id, $line['note']);
673                                         }
674                                         $reply['content'] .= "</div>";
675
676                                         if (!$line['lang']) $line['lang'] = 'en';
677
678                                         $reply['content'] .= "<div class=\"cdmContentInner\" lang=\"".$line['lang']."\">";
679
680                         if ($line["orig_feed_id"]) {
681
682                                 $tmp_result = $this->dbh->query("SELECT * FROM ttrss_archived_feeds
683                                         WHERE id = ".$line["orig_feed_id"] . " AND owner_uid = " . $_SESSION["uid"]);
684
685                                                 if ($this->dbh->num_rows($tmp_result) != 0) {
686
687                                                         $reply['content'] .= "<div clear='both'>";
688                                                         $reply['content'] .= __("Originally from:");
689
690                                                         $reply['content'] .= "&nbsp;";
691
692                                                         $tmp_line = $this->dbh->fetch_assoc($tmp_result);
693
694                                                         $reply['content'] .= "<a target='_blank' rel='noopener noreferrer'
695                                                                 href=' " . htmlspecialchars($tmp_line['site_url']) . "'>" .
696                                                                 $tmp_line['title'] . "</a>";
697
698                                                         $reply['content'] .= "&nbsp;";
699
700                                                         $reply['content'] .= "<a target='_blank' rel='noopener noreferrer' href='" . htmlspecialchars($tmp_line['feed_url']) . "'>";
701                                                         $reply['content'] .= "<img title='".__('Feed URL')."'class='tinyFeedIcon' src='images/pub_unset.png'></a>";
702
703                                                         $reply['content'] .= "</div>";
704                                                 }
705                                         }
706
707                                         $reply['content'] .= "<span id=\"CWRAP-$id\">";
708
709                                         $reply['content'] .= "<span id=\"CENCW-$id\" class=\"cencw\" style=\"display : none\">";
710                                         $reply['content'] .= htmlspecialchars($line["content"]);
711                                         $reply['content'] .= "</span>";
712
713                                         $reply['content'] .= "</span>";
714
715                                         $reply['content'] .= "</div>";
716
717                                         $reply['content'] .= "<div class=\"cdmIntermediate\">";
718
719                                         $always_display_enclosures = sql_bool_to_bool($line["always_display_enclosures"]);
720                                         $reply['content'] .= format_article_enclosures($id, $always_display_enclosures, $line["content"], sql_bool_to_bool($line["hide_images"]));
721
722                                         $reply['content'] .= "</div>";
723
724                                         $reply['content'] .= "<div class=\"cdmFooter\" onclick=\"cdmFooterClick(event)\">";
725
726                                         foreach (PluginHost::getInstance()->get_hooks(PluginHost::HOOK_ARTICLE_LEFT_BUTTON) as $p) {
727                                                 $reply['content'] .= $p->hook_article_left_button($line);
728                                         }
729
730                                         $tags_str = format_tags_string($tags, $id);
731
732                                         $reply['content'] .= "<span class='left'>";
733
734                                         $reply['content'] .= "<img src='images/tag.png' alt='Tags' title='Tags'>
735                                                 <span id=\"ATSTR-$id\">$tags_str</span>
736                                                 <a title=\"".__('Edit tags for this article')."\"
737                                                 href=\"#\" onclick=\"editArticleTags($id)\">(+)</a>";
738
739                                         $num_comments = (int) $line["num_comments"];
740                                         $entry_comments = "";
741
742                                         if ($num_comments > 0) {
743                                                 if ($line["comments"]) {
744                                                         $comments_url = htmlspecialchars($line["comments"]);
745                                                 } else {
746                                                         $comments_url = htmlspecialchars($line["link"]);
747                                                 }
748                                                 $entry_comments = "<a class=\"postComments\"
749                                                         target='_blank' rel='noopener noreferrer' href=\"$comments_url\">$num_comments ".
750                                                         _ngettext("comment", "comments", $num_comments)."</a>";
751
752                                         } else {
753                                                 if ($line["comments"] && $line["link"] != $line["comments"]) {
754                                                         $entry_comments = "<a class=\"postComments\" target='_blank' rel='noopener noreferrer' href=\"".htmlspecialchars($line["comments"])."\">".__("comments")."</a>";
755                                                 }
756                                         }
757
758                                         if ($entry_comments) $reply['content'] .= "&nbsp;($entry_comments)";
759
760                                         $reply['content'] .= "</span>";
761                                         $reply['content'] .= "<div>";
762
763 //                                      $reply['content'] .= "$marked_pic";
764 //                                      $reply['content'] .= "$published_pic";
765
766                                         foreach (PluginHost::getInstance()->get_hooks(PluginHost::HOOK_ARTICLE_BUTTON) as $p) {
767                                                 $reply['content'] .= $p->hook_article_button($line);
768                                         }
769
770                                         $reply['content'] .= "</div>";
771                                         $reply['content'] .= "</div>";
772
773                                         $reply['content'] .= "</div>";
774
775                                         $reply['content'] .= "</div>";
776
777                                 }
778
779                                 ++$lnum;
780                         }
781
782                         if ($_REQUEST["debug"]) $timing_info = print_checkpoint("PE", $timing_info);
783
784                 } else if (!is_numeric($result)) {
785                         $message = "";
786
787                         switch ($view_mode) {
788                                 case "unread":
789                                         $message = __("No unread articles found to display.");
790                                         break;
791                                 case "updated":
792                                         $message = __("No updated articles found to display.");
793                                         break;
794                                 case "marked":
795                                         $message = __("No starred articles found to display.");
796                                         break;
797                                 default:
798                                         if ($feed < LABEL_BASE_INDEX) {
799                                                 $message = __("No articles found to display. You can assign articles to labels manually from article header context menu (applies to all selected articles) or use a filter.");
800                                         } else {
801                                                 $message = __("No articles found to display.");
802                                         }
803                         }
804
805                         if (!$offset && $message) {
806                                 $reply['content'] = "<div class='whiteBox'>$message";
807
808                                 $reply['content'] .= "<p><span class=\"insensitive\">";
809
810                                 $result = $this->dbh->query("SELECT ".SUBSTRING_FOR_DATE."(MAX(last_updated), 1, 19) AS last_updated FROM ttrss_feeds
811                                         WHERE owner_uid = " . $_SESSION['uid']);
812
813                                 $last_updated = $this->dbh->fetch_result($result, 0, "last_updated");
814                                 $last_updated = make_local_datetime($last_updated, false);
815
816                                 $reply['content'] .= sprintf(__("Feeds last updated at %s"), $last_updated);
817
818                                 $result = $this->dbh->query("SELECT COUNT(id) AS num_errors
819                                         FROM ttrss_feeds WHERE last_error != '' AND owner_uid = ".$_SESSION["uid"]);
820
821                                 $num_errors = $this->dbh->fetch_result($result, 0, "num_errors");
822
823                                 if ($num_errors > 0) {
824                                         $reply['content'] .= "<br/>";
825                                         $reply['content'] .= "<a class=\"insensitive\" href=\"#\" onclick=\"showFeedsWithErrors()\">".
826                                                 __('Some feeds have update errors (click for details)')."</a>";
827                                 }
828                                 $reply['content'] .= "</span></p></div>";
829
830                         }
831                 } else if (is_numeric($result) && $result == -1) {
832                         $reply['first_id_changed'] = true;
833                 }
834
835                 if ($_REQUEST["debug"]) $timing_info = print_checkpoint("H2", $timing_info);
836
837                 return array($topmost_article_ids, $headlines_count, $feed, $disable_cache,
838                         $vgroup_last_feed, $reply);
839         }
840
841         function catchupAll() {
842                 $this->dbh->query("UPDATE ttrss_user_entries SET
843                                                 last_read = NOW(), unread = false WHERE unread = true AND owner_uid = " . $_SESSION["uid"]);
844                 ccache_zero_all($_SESSION["uid"]);
845         }
846
847         function view() {
848                 $timing_info = microtime(true);
849
850                 $reply = array();
851
852                 if ($_REQUEST["debug"]) $timing_info = print_checkpoint("0", $timing_info);
853
854                 $feed = $this->dbh->escape_string($_REQUEST["feed"]);
855                 $method = $this->dbh->escape_string($_REQUEST["m"]);
856                 $view_mode = $this->dbh->escape_string($_REQUEST["view_mode"]);
857                 $limit = 30;
858                 @$cat_view = $_REQUEST["cat"] == "true";
859                 @$next_unread_feed = $this->dbh->escape_string($_REQUEST["nuf"]);
860                 @$offset = $this->dbh->escape_string($_REQUEST["skip"]);
861                 @$vgroup_last_feed = $this->dbh->escape_string($_REQUEST["vgrlf"]);
862                 $order_by = $this->dbh->escape_string($_REQUEST["order_by"]);
863                 $check_first_id = $this->dbh->escape_string($_REQUEST["fid"]);
864
865                 if (is_numeric($feed)) $feed = (int) $feed;
866
867                 /* Feed -5 is a special case: it is used to display auxiliary information
868                  * when there's nothing to load - e.g. no stuff in fresh feed */
869
870                 if ($feed == -5) {
871                         print json_encode($this->generate_dashboard_feed());
872                         return;
873                 }
874
875                 $result = false;
876
877                 if ($feed < LABEL_BASE_INDEX) {
878                         $label_feed = feed_to_label_id($feed);
879                         $result = $this->dbh->query("SELECT id FROM ttrss_labels2 WHERE
880                                                         id = '$label_feed' AND owner_uid = " . $_SESSION['uid']);
881                 } else if (!$cat_view && is_numeric($feed) && $feed > 0) {
882                         $result = $this->dbh->query("SELECT id FROM ttrss_feeds WHERE
883                                                         id = '$feed' AND owner_uid = " . $_SESSION['uid']);
884                 } else if ($cat_view && is_numeric($feed) && $feed > 0) {
885                         $result = $this->dbh->query("SELECT id FROM ttrss_feed_categories WHERE
886                                                         id = '$feed' AND owner_uid = " . $_SESSION['uid']);
887                 }
888
889                 if ($result && $this->dbh->num_rows($result) == 0) {
890                         print json_encode($this->generate_error_feed(__("Feed not found.")));
891                         return;
892                 }
893
894                 /* Updating a label ccache means recalculating all of the caches
895                  * so for performance reasons we don't do that here */
896
897                 if ($feed >= 0) {
898                         ccache_update($feed, $_SESSION["uid"], $cat_view);
899                 }
900
901                 set_pref("_DEFAULT_VIEW_MODE", $view_mode);
902                 set_pref("_DEFAULT_VIEW_ORDER_BY", $order_by);
903
904                 /* bump login timestamp if needed */
905                 if (time() - $_SESSION["last_login_update"] > 3600) {
906                         $this->dbh->query("UPDATE ttrss_users SET last_login = NOW() WHERE id = " .
907                                 $_SESSION["uid"]);
908                         $_SESSION["last_login_update"] = time();
909                 }
910
911                 if (!$cat_view && is_numeric($feed) && $feed > 0) {
912                         $this->dbh->query("UPDATE ttrss_feeds SET last_viewed = NOW()
913                                                         WHERE id = '$feed' AND owner_uid = ".$_SESSION["uid"]);
914                 }
915
916                 $reply['headlines'] = array();
917
918                 $override_order = false;
919                 $skip_first_id_check = false;
920
921                 switch ($order_by) {
922                 case "title":
923                         $override_order = "ttrss_entries.title";
924                         break;
925                 case "date_reverse":
926                         $override_order = "score DESC, date_entered, updated";
927                         $skip_first_id_check = true;
928                         break;
929                 case "feed_dates":
930                         $override_order = "updated DESC";
931                         break;
932                 }
933
934                 if ($_REQUEST["debug"]) $timing_info = print_checkpoint("04", $timing_info);
935
936                 $ret = $this->format_headlines_list($feed, $method,
937                         $view_mode, $limit, $cat_view, $next_unread_feed, $offset,
938                         $vgroup_last_feed, $override_order, true, $check_first_id, $skip_first_id_check);
939
940                 //$topmost_article_ids = $ret[0];
941                 $headlines_count = $ret[1];
942                 /* $returned_feed = $ret[2]; */
943                 $disable_cache = $ret[3];
944                 $vgroup_last_feed = $ret[4];
945
946                 //$reply['headlines']['content'] =& $ret[5]['content'];
947                 //$reply['headlines']['toolbar'] =& $ret[5]['toolbar'];
948
949                 $reply['headlines'] = $ret[5];
950
951                 if (!$next_unread_feed)
952                         $reply['headlines']['id'] = $feed;
953                 else
954                         $reply['headlines']['id'] = $next_unread_feed;
955
956                 $reply['headlines']['is_cat'] = (bool) $cat_view;
957
958                 if ($_REQUEST["debug"]) $timing_info = print_checkpoint("05", $timing_info);
959
960                 $reply['headlines-info'] = array("count" => (int) $headlines_count,
961                                                 "vgroup_last_feed" => $vgroup_last_feed,
962                                                 "disable_cache" => (bool) $disable_cache);
963
964                 if ($_REQUEST["debug"]) $timing_info = print_checkpoint("30", $timing_info);
965
966                 $reply['runtime-info'] = make_runtime_info();
967
968                 print json_encode($reply);
969
970         }
971
972         private function generate_dashboard_feed() {
973                 $reply = array();
974
975                 $reply['headlines']['id'] = -5;
976                 $reply['headlines']['is_cat'] = false;
977
978                 $reply['headlines']['toolbar'] = '';
979
980                 $reply['headlines']['content'] = "<div class='whiteBox'>".__('No feed selected.');
981
982                 $reply['headlines']['content'] .= "<p><span class=\"insensitive\">";
983
984                 $result = $this->dbh->query("SELECT ".SUBSTRING_FOR_DATE."(MAX(last_updated), 1, 19) AS last_updated FROM ttrss_feeds
985                         WHERE owner_uid = " . $_SESSION['uid']);
986
987                 $last_updated = $this->dbh->fetch_result($result, 0, "last_updated");
988                 $last_updated = make_local_datetime($last_updated, false);
989
990                 $reply['headlines']['content'] .= sprintf(__("Feeds last updated at %s"), $last_updated);
991
992                 $result = $this->dbh->query("SELECT COUNT(id) AS num_errors
993                         FROM ttrss_feeds WHERE last_error != '' AND owner_uid = ".$_SESSION["uid"]);
994
995                 $num_errors = $this->dbh->fetch_result($result, 0, "num_errors");
996
997                 if ($num_errors > 0) {
998                         $reply['headlines']['content'] .= "<br/>";
999                         $reply['headlines']['content'] .= "<a class=\"insensitive\" href=\"#\" onclick=\"showFeedsWithErrors()\">".
1000                                 __('Some feeds have update errors (click for details)')."</a>";
1001                 }
1002                 $reply['headlines']['content'] .= "</span></p>";
1003
1004                 $reply['headlines-info'] = array("count" => 0,
1005                         "vgroup_last_feed" => '',
1006                         "unread" => 0,
1007                         "disable_cache" => true);
1008
1009                 return $reply;
1010         }
1011
1012         private function generate_error_feed($error) {
1013                 $reply = array();
1014
1015                 $reply['headlines']['id'] = -7;
1016                 $reply['headlines']['is_cat'] = false;
1017
1018                 $reply['headlines']['toolbar'] = '';
1019                 $reply['headlines']['content'] = "<div class='whiteBox'>". $error . "</div>";
1020
1021                 $reply['headlines-info'] = array("count" => 0,
1022                         "vgroup_last_feed" => '',
1023                         "unread" => 0,
1024                         "disable_cache" => true);
1025
1026                 return $reply;
1027         }
1028
1029         function quickAddFeed() {
1030                 print_hidden("op", "rpc");
1031                 print_hidden("method", "addfeed");
1032
1033                 print "<div id='fadd_error_message' style='display : none' class='alert alert-danger'></div>";
1034
1035                 print "<div id='fadd_multiple_notify' style='display : none'>";
1036                 print_notice("Provided URL is a HTML page referencing multiple feeds, please select required feed from the dropdown menu below.");
1037                 print "<p></div>";
1038
1039                 print "<div class=\"dlgSec\">".__("Feed or site URL")."</div>";
1040                 print "<div class=\"dlgSecCont\">";
1041
1042                 print "<div style='float : right'>
1043                         <img style='display : none'
1044                                 id='feed_add_spinner' src='images/indicator_white.gif'></div>";
1045
1046                 print "<input style=\"font-size : 16px; width : 20em;\"
1047                         placeHolder=\"".__("Feed or site URL")."\"
1048                         dojoType=\"dijit.form.ValidationTextBox\" required=\"1\" name=\"feed\" id=\"feedDlg_feedUrl\">";
1049
1050                 print "<hr/>";
1051
1052                 if (get_pref('ENABLE_FEED_CATS')) {
1053                         print __('Place in category:') . " ";
1054                         print_feed_cat_select("cat", false, 'dojoType="dijit.form.Select"');
1055                 }
1056
1057                 print "</div>";
1058
1059                 print '<div id="feedDlg_feedsContainer" style="display : none">
1060
1061                                 <div class="dlgSec">' . __('Available feeds') . '</div>
1062                                 <div class="dlgSecCont">'.
1063                                 '<select id="feedDlg_feedContainerSelect"
1064                                         dojoType="dijit.form.Select" size="3">
1065                                         <script type="dojo/method" event="onChange" args="value">
1066                                                 dijit.byId("feedDlg_feedUrl").attr("value", value);
1067                                         </script>
1068                                 </select>'.
1069                                 '</div></div>';
1070
1071                 print "<div id='feedDlg_loginContainer' style='display : none'>
1072
1073                                 <div class=\"dlgSec\">".__("Authentication")."</div>
1074                                 <div class=\"dlgSecCont\">".
1075
1076                                 " <input dojoType=\"dijit.form.TextBox\" name='login'\"
1077                                         placeHolder=\"".__("Login")."\"
1078                                         autocomplete=\"new-password\"
1079                                         style=\"width : 10em;\"> ".
1080                                 " <input
1081                                         placeHolder=\"".__("Password")."\"
1082                                         dojoType=\"dijit.form.TextBox\" type='password'
1083                                         autocomplete=\"new-password\"
1084                                         style=\"width : 10em;\" name='pass'\">
1085                         </div></div>";
1086
1087
1088                 print "<div style=\"clear : both\">
1089                         <input type=\"checkbox\" name=\"need_auth\" dojoType=\"dijit.form.CheckBox\" id=\"feedDlg_loginCheck\"
1090                                         onclick='checkboxToggleElement(this, \"feedDlg_loginContainer\")'>
1091                                 <label for=\"feedDlg_loginCheck\">".
1092                                 __('This feed requires authentication.')."</div>";
1093
1094                 print "</form>";
1095
1096                 print "<div class=\"dlgButtons\">
1097                         <button dojoType=\"dijit.form.Button\" onclick=\"return dijit.byId('feedAddDlg').execute()\">".__('Subscribe')."</button>";
1098
1099                 if (!(defined('_DISABLE_FEED_BROWSER') && _DISABLE_FEED_BROWSER)) {
1100                         print "<button dojoType=\"dijit.form.Button\" onclick=\"return feedBrowser()\">".__('More feeds')."</button>";
1101                 }
1102
1103                 print "<button dojoType=\"dijit.form.Button\" onclick=\"return dijit.byId('feedAddDlg').hide()\">".__('Cancel')."</button>
1104                         </div>";
1105
1106                 //return;
1107         }
1108
1109         function feedBrowser() {
1110                 if (defined('_DISABLE_FEED_BROWSER') && _DISABLE_FEED_BROWSER) return;
1111
1112                 $browser_search = $this->dbh->escape_string($_REQUEST["search"]);
1113
1114                 print_hidden("op", "rpc");
1115                 print_hidden("method", "updateFeedBrowser");
1116
1117                 print "<div dojoType=\"dijit.Toolbar\">
1118                         <div style='float : right'>
1119                         <img style='display : none'
1120                                 id='feed_browser_spinner' src='images/indicator_white.gif'>
1121                         <input name=\"search\" dojoType=\"dijit.form.TextBox\" size=\"20\" type=\"search\"
1122                                 onchange=\"dijit.byId('feedBrowserDlg').update()\" value=\"$browser_search\">
1123                         <button dojoType=\"dijit.form.Button\" onclick=\"dijit.byId('feedBrowserDlg').update()\">".__('Search')."</button>
1124                 </div>";
1125
1126                 print " <select name=\"mode\" dojoType=\"dijit.form.Select\" onchange=\"dijit.byId('feedBrowserDlg').update()\">
1127                         <option value='1'>" . __('Popular feeds') . "</option>
1128                         <option value='2'>" . __('Feed archive') . "</option>
1129                         </select> ";
1130
1131                 print __("limit:");
1132
1133                 print " <select dojoType=\"dijit.form.Select\" name=\"limit\" onchange=\"dijit.byId('feedBrowserDlg').update()\">";
1134
1135                 foreach (array(25, 50, 100, 200) as $l) {
1136                         //$issel = ($l == $limit) ? "selected=\"1\"" : "";
1137                         print "<option value=\"$l\">$l</option>";
1138                 }
1139
1140                 print "</select> ";
1141
1142                 print "</div>";
1143
1144                 require_once "feedbrowser.php";
1145
1146                 print "<ul class='browseFeedList' id='browseFeedList'>";
1147                 print make_feed_browser("", 25);
1148                 print "</ul>";
1149
1150                 print "<div align='center'>
1151                         <button dojoType=\"dijit.form.Button\" onclick=\"dijit.byId('feedBrowserDlg').execute()\">".__('Subscribe')."</button>
1152                         <button dojoType=\"dijit.form.Button\" style='display : none' id='feed_archive_remove' onclick=\"dijit.byId('feedBrowserDlg').removeFromArchive()\">".__('Remove')."</button>
1153                         <button dojoType=\"dijit.form.Button\" onclick=\"dijit.byId('feedBrowserDlg').hide()\" >".__('Cancel')."</button></div>";
1154
1155         }
1156
1157         function search() {
1158                 $this->params = explode(":", $this->dbh->escape_string($_REQUEST["param"]), 2);
1159
1160                 $active_feed_id = sprintf("%d", $this->params[0]);
1161                 $is_cat = $this->params[1] != "false";
1162
1163                 print "<div class=\"dlgSec\">".__('Look for')."</div>";
1164
1165                 print "<div class=\"dlgSecCont\">";
1166
1167                 print "<input dojoType=\"dijit.form.ValidationTextBox\"
1168                         style=\"font-size : 16px; width : 20em;\"
1169                         required=\"1\" name=\"query\" type=\"search\" value=''>";
1170
1171                 print "<hr/><span style='float : right'>".T_sprintf('in %s', getFeedTitle($active_feed_id, $is_cat))."</span>";
1172
1173                 if (DB_TYPE == "pgsql") {
1174                         print "<hr/>";
1175                         print_select("search_language", "", Pref_Feeds::$feed_languages,
1176                                 "dojoType='dijit.form.Select' title=\"".__('Used for word stemming')."\"");
1177                 }
1178
1179                 print "</div>";
1180
1181                 print "<div class=\"dlgButtons\">";
1182
1183                 if (count(PluginHost::getInstance()->get_hooks(PluginHost::HOOK_SEARCH)) == 0) {
1184                         print "<div style=\"float : left\">
1185                                 <a class=\"visibleLink\" target=\"_blank\" href=\"http://tt-rss.org/wiki/SearchSyntax\">".__("Search syntax")."</a>
1186                                 </div>";
1187                 }
1188
1189                 print "<button dojoType=\"dijit.form.Button\" onclick=\"dijit.byId('searchDlg').execute()\">".__('Search')."</button>
1190                 <button dojoType=\"dijit.form.Button\" onclick=\"dijit.byId('searchDlg').hide()\">".__('Cancel')."</button>
1191                 </div>";
1192         }
1193
1194         function update_debugger() {
1195                 header("Content-type: text/html");
1196
1197                 $feed_id = (int)$_REQUEST["feed_id"];
1198                 @$do_update = $_REQUEST["action"] == "do_update";
1199                 $csrf_token = $_REQUEST["csrf_token"];
1200
1201                 $refetch_checked = isset($_REQUEST["force_refetch"]) ? "checked" : "";
1202                 $rehash_checked = isset($_REQUEST["force_rehash"]) ? "checked" : "";
1203
1204                 ?>
1205                 <html>
1206                 <head>
1207                         <link rel="stylesheet" type="text/css" href="css/utility.css">
1208                         <title>Feed Debugger</title>
1209                 </head>
1210                 <body class="small_margins">
1211                 <h1>Feed Debugger: <?php echo "$feed_id: " . getFeedTitle($feed_id) ?></h1>
1212                 <form method="GET" action="">
1213                         <input type="hidden" name="op" value="feeds">
1214                         <input type="hidden" name="method" value="update_debugger">
1215                         <input type="hidden" name="xdebug" value="1">
1216                         <input type="hidden" name="csrf_token" value="<?php echo $csrf_token ?>">
1217                         <input type="hidden" name="action" value="do_update">
1218                         <input type="hidden" name="feed_id" value="<?php echo $feed_id ?>">
1219                         <input type="checkbox" name="force_refetch" value="1" <?php echo $refetch_checked ?>> Force refetch<br/>
1220                         <input type="checkbox" name="force_rehash" value="1" <?php echo $rehash_checked ?>> Force rehash<br/>
1221
1222                         <p/><button type="submit">Continue</button>
1223                 </form>
1224
1225                 <hr>
1226
1227                 <pre><?php
1228
1229                 if ($do_update) {
1230                         include "rssfuncs.php";
1231                         update_rss_feed($feed_id, true, true);
1232                 }
1233
1234                 ?></pre>
1235
1236                 </body>
1237                 </html>
1238                 <?php
1239
1240         }
1241 }
1242 ?>