qvitter.php 48 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887
  1. <?php
  2. /* · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · ·
  3. · ·
  4. · ·
  5. · Q V I T T E R ·
  6. · ·
  7. · https://git.gnu.io/h2p/Qvitter ·
  8. · ·
  9. · ·
  10. · <o) ·
  11. · /_//// ·
  12. · (____/ ·
  13. · (o< ·
  14. · o> \\\\_\ ·
  15. · \\) \____) ·
  16. · ·
  17. · ·
  18. · ·
  19. · Qvitter is free software: you can redistribute it and / or modify it ·
  20. · under the terms of the GNU Affero General Public License as published by ·
  21. · the Free Software Foundation, either version three of the License or (at ·
  22. · your option) any later version. ·
  23. · ·
  24. · Qvitter is distributed in hope that it will be useful but WITHOUT ANY ·
  25. · WARRANTY; without even the implied warranty of MERCHANTABILTY or FITNESS ·
  26. · FOR A PARTICULAR PURPOSE. See the GNU Affero General Public License for ·
  27. · more details. ·
  28. · ·
  29. · You should have received a copy of the GNU Affero General Public License ·
  30. · along with Qvitter. If not, see <http://www.gnu.org/licenses/>. ·
  31. · ·
  32. · Contact h@nnesmannerhe.im if you have any questions. ·
  33. · ·
  34. · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · · */
  35. class QvitterAction extends ApiAction
  36. {
  37. function isReadOnly($args)
  38. {
  39. return true;
  40. }
  41. protected function prepare(array $args=array())
  42. {
  43. parent::prepare($args);
  44. // if we're logged in but we have missing or incorrect csrf cookie, logout
  45. if(common_logged_in()) {
  46. $csrf_token = sha1(common_config('qvitter', 'appid').session_id());
  47. if(!isset($_COOKIE['Qvitter-CSRF']) || $_COOKIE['Qvitter-CSRF'] != $csrf_token) {
  48. header('Location: '.common_path('').'main/logout');
  49. die();
  50. }
  51. }
  52. $user = common_current_user();
  53. return true;
  54. }
  55. protected function handle()
  56. {
  57. parent::handle();
  58. $this->showQvitter();
  59. }
  60. function showQvitter()
  61. {
  62. $logged_in_user_nickname = '';
  63. $logged_in_user_obj = false;
  64. $logged_in_user = common_current_user();
  65. if($logged_in_user) {
  66. $logged_in_user_nickname = $logged_in_user->nickname;
  67. $logged_in_user_obj = ApiAction::twitterUserArray($logged_in_user->getProfile());
  68. }
  69. $registrationsclosed = false;
  70. if(common_config('site','closed') == 1 || common_config('site','inviteonly') == 1) {
  71. $registrationsclosed = true;
  72. }
  73. // check if the client's ip address is blocked for registration
  74. if(is_array(QvitterPlugin::settings("blocked_ips"))) {
  75. $client_ip_is_blocked = in_array($_SERVER['REMOTE_ADDR'], QvitterPlugin::settings("blocked_ips"));
  76. }
  77. $sitetitle = common_config('site','name');
  78. $siterootdomain = common_config('site','server');
  79. $qvitterpath = Plugin::staticPath('Qvitter', '');
  80. $apiroot = common_path('api/', StatusNet::isHTTPS());
  81. $attachmentroot = common_path('attachment/', StatusNet::isHTTPS());
  82. $instanceurl = common_path('', StatusNet::isHTTPS());
  83. $favicon_path = QvitterPlugin::settings("favicon_path");
  84. // user's browser's language setting
  85. $user_browser_language = 'en'; // use english if we can't find the browser language
  86. if(isset($_SERVER['HTTP_ACCEPT_LANGUAGE'])) {
  87. $user_browser_language = substr($_SERVER['HTTP_ACCEPT_LANGUAGE'], 0, 2);
  88. }
  89. common_set_returnto(''); // forget this
  90. // if this is a profile we add a link header for LRDD Discovery (see WebfingerPlugin.php)
  91. if(substr_count($_SERVER['REQUEST_URI'], '/') == 1) {
  92. $nickname = substr($_SERVER['REQUEST_URI'],1);
  93. if(preg_match("/^[a-zA-Z0-9]+$/", $nickname) == 1) {
  94. $acct = 'acct:'. $nickname .'@'. common_config('site', 'server');
  95. $url = common_local_url('webfinger') . '?resource='.$acct;
  96. foreach (array(Discovery::JRD_MIMETYPE, Discovery::XRD_MIMETYPE) as $type) {
  97. header('Link: <'.$url.'>; rel="'. Discovery::LRDD_REL.'"; type="'.$type.'"');
  98. }
  99. }
  100. }
  101. ?><!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN"
  102. "http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd">
  103. <html xmlns="http://www.w3.org/1999/xhtml">
  104. <head>
  105. <title><?php print $sitetitle; ?></title>
  106. <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
  107. <meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=0">
  108. <link rel="stylesheet" type="text/css" href="<?php print $qvitterpath; ?>css/qvitter.css?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/css/qvitter.css')); ?>" />
  109. <link rel="stylesheet" type="text/css" href="<?php print $qvitterpath; ?>css/jquery.minicolors.css" />
  110. <link rel="apple-touch-icon" sizes="57x57" href="<?php print $favicon_path ?>apple-touch-icon-57x57.png">
  111. <link rel="apple-touch-icon" sizes="60x60" href="<?php print $favicon_path ?>apple-touch-icon-60x60.png">
  112. <link rel="apple-touch-icon" sizes="72x72" href="<?php print $favicon_path ?>apple-touch-icon-72x72.png">
  113. <link rel="apple-touch-icon" sizes="76x76" href="<?php print $favicon_path ?>apple-touch-icon-76x76.png">
  114. <link rel="apple-touch-icon" sizes="114x114" href="<?php print $favicon_path ?>apple-touch-icon-114x114.png">
  115. <link rel="apple-touch-icon" sizes="120x120" href="<?php print $favicon_path ?>apple-touch-icon-120x120.png">
  116. <link rel="apple-touch-icon" sizes="144x144" href="<?php print $favicon_path ?>apple-touch-icon-144x144.png">
  117. <link rel="apple-touch-icon" sizes="152x152" href="<?php print $favicon_path ?>apple-touch-icon-152x152.png">
  118. <link rel="apple-touch-icon" sizes="180x180" href="<?php print $favicon_path ?>apple-touch-icon-180x180.png">
  119. <link rel="icon" type="image/png" href="<?php print $favicon_path ?>favicon-16x16.png" sizes="16x16">
  120. <link rel="icon" type="image/png" href="<?php print $favicon_path ?>favicon-32x32.png" sizes="32x32">
  121. <link rel="icon" type="image/png" href="<?php print $favicon_path ?>android-chrome-192x192.png" sizes="192x192">
  122. <link rel="icon" type="image/png" href="<?php print $favicon_path ?>favicon-96x96.png" sizes="96x96">
  123. <link rel="manifest" href="<?php print $favicon_path ?>manifest.json">
  124. <link rel="mask-icon" href="<?php print $favicon_path ?>safari-pinned-tab.svg" color="#a22430">
  125. <meta name="apple-mobile-web-app-title" content="<?php print $sitetitle; ?>">
  126. <meta name="application-name" content="<?php print $sitetitle; ?>">
  127. <meta name="msapplication-TileColor" content="#da532c">
  128. <meta name="msapplication-TileImage" content="<?php print $favicon_path ?>mstile-144x144.png">
  129. <meta name="theme-color" content="#ffffff">
  130. <?php
  131. // if qvitter is a webapp and this is a users url we add feeds
  132. if(substr_count($_SERVER['REQUEST_URI'], '/') == 1) {
  133. $nickname = substr($_SERVER['REQUEST_URI'],1);
  134. if(preg_match("/^[a-zA-Z0-9]+$/", $nickname) == 1) {
  135. $user = User::getKV('nickname', $nickname);
  136. if(!isset($user->id)) {
  137. //error_log("QVITTER: Could not get user id for user with nickname: $nickname – REQUEST_URI: ".$_SERVER['REQUEST_URI']);
  138. }
  139. else {
  140. print '<link title="Notice feed for '.$nickname.' (Activity Streams JSON)" type="application/stream+json" href="'.$instanceurl.'api/statuses/user_timeline/'.$user->id.'.as" rel="alternate">'."\n";
  141. print ' <link title="Notice feed for '.$nickname.' (RSS 1.0)" type="application/rdf+xml" href="'.$instanceurl.$nickname.'/rss" rel="alternate">'."\n";
  142. print ' <link title="Notice feed for '.$nickname.' (RSS 2.0)" type="application/rss+xml" href="'.$instanceurl.'api/statuses/user_timeline/'.$user->id.'.rss" rel="alternate">'."\n";
  143. print ' <link title="Notice feed for '.$nickname.' (Atom)" type="application/atom+xml" href="'.$instanceurl.'api/statuses/user_timeline/'.$user->id.'.atom" rel="alternate">'."\n";
  144. print ' <link title="FOAF for '.$nickname.'" type="application/rdf+xml" href="'.$instanceurl.$nickname.'/foaf" rel="meta">'."\n";
  145. print ' <link href="'.$instanceurl.$nickname.'/microsummary" rel="microsummary">'."\n";
  146. // rel="me" for the IndieWeb audience
  147. // (no indieweb for users of older gnu social versions)
  148. if(method_exists('Profile','getHomepage')) {
  149. $user_homepage = $user->getProfile()->getHomepage();
  150. $relMes = array(
  151. ['href' => $user->getProfile()->getHomepage(),
  152. 'text' => _('Homepage'),
  153. 'image' => null],
  154. );
  155. Event::handle('OtherAccountProfiles', array($user->getProfile(), &$relMes));
  156. foreach ($relMes as $relMe) {
  157. print ' <link href="'.htmlspecialchars($relMe['href']).'" title="'.$relMe['text'].'" rel="me" />'."\n";
  158. }
  159. }
  160. // maybe openid
  161. if (array_key_exists('OpenID', StatusNet::getActivePlugins())) {
  162. print ' <link rel="openid2.provider" href="'.common_local_url('openidserver').'"/>'."\n";
  163. print ' <link rel="openid2.local_id" href="'.$user->getProfile()->profileurl.'"/>'."\n";
  164. print ' <link rel="openid2.server" href="'.common_local_url('openidserver').'"/>'."\n";
  165. print ' <link rel="openid2.delegate" href="'.$user->getProfile()->profileurl.'"/>'."\n";
  166. }
  167. }
  168. }
  169. }
  170. elseif(substr($_SERVER['REQUEST_URI'],0,7) == '/group/') {
  171. $group_id_or_name = substr($_SERVER['REQUEST_URI'],7);
  172. if(stristr($group_id_or_name,'/id')) {
  173. $group_id_or_name = substr($group_id_or_name, 0, strpos($group_id_or_name,'/id'));
  174. $group = User_group::getKV('id', $group_id_or_name);
  175. if($group instanceof User_group) {
  176. $group_name = $group->nickname;
  177. $group_id = $group_id_or_name;
  178. }
  179. } else {
  180. $group = Local_group::getKV('nickname', $group_id_or_name);
  181. if($group instanceof Local_group) {
  182. $group_id = $group->group_id;
  183. $group_name = $group_id_or_name;
  184. }
  185. }
  186. if(preg_match("/^[a-zA-Z0-9]+$/", $group_id_or_name) == 1 && isset($group_name) && isset($group_id)) {
  187. ?>
  188. <link rel="alternate" href="<?php echo htmlspecialchars(common_local_url('ApiTimelineGroup', array('id'=>$group_id, 'format'=>'as'))); ?>" type="application/stream+json" title="Notice feed for '<?php echo htmlspecialchars($group_name); ?>' group (Activity Streams JSON)" />
  189. <link rel="alternate" href="<?php echo htmlspecialchars(common_local_url('grouprss', array('nickname'=>$group_name))); ?>" type="application/rdf+xml" title="Notice feed for '<?php echo htmlspecialchars($group_name); ?>' group (RSS 1.0)" />
  190. <link rel="alternate" href="<?php echo htmlspecialchars(common_local_url('ApiTimelineGroup', array('id'=>$group_id, 'format'=>'rss'))); ?>" type="application/rss+xml" title="Notice feed for '<?php echo htmlspecialchars($group_name); ?>' group (RSS 2.0)" />
  191. <link rel="alternate" href="<?php echo htmlspecialchars(common_local_url('ApiTimelineGroup', array('id'=>$group_id, 'format'=>'atom'))); ?>" type="application/atom+xml" title="Notice feed for '<?php echo htmlspecialchars($group_name); ?>' group (Atom)" />
  192. <link rel="meta" href="<?php echo htmlspecialchars(common_local_url('foafgroup', array('nickname'=>$group_name))); ?>" type="application/rdf+xml" title="FOAF for '<?php echo htmlspecialchars($group_name); ?>' group" />
  193. <?php
  194. }
  195. }
  196. // oembed discovery for local notices, and twitter cards
  197. if(substr($_SERVER['REQUEST_URI'],0,8) == '/notice/'
  198. && $this->arg('notice')
  199. && array_key_exists('Oembed', StatusNet::getActivePlugins())) {
  200. $notice = Notice::getKV('id', $this->arg('notice'));
  201. if($notice instanceof Notice) {
  202. $profile = $notice->getProfile();
  203. if ($notice->isLocal() && $profile instanceof Profile) {
  204. // maybe get thumbnail url
  205. $embed_thumbnail_url = false;
  206. $attachments = $notice->attachments();
  207. if (!empty($attachments)) {
  208. foreach ($attachments as $attachment) {
  209. if(is_object($attachment)) {
  210. try {
  211. $thumb = $attachment->getThumbnail();
  212. } catch (ServerException $e) {
  213. //
  214. }
  215. if(!empty($thumb) && method_exists('File_thumbnail','url')) {
  216. try {
  217. $embed_thumbnail_url = File_thumbnail::url($thumb->filename);
  218. break; // only first one
  219. } catch (ClientException $e) {
  220. //
  221. }
  222. }
  223. }
  224. }
  225. }
  226. try {
  227. $notice_url = $notice->getUrl();
  228. print '<link title="oEmbed" href="'.common_local_url('apiqvitteroembednotice', array('id' => $notice->id, 'format'=>'json')).'?url='.urlencode($notice_url).'" type="application/json+oembed" rel="alternate">'."\n";
  229. print '<link title="oEmbed" href="'.common_local_url('apiqvitteroembednotice', array('id' => $notice->id, 'format'=>'xml')).'?url='.urlencode($notice_url).'" type="application/xml+oembed" rel="alternate">'."\n";
  230. } catch (Exception $e) {
  231. //
  232. }
  233. // single notice feeds
  234. try {
  235. $single_notice_json = common_local_url('ApiStatusesShow', array( 'id' => $notice->getID(),'format' => 'json'));
  236. $single_notice_atom = common_local_url('ApiStatusesShow', array( 'id' => $notice->getID(),'format' => 'atom'));
  237. print '<link title="Single notice (JSON)" href="'.$single_notice_json.'" type="application/stream+json" rel="alternate">'."\n";
  238. print '<link title="Single notice (Atom)" href="'.$single_notice_atom.'" type="application/atom+xml" rel="alternate">'."\n";
  239. } catch (Exception $e) {
  240. //
  241. }
  242. // twitter cards
  243. print '<meta name="twitter:card" content="summary" />'."\n";
  244. print '<meta name="twitter:title" content="'.htmlspecialchars($profile->fullname).' (@'.$profile->nickname.')" />'."\n";
  245. print '<meta name="twitter:description" content="'.htmlspecialchars($notice->content).'" />'."\n";
  246. if($embed_thumbnail_url) {
  247. print '<meta name="twitter:image" content="'.$embed_thumbnail_url.'" />'."\n";
  248. }
  249. // opengraph
  250. print '<meta property="og:description" content="'.htmlspecialchars($notice->content).'" />'."\n";
  251. print '<meta property="og:site_name" content="'.$sitetitle.'" />'."\n";
  252. if($embed_thumbnail_url) {
  253. print '<meta property="og:image" content="'.$embed_thumbnail_url.'" />'."\n";
  254. }
  255. }
  256. }
  257. }
  258. ?>
  259. <script>
  260. /*
  261. @licstart The following is the entire license notice for the
  262. JavaScript code in this page.
  263. Copyright (C) 2015 Hannes Mannerheim and other contributors
  264. This program is free software: you can redistribute it and/or modify
  265. it under the terms of the GNU Affero General Public License as
  266. published by the Free Software Foundation, either version 3 of the
  267. License, or (at your option) any later version.
  268. This program is distributed in the hope that it will be useful,
  269. but WITHOUT ANY WARRANTY; without even the implied warranty of
  270. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  271. GNU Affero General Public License for more details.
  272. You should have received a copy of the GNU Affero General Public License
  273. along with this program. If not, see <http://www.gnu.org/licenses/>.
  274. @licend The above is the entire license notice
  275. for the JavaScript code in this page.
  276. */
  277. window.usersLanguageCode = <?php print json_encode($user_browser_language) ?>;
  278. window.usersLanguageNameInEnglish = <?php print json_encode(Locale::getDisplayLanguage($user_browser_language, 'en')) ?>;
  279. window.englishLanguageData = <?php print file_get_contents(QVITTERDIR.'/locale/en.json'); ?>;
  280. window.defaultAvatarStreamSize = <?php print json_encode(Avatar::defaultImage(AVATAR_STREAM_SIZE)) ?>;
  281. window.defaultAvatarProfileSize = <?php print json_encode(Avatar::defaultImage(AVATAR_PROFILE_SIZE)) ?>;
  282. window.textLimit = <?php print json_encode((int)common_config('site','textlimit')) ?>;
  283. window.registrationsClosed = <?php print json_encode($registrationsclosed) ?>;
  284. window.thisSiteThinksItIsHttpButIsActuallyHttps = <?php
  285. // this is due to a crazy setup at quitter.se, sorry about that
  286. $siteSSL = common_config('site', 'ssl');
  287. if(isset($_SERVER['HTTPS'])
  288. && $_SERVER['HTTPS'] != 'off'
  289. && $siteSSL == 'never' ) {
  290. $this_site_thinks_it_is_http_but_is_actually_https = true;
  291. print 'true';
  292. }
  293. else {
  294. $this_site_thinks_it_is_http_but_is_actually_https = false;
  295. print 'false';
  296. }
  297. ?>;
  298. window.siteTitle = <?php print json_encode($sitetitle) ?>;
  299. window.loggedIn = <?php
  300. $logged_in_user_json = json_encode($logged_in_user_obj);
  301. $logged_in_user_json = str_replace('http:\/\/quitter.se\/','https:\/\/quitter.se\/',$logged_in_user_json);
  302. print $logged_in_user_json;
  303. ?>;
  304. window.timeBetweenPolling = <?php print QvitterPlugin::settings("timebetweenpolling"); ?>;
  305. window.apiRoot = <?php
  306. $api_root = common_path("api/", StatusNet::isHTTPS());
  307. if($this_site_thinks_it_is_http_but_is_actually_https) {
  308. $api_root = str_replace('http://','https://',$api_root);
  309. }
  310. print '\''.$api_root.'\'';
  311. ?>;
  312. window.fullUrlToThisQvitterApp = '<?php print $qvitterpath; ?>';
  313. window.siteRootDomain = '<?php print $siterootdomain; ?>';
  314. window.siteInstanceURL = '<?php print $instanceurl; ?>';
  315. window.defaultLinkColor = '<?php print QvitterPlugin::settings("defaultlinkcolor"); ?>';
  316. window.defaultBackgroundColor = '<?php print QvitterPlugin::settings("defaultbackgroundcolor"); ?>';
  317. window.siteBackground = '<?php print QvitterPlugin::settings("sitebackground"); ?>';
  318. window.enableWelcomeText = <?php print json_encode(QvitterPlugin::settings("enablewelcometext")); ?>;
  319. window.customWelcomeText = <?php print json_encode(QvitterPlugin::settings("customwelcometext")); ?>;
  320. window.urlShortenerAPIURL = '<?php print QvitterPlugin::settings("urlshortenerapiurl"); ?>';
  321. window.urlShortenerSignature = '<?php print QvitterPlugin::settings("urlshortenersignature"); ?>';
  322. window.urlshortenerFormat = '<?php print QvitterPlugin::settings("urlshortenerformat"); ?>';
  323. window.commonSessionToken = '<?php print common_session_token(); ?>';
  324. window.siteMaxThumbnailSize = <?php print common_config('thumbnail', 'maxsize'); ?>;
  325. window.siteAttachmentURLBase = '<?php print $attachmentroot; ?>';
  326. window.siteEmail = '<?php print common_config('site', 'email'); ?>';
  327. window.siteLicenseTitle = '<?php print common_config('license', 'title'); ?>';
  328. window.siteLicenseURL = '<?php print common_config('license', 'url'); ?>';
  329. window.customTermsOfUse = <?php print json_encode(QvitterPlugin::settings("customtermsofuse")); ?>;
  330. window.siteLocalOnlyDefaultPath = <?php print (common_config('public', 'localonly') ? 'true' : 'false'); ?>;
  331. <?php
  332. // Get all topics in Qvitter's namespace in Profile_prefs
  333. if($logged_in_user) {
  334. try {
  335. $qvitter_profile_prefs = Profile_prefs::getNamespace(Profile::current(),'qvitter');
  336. } catch (Exception $e) {
  337. $qvitter_profile_prefs = array();
  338. }
  339. if(count($qvitter_profile_prefs)>0) {
  340. $topic_data = new stdClass();
  341. foreach($qvitter_profile_prefs as $pref) {
  342. $topic_data->{$pref->topic} = $pref->data;
  343. }
  344. print 'window.qvitterProfilePrefs = '.json_encode($topic_data).';';
  345. }
  346. else {
  347. print 'window.qvitterProfilePrefs = false;';
  348. }
  349. }
  350. // keyboard shortcuts can be disabled
  351. $disable_keyboard_shortcuts = false;
  352. if($logged_in_user) {
  353. try {
  354. $disable_keyboard_shortcuts = Profile_prefs::getData($logged_in_user->getProfile(), 'qvitter', 'disable_keyboard_shortcuts');
  355. if($disable_keyboard_shortcuts == '1' || $disable_keyboard_shortcuts == 1) {
  356. $disable_keyboard_shortcuts = true;
  357. }
  358. } catch (Exception $e) {
  359. //
  360. }
  361. }
  362. print 'window.disableKeyboardShortcuts = '.var_export($disable_keyboard_shortcuts, true).';';
  363. ?>
  364. // available language files and their last update time
  365. window.availableLanguages = {<?php
  366. // scan all files in the locale directory and create a json object with their change date added
  367. $available_languages = array_diff(scandir(QVITTERDIR.'/locale'), array('..', '.'));
  368. foreach($available_languages as $lankey=>$lan) {
  369. $lancode = substr($lan,0,strpos($lan,'.'));
  370. // for the paranthesis containing language region to work with rtl in ltr enviroment and vice versa, we add a
  371. // special rtl or ltr html char after the paranthesis
  372. // this list is incomplete, but if any rtl language gets a regional translation, it will probably be arabic
  373. $rtl_or_ltr_special_char = '&lrm;';
  374. $base_lancode = substr($lancode,0,strpos($lancode,'_'));
  375. if($base_lancode == 'ar'
  376. || $base_lancode == 'fa'
  377. || $base_lancode == 'he') {
  378. $rtl_or_ltr_special_char = '&rlm;';
  379. }
  380. // also make an array with all language names, to use for generating menu
  381. $languagecodesandnames[$lancode]['english_name'] = Locale::getDisplayName($lancode, 'en');
  382. $languagecodesandnames[$lancode]['name'] = Locale::getDisplayName($lancode, $lancode);
  383. $languagecodesandnames[$lancode]['tooltip'] = $languagecodesandnames[$lancode]['name'].' – '.$languagecodesandnames[$lancode]['english_name'];
  384. if($languagecodesandnames[$lancode]['name'] == $languagecodesandnames[$lancode]['english_name']) {
  385. $languagecodesandnames[$lancode]['tooltip'] = $languagecodesandnames[$lancode]['english_name'];
  386. }
  387. // ahorita meme only on quitter.es
  388. if($lancode == 'es_ahorita') {
  389. if($siterootdomain == 'quitter.es') {
  390. $languagecodesandnames[$lancode]['name'] = 'español (ahorita)';
  391. }
  392. else {
  393. unset($available_languages[$lankey]);
  394. unset($languagecodesandnames[$lancode]);
  395. continue;
  396. }
  397. }
  398. print "\n".' "'.$lancode.'": "'.$lan.'?changed='.date('YmdHis',filemtime(QVITTERDIR.'/locale/'.$lan)).'",';
  399. }
  400. ?>
  401. };
  402. </script>
  403. <?php
  404. // event for other plugins to use to add head elements to qvitter
  405. Event::handle('QvitterEndShowHeadElements', array($this));
  406. ?>
  407. </head>
  408. <body class="<?php
  409. // rights as body classes
  410. if($logged_in_user) {
  411. if($logged_in_user_obj['rights']['silence']){
  412. print 'has-right-to-silence';
  413. }
  414. if($logged_in_user_obj['rights']['sandbox']){
  415. print ' has-right-to-sandbox';
  416. }
  417. }
  418. ?>" style="background-color:<?php print QvitterPlugin::settings("defaultbackgroundcolor"); ?>">
  419. <?php
  420. // add an accessibility toggle link to switch to standard UI, if we're logged in
  421. if($logged_in_user) {
  422. print '<a id="accessibility-toggle-link" href="#"></a>';
  423. }
  424. ?>
  425. <input id="upload-image-input" class="upload-image-input" type="file" name="upload-image-input">
  426. <div class="topbar">
  427. <a href="<?php
  428. // if we're logged in, the logo links to the home stream
  429. // if logged out it links to the site's public stream
  430. if($logged_in_user) {
  431. print $instanceurl.$logged_in_user_nickname.'/all';
  432. }
  433. else {
  434. print $instanceurl.'main/public';
  435. }
  436. ?>"><div id="logo"></div></a><?php
  437. // menu for logged in users
  438. if($logged_in_user) { ?>
  439. <a id="settingslink">
  440. <div class="dropdown-toggle">
  441. <div class="nav-session" style="background-image:url('<?php print htmlspecialchars($logged_in_user_obj['profile_image_url_profile_size']) ?>')"></div>
  442. </div>
  443. </a><?php
  444. }
  445. ?><div id="top-compose" class="hidden"></div>
  446. <ul class="quitter-settings dropdown-menu">
  447. <li class="dropdown-caret right">
  448. <span class="caret-outer"></span>
  449. <span class="caret-inner"></span>
  450. </li>
  451. <li class="fullwidth"><a id="top-menu-profile-link" class="no-hover-card" href="<?php print $instanceurl.$logged_in_user_obj['screen_name']; ?>"><div id="top-menu-profile-link-fullname"><?php print htmlspecialchars($logged_in_user_obj['name']); ?></div><div id="top-menu-profile-link-view-profile"></div></a></li>
  452. <li class="fullwidth dropdown-divider"></li>
  453. <li class="fullwidth"><a id="faq-link"></a></li>
  454. <li class="fullwidth"><a id="tou-link"></a></li>
  455. <?php
  456. if($disable_keyboard_shortcuts === true) {
  457. print '<li class="fullwidth"><a id="shortcuts-link" class="disabled" href="'.$instanceurl.'settings/qvitter"></a></li>';
  458. }
  459. else {
  460. print '<li class="fullwidth"><a id="shortcuts-link"></a></li>';
  461. }
  462. if (common_config('invite', 'enabled') && !common_config('site', 'closed')) { ?>
  463. <li class="fullwidth"><a id="invite-link" href="<?php print $instanceurl; ?>main/invite"></a></li>
  464. <?php } ?>
  465. <li class="fullwidth"><a id="classic-link"></a></li>
  466. <li class="fullwidth dropdown-divider"></li>
  467. <li class="fullwidth"><a id="logout"></a></li>
  468. <li class="fullwidth language dropdown-divider"></li>
  469. <?php
  470. // languages
  471. foreach($languagecodesandnames as $lancode=>$lan) {
  472. print '<li class="language"><a class="language-link" data-tooltip="'.$lan['tooltip'].'" data-lang-code="'.$lancode.'">'.$lan['name'].'</a></li>';
  473. }
  474. ?>
  475. <li class="fullwidth language dropdown-divider"></li>
  476. <li class="fullwidth"><a href="https://git.gnu.io/h2p/Qvitter/tree/master/locale" target="_blank" id="add-edit-language-link"></a></li>
  477. </ul>
  478. <div class="global-nav">
  479. <div class="global-nav-inner">
  480. <div class="container">
  481. <div id="search">
  482. <input type="text" spellcheck="false" autocomplete="off" name="q" placeholder="Sök" id="search-query" class="search-input">
  483. <span class="search-icon">
  484. <button class="icon nav-search" type="submit" tabindex="-1">
  485. <span> Sök </span>
  486. </button>
  487. </span>
  488. </div>
  489. <ul class="language-dropdown">
  490. <li class="dropdown">
  491. <a class="dropdown-toggle">
  492. <small></small>
  493. <span class="current-language"></span>
  494. <b class="caret"></b>
  495. </a>
  496. <ul class="dropdown-menu">
  497. <li class="dropdown-caret right">
  498. <span class="caret-outer"></span>
  499. <span class="caret-inner"></span>
  500. </li>
  501. <?php
  502. // languages
  503. foreach($languagecodesandnames as $lancode=>$lan) {
  504. print '<li><a class="language-link" data-tooltip="'.$lan['english_name'].'" data-lang-code="'.$lancode.'">'.$lan['name'].'</a></li>';
  505. }
  506. ?>
  507. </ul>
  508. </li>
  509. </ul>
  510. </div>
  511. </div>
  512. </div>
  513. </div>
  514. <div id="no-js-error">Please enable javascript to use this site.<script>var element = document.getElementById('no-js-error'); element.parentNode.removeChild(element);</script></div>
  515. <div id="page-container">
  516. <?php
  517. $site_notice = common_config('site', 'notice');
  518. if(!empty($site_notice)) {
  519. print '<div id="site-notice">'.common_config('site', 'notice').'</div>';
  520. }
  521. // welcome text, login and register container if logged out
  522. if($logged_in_user === null) { ?>
  523. <div class="front-welcome-text <?php if ($registrationsclosed) { print 'registrations-closed'; } ?>"></div>
  524. <div id="login-register-container">
  525. <div id="login-content">
  526. <form id="form_login" class="form_settings" action="<?php print common_local_url('qvitterlogin'); ?>" method="post">
  527. <div id="username-container">
  528. <input id="nickname" name="nickname" type="text" value="<?php print $logged_in_user_nickname ?>" tabindex="1" />
  529. </div>
  530. <table class="password-signin"><tbody><tr>
  531. <td class="flex-table-primary">
  532. <div class="placeholding-input">
  533. <input id="password" name="password" type="password" tabindex="2" value="" />
  534. </div>
  535. </td>
  536. <td class="flex-table-secondary">
  537. <button class="submit" type="submit" id="submit-login" tabindex="4"></button>
  538. </td>
  539. </tr></tbody></table>
  540. <div id="remember-forgot">
  541. <input type="checkbox" id="rememberme" name="rememberme" value="yes" tabindex="3" checked="checked"> <span id="rememberme_label"></span> · <a id="forgot-password" href="<?php print $instanceurl ?>main/recoverpassword" ></a>
  542. <input type="hidden" id="token" name="token" value="<?php print common_session_token(); ?>">
  543. <?php
  544. if (array_key_exists('OpenID', StatusNet::getActivePlugins())) {
  545. print '<a href="'.$instanceurl.'main/openid" id="openid-login" title="OpenID" donthijack>OpenID</a>';
  546. }
  547. ?>
  548. </div>
  549. </form>
  550. </div>
  551. <?php
  552. if($registrationsclosed === false && $client_ip_is_blocked === false) {
  553. ?><div class="front-signup">
  554. <h2></h2>
  555. <div class="signup-input-container"><input placeholder="" type="text" name="user[name]" autocomplete="off" class="text-input" id="signup-user-name"></div>
  556. <div class="signup-input-container"><input placeholder="" type="text" name="user[email]" autocomplete="off" id="signup-user-email"></div>
  557. <div class="signup-input-container"><input placeholder="" type="password" name="user[user_password]" class="text-input" id="signup-user-password"></div>
  558. <button id="signup-btn-step1" class="signup-btn" type="submit"></button>
  559. </div>
  560. <div id="other-servers-link"></div><?php }
  561. ?><div id="qvitter-notice-logged-out"><?php print common_config('site', 'qvitternoticeloggedout'); ?></div><?php
  562. // event for other plugins to add html to the logged in sidebar
  563. Event::handle('QvitterEndShowSidebarLoggedOut', array($this));
  564. ?></div><?php
  565. }
  566. // box containing the logged in users queet count and compose form
  567. if($logged_in_user) { ?>
  568. <div id="user-container" style="display:none;">
  569. <div id="user-header" style="background-image:url('<?php print htmlspecialchars($logged_in_user_obj['cover_photo']) ?>')">
  570. <div id="mini-logged-in-user-cog-wheel"></div>
  571. <div class="profile-header-inner-overlay"></div>
  572. <div id="user-avatar-container"><img id="user-avatar" src="<?php print htmlspecialchars($logged_in_user_obj['profile_image_url_profile_size']) ?>" /></div>
  573. <div id="user-name"><?php print htmlspecialchars($logged_in_user_obj['name']) ?></div>
  574. <div id="user-screen-name"><?php print htmlspecialchars($logged_in_user_obj['screen_name']) ?></div>
  575. </div>
  576. <ul id="user-body">
  577. <li><a href="<?php print $instanceurl.$logged_in_user->nickname ?>" id="user-queets"><span class="label"></span><strong><?php print $logged_in_user_obj['statuses_count'] ?></strong></a></li>
  578. <li><a href="<?php print $instanceurl.$logged_in_user->nickname ?>/subscriptions" id="user-following"><span class="label"></span><strong><?php print $logged_in_user_obj['friends_count'] ?></strong></a></li>
  579. <li><a href="<?php print $instanceurl.$logged_in_user->nickname ?>/groups" id="user-groups"><span class="label"></span><strong><?php print $logged_in_user_obj['groups_count'] ?></strong></a></li>
  580. </ul>
  581. <div id="user-footer">
  582. <div id="user-footer-inner">
  583. <div id="queet-box" class="queet-box queet-box-syntax" data-start-text=""></div>
  584. <div class="syntax-middle"></div>
  585. <div class="syntax-two" contenteditable="true"></div>
  586. <div class="mentions-suggestions"></div>
  587. <div class="queet-toolbar">
  588. <div class="queet-box-extras">
  589. <button class="upload-image"></button>
  590. <button class="shorten disabled">URL</button>
  591. </div>
  592. <div class="queet-button">
  593. <span class="queet-counter"></span>
  594. <button></button>
  595. </div>
  596. </div>
  597. </div>
  598. </div>
  599. <div id="main-menu" class="menu-container"><?php
  600. if($logged_in_user) {
  601. ?><a href="<?php print $instanceurl.$logged_in_user->nickname ?>/all" class="stream-selection friends-timeline"><i class="chev-right"></i></a>
  602. <a href="<?php print $instanceurl.$logged_in_user->nickname ?>/notifications" class="stream-selection notifications"><span id="unseen-notifications"></span><i class="chev-right"></i></a>
  603. <a href="<?php print $instanceurl.$logged_in_user->nickname ?>" class="stream-selection my-timeline"><i class="chev-right"></i></a>
  604. <a href="<?php print $instanceurl.$logged_in_user->nickname ?>/favorites" class="stream-selection favorites"><i class="chev-right"></i></a>
  605. <a href="<?php print $instanceurl ?>main/public" class="stream-selection public-timeline"><i class="chev-right"></i></a>
  606. <a href="<?php print $instanceurl ?>main/all" class="stream-selection public-and-external-timeline"><i class="chev-right"></i></a>
  607. <?php
  608. }
  609. ?>
  610. </div>
  611. <div class="menu-container" id="bookmark-container"></div>
  612. <div class="menu-container" id="history-container"></div>
  613. <div id="clear-history"></div>
  614. <div id="qvitter-notice"><?php print common_config('site', 'qvitternotice'); ?></div><?php
  615. // event for other plugins to add html to the logged in sidebar
  616. Event::handle('QvitterEndShowSidebarLoggedIn', array($this));
  617. ?></div><?php
  618. } ?>
  619. <div id="feed">
  620. <div id="feed-header">
  621. <div id="feed-header-inner">
  622. <h2>
  623. <span id="stream-header"></span>
  624. </h2>
  625. <div class="reload-stream"></div>
  626. </div>
  627. <div id="feed-header-description"></div>
  628. </div>
  629. <div id="new-queets-bar-container" class="hidden"><div id="new-queets-bar"></div></div>
  630. <div id="feed-body"></div>
  631. </div>
  632. <div id="hidden-html"><?php
  633. // adds temporary support for microformats and linkbacks on the notice page
  634. if(substr($_SERVER['REQUEST_URI'],0,8) == '/notice/' && $this->arg('notice')) {
  635. echo '<ol class="notices xoxo">';
  636. if($notice instanceof Notice) {
  637. $widget = new NoticeListItem($notice, $this);
  638. $widget->show();
  639. $this->flush();
  640. }
  641. echo '</ol>';
  642. }
  643. Event::handle('QvitterHiddenHtml', array($this));
  644. ?></div>
  645. <div id="footer"><div id="footer-spinner-container"></div></div>
  646. </div>
  647. <script type="text/javascript" src="<?php print $qvitterpath; ?>js/lib/jquery-2.1.4.min.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/lib/jquery-2.1.4.min.js')); ?>"></script>
  648. <script type="text/javascript" src="<?php print $qvitterpath; ?>js/lib/jquery-ui.min.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/lib/jquery-ui.min.js')); ?>"></script>
  649. <script type="text/javascript" src="<?php print $qvitterpath; ?>js/lib/jquery.minicolors.min.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/lib/jquery.minicolors.min.js')); ?>"></script>
  650. <script type="text/javascript" src="<?php print $qvitterpath; ?>js/lib/jquery.jWindowCrop.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/lib/jquery.jWindowCrop.js')); ?>"></script>
  651. <script type="text/javascript" src="<?php print $qvitterpath; ?>js/lib/load-image.min.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/lib/load-image.min.js')); ?>"></script>
  652. <script type="text/javascript" src="<?php print $qvitterpath; ?>js/lib/xregexp-all-3.0.0-pre.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/lib/xregexp-all-3.0.0-pre.js')); ?>"></script>
  653. <script type="text/javascript" src="<?php print $qvitterpath; ?>js/lib/lz-string.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/lib/lz-string.js')); ?>"></script>
  654. <script type="text/javascript" src="<?php print $qvitterpath; ?>js/lib/bowser.min.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/lib/bowser.min.js')); ?>"></script>
  655. <script charset="utf-8" type="text/javascript" src="<?php print $qvitterpath; ?>js/dom-functions.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/dom-functions.js')); ?>"></script>
  656. <script charset="utf-8" type="text/javascript" src="<?php print $qvitterpath; ?>js/misc-functions.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/misc-functions.js')); ?>"></script>
  657. <script charset="utf-8" type="text/javascript" src="<?php print $qvitterpath; ?>js/ajax-functions.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/ajax-functions.js')); ?>"></script>
  658. <script charset="utf-8" type="text/javascript" src="<?php print $qvitterpath; ?>js/stream-router.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/stream-router.js')); ?>"></script>
  659. <script charset="utf-8" type="text/javascript" src="<?php print $qvitterpath; ?>js/qvitter.js?changed=<?php print date('YmdHis',filemtime(QVITTERDIR.'/js/qvitter.js')); ?>"></script>
  660. <?php
  661. // event for other plugins to add scripts to qvitter
  662. Event::handle('QvitterEndShowScripts', array($this));
  663. // we might have custom javascript in the config file that we want to add
  664. if(QvitterPlugin::settings('js')) {
  665. print '<script type="text/javascript">'.QvitterPlugin::settings('js').'</script>';
  666. }
  667. ?>
  668. <div id="dynamic-styles">
  669. <style>
  670. a, a:visited, a:active,
  671. ul.stats li:hover a,
  672. ul.stats li:hover a strong,
  673. #user-body a:hover div strong,
  674. #user-body a:hover div div,
  675. .permalink-link:hover,
  676. .stream-item.expanded > .queet .stream-item-expand,
  677. .stream-item-footer .with-icn .requeet-text a b:hover,
  678. .queet-text span.attachment.more,
  679. .stream-item-header .created-at a:hover,
  680. .stream-item-header a.account-group:hover .name,
  681. .queet:hover .stream-item-expand,
  682. .show-full-conversation:hover,
  683. #new-queets-bar,
  684. .menu-container div,
  685. .cm-mention, .cm-tag, .cm-group, .cm-url, .cm-email,
  686. div.syntax-middle span,
  687. #user-body strong,
  688. ul.stats,
  689. .stream-item:not(.temp-post) ul.queet-actions li .icon:not(.is-mine):hover:before,
  690. .show-full-conversation,
  691. #user-body #user-queets:hover .label,
  692. #user-body #user-groups:hover .label,
  693. #user-body #user-following:hover .label,
  694. ul.stats a strong,
  695. .queet-box-extras button,
  696. #openid-login:hover:after,
  697. .post-to-group,
  698. .stream-item-header .addressees .reply-to .h-card.not-mentioned-inline {
  699. color:/*COLORSTART*/<?php print QvitterPlugin::settings("defaultlinkcolor"); ?>/*COLOREND*/;
  700. }
  701. #unseen-notifications,
  702. .stream-item.notification.not-seen > .queet::before,
  703. #top-compose,
  704. #logo,
  705. .queet-toolbar button,
  706. #user-header,
  707. .profile-header-inner,
  708. .topbar,
  709. .menu-container,
  710. .member-button.member,
  711. .external-follow-button.following,
  712. .qvitter-follow-button.following,
  713. .save-profile-button,
  714. .crop-and-save-button,
  715. .topbar .global-nav.show-logo:before,
  716. .topbar .global-nav.pulse-logo:before,
  717. .dropdown-menu li:not(.dropdown-caret) a:hover {
  718. background-color:/*BACKGROUNDCOLORSTART*/<?php print QvitterPlugin::settings("defaultlinkcolor"); ?>/*BACKGROUNDCOLOREND*/;
  719. }
  720. .queet-box-syntax[contenteditable="true"]:focus,
  721. .stream-item.selected-by-keyboard::before {
  722. border-color:/*BORDERCOLORSTART*/#999999/*BORDERCOLOREND*/;
  723. }
  724. #user-footer-inner,
  725. .inline-reply-queetbox,
  726. #popup-faq #faq-container p.indent {
  727. background-color:/*LIGHTERBACKGROUNDCOLORSTART*/rgb(205,230,239)/*LIGHTERBACKGROUNDCOLOREND*/;
  728. }
  729. #user-footer-inner,
  730. .queet-box,
  731. .queet-box-syntax[contenteditable="true"],
  732. .inline-reply-queetbox,
  733. span.inline-reply-caret,
  734. .stream-item.expanded .stream-item.first-visible-after-parent,
  735. #popup-faq #faq-container p.indent,
  736. .post-to-group,
  737. .quoted-notice:hover,
  738. .oembed-item:hover,
  739. .stream-item:hover:not(.expanded) .quoted-notice:hover,
  740. .stream-item:hover:not(.expanded) .oembed-item:hover {
  741. border-color:/*LIGHTERBORDERCOLORSTART*/rgb(155,206,224)/*LIGHTERBORDERCOLOREND*/;
  742. }
  743. span.inline-reply-caret .caret-inner {
  744. border-bottom-color:/*LIGHTERBORDERBOTTOMCOLORSTART*/rgb(205,230,239)/*LIGHTERBORDERBOTTOMCOLOREND*/;
  745. }
  746. .modal-close .icon,
  747. .chev-right,
  748. .close-right,
  749. button.icon.nav-search,
  750. .member-button .join-text i,
  751. .external-member-button .join-text i,
  752. .external-follow-button .follow-text i,
  753. .qvitter-follow-button .follow-text i,
  754. #logo,
  755. .upload-cover-photo,
  756. .upload-avatar,
  757. .upload-background-image,
  758. button.shorten i,
  759. .reload-stream,
  760. .topbar .global-nav:before,
  761. .stream-item.notification.repeat .dogear,
  762. .stream-item.notification.like .dogear,
  763. .ostatus-link,
  764. .close-edit-profile-window {
  765. background-image: url("<?php print QvitterPlugin::settings("sprite"); ?>");
  766. background-size: 500px 1329px;
  767. }
  768. @media (max-width: 910px) {
  769. #search-query,
  770. .menu-container a,
  771. .menu-container a.current,
  772. .stream-selection.friends-timeline:after,
  773. .stream-selection.notifications:after,
  774. .stream-selection.my-timeline:after,
  775. .stream-selection.public-timeline:after {
  776. background-image: url("<?php print QvitterPlugin::settings("sprite"); ?>");
  777. background-size: 500px 1329px;
  778. }
  779. }
  780. </style>
  781. </div>
  782. </body>
  783. </html>
  784. <?php
  785. }
  786. }