. /** * Show latest and greatest profile flags * * @category Action * @package GNUsocial * @author Evan Prodromou * @copyright 2009 StatusNet, Inc. * @license http://www.fsf.org/licensing/licenses/agpl.html AGPLv3 */ defined('GNUSOCIAL') || die(); /** * Show the latest and greatest profile flags * * @category Action * @package GNUsocial * @author Evan Prodromou * @copyright 2009 StatusNet, Inc. * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later */ class AdminprofileflagAction extends Action { public $page; public $profiles; /** * Take arguments for running * * @param array $args $_REQUEST args * * @return bool success flag */ public function prepare(array $args = []) { parent::prepare($args); $user = common_current_user(); // User must be logged in. if (!common_logged_in()) { // TRANS: Error message displayed when trying to perform an action that requires a logged in user. $this->clientError(_m('Not logged in.')); } $user = common_current_user(); // ...because they're logged in assert(!empty($user)); // It must be a "real" login, not saved cookie login if (!common_is_real_login()) { // Cookie theft is too easy; we require automatic // logins to re-authenticate before admining the site common_set_returnto($this->selfUrl()); if (Event::handle('RedirectToLogin', [$this, $user])) { common_redirect(common_local_url('login'), 303); } } // User must have the right to review flags if (!$user->hasRight(UserFlagPlugin::REVIEWFLAGS)) { // TRANS: Error message displayed when trying to review profile flags while not authorised. $this->clientError(_m('You cannot review profile flags.')); } $this->page = $this->trimmed('page'); if (empty($this->page)) { $this->page = 1; } $this->profiles = $this->getProfiles(); return true; } /** * Handle request * * @param array $args $_REQUEST args; handled in prepare() * * @return void */ public function handle() { parent::handle(); $this->showPage(); } /** * Title of this page * * @return string Title of the page */ public function title() { // TRANS: Title for page with a list of profiles that were flagged for review. return _m('Flagged profiles'); } /** * save the profile flag * * @return void */ public function showContent() { $pl = new FlaggedProfileList($this->profiles, $this); $cnt = $pl->show(); $this->pagination( $this->page > 1, $cnt > PROFILES_PER_PAGE, $this->page, 'adminprofileflag' ); } /** * Retrieve this action's profiles * * @return Profile $profile Profile query results */ public function getProfiles() { $ufp = new User_flag_profile(); $ufp->selectAdd(); $ufp->selectAdd('profile_id'); $ufp->selectAdd('count(*) as flag_count'); $ufp->whereAdd('cleared is NULL'); $ufp->groupBy('profile_id'); $ufp->orderBy('flag_count DESC, profile_id DESC'); $offset = ($this->page - 1) * PROFILES_PER_PAGE; $limit = PROFILES_PER_PAGE + 1; $ufp->limit($offset, $limit); $profiles = []; if ($ufp->find()) { while ($ufp->fetch()) { $profile = Profile::getKV('id', $ufp->profile_id); if (!empty($profile)) { $profiles[] = $profile; } } } $ufp->free(); return new ArrayWrapper($profiles); } } /** * Specialization of ProfileList to show flagging information * * Most of the hard part is done in FlaggedProfileListItem. * * @category Widget * @package GNUsocial * @author Evan Prodromou * @copyright 2009 StatusNet, Inc. * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later */ class FlaggedProfileList extends ProfileList { /** * Factory method for creating new list items * * @param Profile $profile Profile to create an item for * * @return ProfileListItem newly-created item */ public function newListItem(Profile $profile) { return new FlaggedProfileListItem($profile, $this->action); } } /** * Specialization of ProfileListItem to show flagging information * * @category Widget * @package GNUsocial * @author Evan Prodromou * @copyright 2009 StatusNet, Inc. * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later */ class FlaggedProfileListItem extends ProfileListItem { const MAX_FLAGGERS = 5; public $user; public $r2args; /** * Overload parent's action list with our own moderation-oriented buttons * * @return void */ public function showActions() { $this->user = common_current_user(); list($action, $this->r2args) = $this->out->returnToArgs(); $this->r2args['action'] = $action; $this->startActions(); if (Event::handle('StartProfileListItemActionElements', [$this])) { $this->out->elementStart('li', 'entity_moderation'); // TRANS: Header for moderation menu with action buttons for flagged profiles (like 'sandbox', 'silence', ...). $this->out->element('p', null, _m('Moderate')); $this->out->elementStart('ul'); $this->showSandboxButton(); $this->showSilenceButton(); $this->showDeleteButton(); $this->showClearButton(); $this->out->elementEnd('ul'); $this->out->elementEnd('li'); Event::handle('EndProfileListItemActionElements', [$this]); } $this->endActions(); } /** * Show a button to sandbox the profile * * @return void */ public function showSandboxButton() { if ($this->user->hasRight(Right::SANDBOXUSER)) { $this->out->elementStart('li', 'entity_sandbox'); if ($this->profile->isSandboxed()) { $usf = new UnSandboxForm($this->out, $this->profile, $this->r2args); $usf->show(); } else { $sf = new SandboxForm($this->out, $this->profile, $this->r2args); $sf->show(); } $this->out->elementEnd('li'); } } /** * Show a button to silence the profile * * @return void */ public function showSilenceButton() { if ($this->user->hasRight(Right::SILENCEUSER)) { $this->out->elementStart('li', 'entity_silence'); if ($this->profile->isSilenced()) { $usf = new UnSilenceForm($this->out, $this->profile, $this->r2args); $usf->show(); } else { $sf = new SilenceForm($this->out, $this->profile, $this->r2args); $sf->show(); } $this->out->elementEnd('li'); } } /** * Show a button to delete user and profile * * @return void */ public function showDeleteButton() { if ($this->user->hasRight(Right::DELETEUSER)) { $this->out->elementStart('li', 'entity_delete'); $df = new DeleteUserForm($this->out, $this->profile, $this->r2args); $df->show(); $this->out->elementEnd('li'); } } /** * Show a button to clear flags * * @return void */ public function showClearButton() { if ($this->user->hasRight(UserFlagPlugin::CLEARFLAGS)) { $this->out->elementStart('li', 'entity_clear'); $cf = new ClearFlagForm($this->out, $this->profile, $this->r2args); $cf->show(); $this->out->elementEnd('li'); } } /** * Overload parent function to add flaggers list * * @return void */ public function endProfile() { $this->showFlaggersList(); parent::endProfile(); } /** * Show a list of people who've flagged this profile * * @return void */ public function showFlaggersList() { $flaggers = []; $ufp = new User_flag_profile(); $ufp->selectAdd(); $ufp->selectAdd('user_id'); $ufp->profile_id = $this->profile->id; $ufp->orderBy('created, user_id'); if ($ufp->find()) { // XXX: this should always happen while ($ufp->fetch()) { $user = User::getKV('id', $ufp->user_id); if (!empty($user)) { // XXX: this would also be unusual $flaggers[] = clone $user; } } } $cnt = count($flaggers); $others = 0; if ($cnt > self::MAX_FLAGGERS) { $flaggers = array_slice($flaggers, 0, self::MAX_FLAGGERS); $others = $cnt - self::MAX_FLAGGERS; } $lnks = []; foreach ($flaggers as $flagger) { $url = common_local_url( 'showstream', ['nickname' => $flagger->nickname] ); $lnks[] = XMLStringer::estring( 'a', ['href' => $url, 'class' => 'flagger'], $flagger->nickname ); } if ($cnt > 0) { if ($others > 0) { $flagging_users = implode(', ', $lnks); // TRANS: Message displayed on a profile if it has been flagged. // TRANS: %1$s is a comma separated list of at most 5 user nicknames that flagged. // TRANS: %2$d is a positive integer of additional flagging users. Also used for plural. $text .= sprintf(_m('Flagged by %1$s and %2$d other', 'Flagged by %1$s and %2$d others', $others), $flagging_users, $others); } else { // TRANS: Message displayed on a profile if it has been flagged. // TRANS: %s is a comma separated list of at most 5 user nicknames that flagged. $text .= sprintf(_m('Flagged by %s'), $flagging_users); } $this->out->elementStart('p', ['class' => 'flaggers']); $this->out->raw($text); $this->out->elementEnd('p'); } } }