296 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			296 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| // This file is part of GNU social - https://www.gnu.org/software/social
 | |
| //
 | |
| // GNU social is free software: you can redistribute it and/or modify
 | |
| // it under the terms of the GNU Affero General Public License as published by
 | |
| // the Free Software Foundation, either version 3 of the License, or
 | |
| // (at your option) any later version.
 | |
| //
 | |
| // GNU social is distributed in the hope that it will be useful,
 | |
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| // GNU Affero General Public License for more details.
 | |
| //
 | |
| // You should have received a copy of the GNU Affero General Public License
 | |
| // along with GNU social.  If not, see <http://www.gnu.org/licenses/>.
 | |
| 
 | |
| /**
 | |
|  * Adds a setting to allow a user to hide #NSFW-hashtagged notices behind a
 | |
|  * blocker image until clicked.
 | |
|  *
 | |
|  * @package   GNUsocial
 | |
|  * @author    MoonMan
 | |
|  * @copyright 2019 Free Software Foundation, Inc http://www.fsf.org
 | |
|  * @license   https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
 | |
|  */
 | |
| 
 | |
| defined('GNUSOCIAL') || die();
 | |
| 
 | |
| /**
 | |
|  * Class SensitiveContentPlugin
 | |
|  * Handles the main UI stuff
 | |
|  *
 | |
|  * @package   GNUsocial
 | |
|  * @author    MoonMan
 | |
|  * @copyright 2019 Free Software Foundation, Inc http://www.fsf.org
 | |
|  * @license   https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
 | |
|  */
 | |
| class SensitiveContentPlugin extends Plugin
 | |
| {
 | |
|     public $blockerimage = 'img/blocker.png';
 | |
|     public $hideforvisitors = false;
 | |
|     const PLUGIN_VERSION = '0.1.0';
 | |
| 
 | |
|     public function onPluginVersion(array &$versions): bool
 | |
|     {
 | |
|         $versions[] = [
 | |
|             'name' => 'Sensitive Content',
 | |
|             'version' => self::PLUGIN_VERSION,
 | |
|             'author' => 'MoonMan',
 | |
|             'homepage' => GNUSOCIAL_ENGINE_REPO_URL . 'tree/master/plugins/SensitiveContent/',
 | |
|             'description' => _m('Mark, hide/show sensitive notices like on Twitter.'),
 | |
|         ];
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * What blocker image to use?
 | |
|      *
 | |
|      * @return string
 | |
|      */
 | |
|     public function getBlockerImage(): string
 | |
|     {
 | |
|         return Plugin::staticPath('SensitiveContent', '') . $this->blockerimage;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @param $profile
 | |
|      * @return bool
 | |
|      */
 | |
|     public function shouldHide($profile): bool
 | |
|     {
 | |
|         if (isset($profile) && $profile instanceof Profile) {
 | |
|             return $this->getHideSensitive($profile);
 | |
|         }
 | |
|         return $this->hideforvisitors;
 | |
|     }
 | |
| 
 | |
|     public function getHideSensitive(Profile $profile): bool
 | |
|     {
 | |
|         $c = Cache::instance();
 | |
| 
 | |
|         // if (!empty($c)) {
 | |
|         //     $hidesensitive = $c->get(Cache::key('profile:hide_sensitive:'.$profile->id));
 | |
|         //     if (is_numeric($hidesensitive)) {
 | |
|         //         return (bool) $hidesensitive;
 | |
|         //     } else {
 | |
|         //         return false;
 | |
|         //     }
 | |
|         // }
 | |
| 
 | |
|         $hidesensitive = $profile->getPref('MoonMan', 'hide_sensitive', '0');
 | |
| 
 | |
|         if (!empty($c)) {
 | |
|             //not using it yet.
 | |
|             $c->set(Cache::key('profile:hide_sensitive:' . $profile->id), $hidesensitive);
 | |
|         }
 | |
| 
 | |
|         //common_log(LOG_DEBUG, "SENSITIVECONTENT hidesensitive? id " . $profile->id . " value " . (bool) $hidesensitive );
 | |
| 
 | |
|         if (is_null($hidesensitive)) {
 | |
|             return false;
 | |
|         }
 | |
|         if (is_numeric($hidesensitive)) {
 | |
|             return (bool)$hidesensitive;
 | |
|         }
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     /* GNU social EVENTS */
 | |
| 
 | |
|     public function onNoticeSimpleStatusArray($notice, &$twitter_status, $scoped)
 | |
|     {
 | |
|         $twitter_status['tags'] = $notice->getTags();
 | |
|     }
 | |
| 
 | |
|     public function onTwitterUserArray($profile, &$twitter_user, $scoped)
 | |
|     {
 | |
|         if ($scoped instanceof Profile && $scoped->sameAs($profile)) {
 | |
|             $twitter_user['hide_sensitive'] = $this->getHideSensitive($scoped);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function onRouterInitialized(URLMapper $m)
 | |
|     {
 | |
|         $m->connect('settings/sensitivecontent',
 | |
|             ['action' => 'sensitivecontentsettings']);
 | |
|     }
 | |
| 
 | |
|     public function onEndAccountSettingsNav($action)
 | |
|     {
 | |
|         $action->menuItem(common_local_url('sensitivecontentsettings'),
 | |
|             _m('MENU', 'Sensitive Content'),
 | |
|             _m('Settings for display of sensitive content.'));
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     public function onEndShowStyles(Action $action)
 | |
|     {
 | |
|         $blocker = $this->getBlockerImage();
 | |
| 
 | |
|         $styles = <<<EOB
 | |
| 
 | |
| /* default no show */
 | |
| html .tagcontainer > footer > .attachments > .inline-attachment > .attachment-wrapper > .sensitive-blocker {
 | |
|     display: none;
 | |
| }
 | |
| 
 | |
| html[data-hidesensitive='true'] .tagcontainer.data-tag-nsfw > footer > .attachments > .inline-attachment > .attachment-wrapper > .sensitive-blocker {
 | |
| display: block;
 | |
| /* hugh, two magic numbers below, sorry :( */
 | |
| width: 98%;
 | |
| height: 90%;
 | |
| position: absolute;
 | |
| /* z-index: 100; */
 | |
| /* background-color: #d4baba; */
 | |
| background-color: black;
 | |
| background-image: url({$blocker});
 | |
| background-repeat: no-repeat;
 | |
| background-position: center center;
 | |
| background-size: contain;
 | |
| transition: opacity 1s ease-in-out;
 | |
| }
 | |
| 
 | |
| html[data-hidesensitive='true'] .tagcontainer.data-tag-nsfw > footer > .attachments > .inline-attachment > .attachment-wrapper > .sensitive-blocker.reveal {
 | |
|     opacity: 0;
 | |
| }
 | |
| 
 | |
| EOB;
 | |
| 
 | |
|         $action->style($styles);
 | |
|     }
 | |
| 
 | |
|     public function onStartShowAttachmentRepresentation($out, $file)
 | |
|     {
 | |
|         $classes = 'sensitive-blocker'; //'sensitive-blocker';
 | |
|         $thumbnail = null;
 | |
|         try {
 | |
|             $thumbnail = $file->getThumbnail();
 | |
|         } catch (Exception $e) {
 | |
|             $thumbnail = null;
 | |
|         }
 | |
|         // $thumb_width_css = $thumbnail ? $thumbnail->width . 'px' : '100%';
 | |
|         // $thumb_height_css = $thumbnail ? $thumbnail->height . 'px' : '100%';
 | |
| 
 | |
|         $out->elementStart('div', [
 | |
|             'class' => 'attachment-wrapper',
 | |
|             /* 'style' => "height: {$thumb_height_css}; width: {$thumb_width_css};", */
 | |
|         ]); // needs height of thumb
 | |
|         $out->elementStart('div', [
 | |
|             'class' => $classes,
 | |
|             'onclick' => 'toggleSpoiler(event)',
 | |
|             /* 'style' => "height: {$thumb_height_css}; width: {$thumb_width_css};", */
 | |
|         ]);
 | |
|         $out->raw(' ');
 | |
|         $out->elementEnd('div');
 | |
|     }
 | |
| 
 | |
|     public function onEndShowAttachmentRepresentation($out, $file)
 | |
|     {
 | |
|         $out->elementEnd('div');
 | |
|     }
 | |
| 
 | |
|     public function onEndShowScripts(Action $action)
 | |
|     {
 | |
|         $profile = $action->getScoped();
 | |
|         $hidesensitive = $this->shouldHide($profile);
 | |
|         $hidesensitive_string = $hidesensitive ? 'true' : 'false';
 | |
| 
 | |
|         $inline = <<<EOB
 | |
| 
 | |
| window.hidesensitive = {$hidesensitive_string};
 | |
| 
 | |
| function toggleSpoiler(evt) {
 | |
|     if (window.hidesensitive) evt.target.classList.toggle('reveal');
 | |
| }
 | |
| EOB;
 | |
|         $action->inlineScript($inline);
 | |
|     }
 | |
| 
 | |
|     public function onEndOpenNoticeListItemElement(NoticeListItem $nli)
 | |
|     {
 | |
|         $rawtags = $nli->getNotice()->getTags();
 | |
|         $classes = 'tagcontainer';
 | |
| 
 | |
|         foreach ($rawtags as $tag) {
 | |
|             $classes = $classes . ' data-tag-' . $tag;
 | |
|         }
 | |
| 
 | |
|         $nli->elementStart('span', ['class' => $classes]);
 | |
|         //$nli->elementEnd('span');
 | |
|     }
 | |
| 
 | |
|     public function onStartCloseNoticeListItemElement(NoticeListItem $nli)
 | |
|     {
 | |
|         $nli->elementEnd('span');
 | |
|     }
 | |
| 
 | |
|     public function onStartHtmlElement($action, &$attrs)
 | |
|     {
 | |
|         $profile = Profile::current();
 | |
|         $hidesensitive = $this->shouldHide($profile);
 | |
| 
 | |
|         $attrs = array_merge($attrs,
 | |
|             ['data-hidesensitive' => ($hidesensitive ? 'true' : 'false')]
 | |
|         );
 | |
|     }
 | |
| 
 | |
|     /* Qvitter's EVENTS */
 | |
| 
 | |
|     public function onQvitterEndShowHeadElements(Action $action)
 | |
|     {
 | |
|         $blocker = $this->getBlockerImage();
 | |
|         common_log(LOG_DEBUG, 'SENSITIVECONTENT ' . $blocker);
 | |
| 
 | |
|         $styles = <<<EOB
 | |
| 
 | |
| .sensitive-blocker {
 | |
|   display: none;
 | |
| }
 | |
| 
 | |
| div.stream-item.notice.sensitive-notice .sensitive-blocker {
 | |
| display: block;
 | |
| width: 100%;
 | |
| height: 100%;
 | |
| position: absolute;
 | |
| z-index: 100;
 | |
| background-color: #d4baba;
 | |
| background-image: url({$blocker});
 | |
| background-repeat: no-repeat;
 | |
| background-position: center center;
 | |
| background-size: contain;
 | |
| transition: opacity 1s ease-in-out;
 | |
| }
 | |
| 
 | |
| .sensitive-blocker:hover {
 | |
|   opacity: .5;
 | |
| }
 | |
| 
 | |
| div.stream-item.notice.expanded.sensitive-notice .sensitive-blocker {
 | |
| display: none;
 | |
| background-color: transparent;
 | |
| background-image: none;
 | |
| }
 | |
| 
 | |
| EOB;
 | |
| 
 | |
|         $action->style($styles);
 | |
|     }
 | |
| 
 | |
|     public function onQvitterEndShowScripts(Action $action)
 | |
|     {
 | |
|         $action->script(Plugin::staticPath('SensitiveContent', '') . 'js/sensitivecontent.js');
 | |
|     }
 | |
| }
 |