Seditio Source
Root |
./othercms/ips_4.3.4/applications/core/modules/admin/membersettings/reputation.php
<?php
/**
 * @brief        Reputation
 * @author        <a href='https://www.invisioncommunity.com'>Invision Power Services, Inc.</a>
 * @copyright    (c) Invision Power Services, Inc.
 * @license        https://www.invisioncommunity.com/legal/standards/
 * @package        Invision Community
 * @since        27 Mar 2013
 */

namespace IPS\core\modules\admin\membersettings;

/* To prevent PHP errors (extending class does not exist) revealing path */
if ( !defined( '\IPS\SUITE_UNIQUE_KEY' ) )
{
   
header( ( isset( $_SERVER['SERVER_PROTOCOL'] ) ? $_SERVER['SERVER_PROTOCOL'] : 'HTTP/1.0' ) . ' 403 Forbidden' );
    exit;
}

/**
 * reputation
 */
class _reputation extends \IPS\Node\Controller
{
   
/**
     * Node Class
     */
   
protected $nodeClass = '\IPS\Content\Reaction';
   
   
/**
     * Title can contain HTML?
     */
   
public $_titleHtml = TRUE;

   
/**
     * Execute
     *
     * @return    void
     */
   
public function execute()
    {
        \
IPS\Dispatcher::i()->checkAcpPermission( 'reps_manage' );
        return
parent::execute();
    }

   
/**
     * Manage
     *
     * @return    void
     */
   
protected function manage()
    {
        \
IPS\Output::i()->title        = \IPS\Member::loggedIn()->language()->addToStack('reputation_title');
       
       
/* Init */
       
$activeTab = \IPS\Request::i()->tab ?: NULL;
       
$activeTabContents = '';
       
$tabs = array();
       
       
/* Settings */
       
if ( \IPS\Member::loggedIn()->hasAcpRestriction( 'core', 'membersettings', 'reps_settings' ) )
        {
           
$tabs['settings'] = 'reputation_settings';
        }

       
/* Reactions */
       
if ( \IPS\Member::loggedIn()->hasAcpRestriction( 'core', 'membersettings', 'reactions_manage' ) )
        {
           
$tabs['reactions'] = 'reactions';
        }

       
/* Leaderboard */
       
if ( \IPS\Member::loggedIn()->hasAcpRestriction( 'core', 'membersettings', 'reps_settings' ) )
        {
           
$tabs['leaderboard'] = 'reputation_leaderboard';
        }
       
       
/* Levels */        
       
if ( \IPS\Settings::i()->reputation_enabled and \IPS\Settings::i()->reputation_show_profile )
        {
           
$tabs['levels'] = 'reputation_levels';
        }
       
       
/* Make sure we have a tab */
       
if ( empty( $tabs ) )
        {
            \
IPS\Output::i()->error( 'no_module_permission', '1C225/1', 403, '' );
        }
        elseif ( !
$activeTab or !array_key_exists( $activeTab, $tabs ) )
        {
           
$_tabs = array_keys( $tabs );
           
$activeTab = array_shift( $_tabs );
        }
       
       
/* Do it */
       
if ( $activeTab === 'reactions' )
        {
            \
IPS\Dispatcher::i()->checkAcpPermission( 'reactions_manage' );
           
parent::manage();
           
$activeTabContents = \IPS\Theme::i()->getTemplate( 'global' )->paddedBlock( \IPS\Output::i()->output );
        }
        else if (
$activeTab === 'settings' )
        {
            \
IPS\Dispatcher::i()->checkAcpPermission( 'reps_settings' );
       
           
/* Random for the "like" preview */
           
$maxMemberId =  \IPS\Db::i()->select( 'MAX(member_id)', 'core_members' )->first();
           
$names = array();
            foreach (
range( 1, ( $maxMemberId > 3 ) ? 3 : $maxMemberId ) as $i )
            {
                do
                {
                   
$randomMemberId = rand( 1, $maxMemberId );
                }
                while (
array_key_exists( $randomMemberId, $names ) );
                               
                try
                {
                   
$where = array( array( 'member_id>=?', $randomMemberId ) );
                    if ( !empty(
$names ) )
                    {
                       
$where[] = \IPS\Db::i()->in( 'member_id', array_keys( $names ), TRUE );
                    }
                   
                   
$member = \IPS\Member::constructFromData( \IPS\Db::i()->select( '*', 'core_members', $where, 'member_id ASC', 1 )->first() );
                   
$names[ $member->member_id ] = '<a>' . htmlentities( $member->name, ENT_DISALLOWED, 'UTF-8', FALSE ) . '</a>';
                }
                catch ( \
Exception $e )
                {
                    break;
                }
            }
            if (
count( $names ) == 3 )
            {
               
$names[] = \IPS\Member::loggedIn()->language()->addToStack( 'like_blurb_others', FALSE, array( 'pluralize' => array( 2 ) ) );
            }
            if ( empty(
$names ) )
            {
               
$blurb = '';
            }
            else
            {
               
$blurb = \IPS\Member::loggedIn()->language()->addToStack( 'like_blurb', FALSE, array( 'htmlsprintf' => array( \IPS\Member::loggedIn()->language()->formatList( $names ) ), 'pluralize' => array( count( $names ) ) ) );
            }
           
           
/* Build Form */
           
$form = new \IPS\Helpers\Form();
           
$form->add( new \IPS\Helpers\Form\YesNo( 'reputation_enabled', \IPS\Settings::i()->reputation_enabled, FALSE, array( 'togglesOn' => array( 'reputation_point_types', 'reputation_protected_groups', 'reputation_can_self_vote', 'reputation_highlight', 'reputation_show_profile', 'overall_reaction_count' ) ) ) );
           
$form->add( new \IPS\Helpers\Form\Select( 'reputation_protected_groups', explode( ',', \IPS\Settings::i()->reputation_protected_groups ), FALSE, array( 'options' => \IPS\Member\Group::groups( TRUE, FALSE ), 'parse' => 'normal', 'multiple' => TRUE ), NULL, NULL, NULL, 'reputation_protected_groups' ) );
           
$form->add( new \IPS\Helpers\Form\YesNo( 'reputation_can_self_vote', \IPS\Settings::i()->reputation_can_self_vote, FALSE, array(), NULL, NULL, NULL, 'reputation_can_self_vote' ) );
           
$form->add( new \IPS\Helpers\Form\Number( 'reputation_highlight', \IPS\Settings::i()->reputation_highlight, FALSE, array( 'unlimited' => 0, 'unlimitedLang' => 'never' ), NULL, \IPS\Member::loggedIn()->language()->addToStack('reputation_highlight_prefix'), \IPS\Member::loggedIn()->language()->addToStack('reputation_highlight_suffix'), 'reputation_highlight' ) );
           
$form->add( new \IPS\Helpers\Form\YesNo( 'reputation_show_profile', \IPS\Settings::i()->reputation_show_profile, FALSE, array(), NULL, NULL, NULL, 'reputation_show_profile' ) );
           
$form->add( new \IPS\Helpers\Form\Radio( 'reaction_count_display', \IPS\Settings::i()->reaction_count_display, FALSE, array( 'options' => array( 'individual' => 'individual_reactions', 'count' => 'overall_reaction_count' ) ) ) );
       
           
/* Save */
           
if ( $form->saveAsSettings() )
            {
               
/* Clear guest page caches */
               
\IPS\Data\Cache::i()->clearAll();

                \
IPS\Session::i()->log( 'acplogs__rep_settings' );
                \
IPS\Output::i()->redirect( \IPS\Http\Url::internal( 'app=core&module=membersettings&controller=reputation&tab=settings' ), 'saved' );
            }
           
           
/* Display */
           
$activeTabContents = (string) $form;
        }
        else if (
$activeTab === 'leaderboard' )
        {
            \
IPS\Dispatcher::i()->checkAcpPermission( 'reps_settings' );
           
           
/* Get available filters */
           
$topMemberFilters = \IPS\Member::topMembersOptions();
       
           
/* Build Form */
           
$form = new \IPS\Helpers\Form();
           
$form->add( new \IPS\Helpers\Form\YesNo( 'reputation_leaderboard_on', \IPS\Settings::i()->reputation_leaderboard_on, FALSE, array( 'togglesOn' => array( 'reputation_leaderboard_default_tab', 'form_header_leaderboard_tabs_leaderboard', 'reputation_show_days_won_trophy', 'reputation_timezone', 'form_header_leaderboard_tabs_members', 'reputation_top_members_overview', 'reputation_overview_max_members', 'reputation_top_members_filters', 'reputation_max_members' ) ), NULL, NULL, NULL, 'reputation_leaderboard_on' ) );
           
$form->add( new \IPS\Helpers\Form\Radio( 'reputation_leaderboard_default_tab', \IPS\Settings::i()->reputation_leaderboard_default_tab, FALSE, array( 'options' => array( 'leaderboard' => 'leaderboard_tabs_leaderboard', 'history' => 'leaderboard_tabs_history', 'members' => 'leaderboard_tabs_members' ) ), NULL, NULL, NULL, 'reputation_leaderboard_default_tab' ) );
           
$form->add( new \IPS\Helpers\Form\Select( 'leaderboard_excluded_groups', \IPS\Settings::i()->leaderboard_excluded_groups, FALSE, array( 'options' => \IPS\Member\Group::groups( TRUE, FALSE ), 'parse' => 'normal', 'multiple' => TRUE ), NULL, NULL, NULL, 'leaderboard_excluded_groups' ) );
           
$form->addHeader('leaderboard_tabs_leaderboard');
           
$form->add( new \IPS\Helpers\Form\YesNo( 'reputation_show_days_won_trophy', \IPS\Settings::i()->reputation_show_days_won_trophy, FALSE, array( 'togglesOn' => array() ), NULL, NULL, NULL, 'reputation_show_days_won_trophy' ) );
           
$form->add( new \IPS\Helpers\Form\Timezone( 'reputation_timezone', \IPS\Settings::i()->reputation_timezone, FALSE, array(), NULL, NULL, NULL, 'reputation_timezone' ) );
           
$form->addHeader('leaderboard_tabs_members');
           
$form->add( new \IPS\Helpers\Form\CheckboxSet( 'reputation_top_members_overview', explode( ',', \IPS\Settings::i()->reputation_top_members_overview ), FALSE, array( 'options' => $topMemberFilters ), NULL, NULL, NULL, 'reputation_max_members' ) );
           
$form->add( new \IPS\Helpers\Form\Number( 'reputation_overview_max_members', \IPS\Settings::i()->reputation_overview_max_members, FALSE, array( 'min' => 3, 'max' => 100 ), NULL, NULL, NULL, 'reputation_overview_max_members' ) );
           
$form->add( new \IPS\Helpers\Form\CheckboxSet( 'reputation_top_members_filters', \IPS\Settings::i()->reputation_top_members_filters == '*' ? array_keys( $topMemberFilters ) : explode( ',', \IPS\Settings::i()->reputation_top_members_filters ), FALSE, array( 'options' => $topMemberFilters ), NULL, NULL, NULL, 'reputation_top_members_filters' ) );
           
$form->add( new \IPS\Helpers\Form\Number( 'reputation_max_members', \IPS\Settings::i()->reputation_max_members, FALSE, array( 'min' => 3, 'max' => '100' ), NULL, NULL, NULL, 'reputation_max_members' ) );
           
           
/* Save */
           
if ( $values = $form->values() )
            {
               
/* Save */
               
$values['reputation_top_members_overview'] = implode( ',', $values['reputation_top_members_overview'] );
               
$values['reputation_top_members_filters'] = $values['reputation_top_members_filters'] == array_keys( $topMemberFilters ) ? '*' : implode( ',', $values['reputation_top_members_filters'] );                
               
$form->saveAsSettings( $values );
               
               
/* Clear guest page caches */
               
\IPS\Data\Cache::i()->clearAll();

                \
IPS\Session::i()->log( 'acplogs__rep_settings' );
                \
IPS\Output::i()->redirect( \IPS\Http\Url::internal( 'app=core&module=membersettings&controller=reputation&tab=leaderboard' ), 'saved' );
            }
           
           
/* Display */
           
$activeTabContents = (string) $form;
        }
        else
        {
           
/* Create the table */
           
$table = new \IPS\Helpers\Table\Db( 'core_reputation_levels', \IPS\Http\Url::internal( 'app=core&module=membersettings&controller=reputation&tab=levels' ) );
           
$table->langPrefix = 'rep_';
           
           
/* Columns */
           
$table->joins = array(
                array(
'select' => 'w.word_custom', 'from' => array( 'core_sys_lang_words', 'w' ), 'where' => "w.word_key=CONCAT( 'core_reputation_level_', core_reputation_levels.level_id ) AND w.lang_id=" . \IPS\Member::loggedIn()->language()->id )
            );
           
$table->include = array( 'word_custom', 'level_image', 'level_points' );
           
$table->mainColumn = 'word_custom';
           
$table->quickSearch = 'word_custom';
           
           
/* Sorting */
           
$table->noSort = array( 'level_image' );
           
$table->sortBy = $table->sortBy ?: 'level_points';
           
$table->sortDirection = $table->sortDirection ?: 'asc';
           
           
/* Parsers */
           
$table->parsers = array(
               
'level_image'    => function( $val )
                {
                    if (
$val )
                    {
                        return
"<img src='" . \IPS\File::get( "core_Theme", $val )->url . "' alt=''>";
                    }
                    return
'';
                }
            );
           
           
/* Buttons */
           
$table->rootButtons = array(
               
'add'    => array(
                   
'title'    => 'add',
                   
'icon'    => 'plus',
                   
'link'    => \IPS\Http\Url::internal( 'app=core&module=membersettings&controller=reputation&do=levelForm' ),
                   
'data'    => array( 'ipsDialog' => '', 'ipsDialog-title' => \IPS\Member::loggedIn()->language()->addToStack('add') )
                ),
            );
           
$table->rowButtons = function( $row )
            {
                return array(
                   
'edit'    => array(
                       
'title'    => 'edit',
                       
'icon'    => 'pencil',
                       
'link'    => \IPS\Http\Url::internal( 'app=core&module=membersettings&controller=reputation&do=levelForm&id=' ) . $row['level_id'],
                       
'data'    => array( 'ipsDialog' => '', 'ipsDialog-title' => \IPS\Member::loggedIn()->language()->addToStack('edit') )
                    ),
                   
'delete'    => array(
                       
'title'    => 'delete',
                       
'icon'    => 'times-circle',
                       
'link'    => \IPS\Http\Url::internal( 'app=core&module=membersettings&controller=reputation&do=deleteLevel&id=' ) . $row['level_id'],
                       
'data'    => array( 'delete' => '' )
                    ),
                );
            };
           
           
/* Display */
           
$activeTabContents = (string) $table;
        }

       
$activeTabContents = \IPS\Theme::i()->getTemplate( 'forms' )->blurb( 'rep_' . $activeTab . '_blurb', TRUE, TRUE ) . $activeTabContents;
           
       
/* Display */
       
if( \IPS\Request::i()->isAjax() )
        {
            \
IPS\Output::i()->output = $activeTabContents;
        }
        else
        {
            \
IPS\Output::i()->output = \IPS\Theme::i()->getTemplate( 'global' )->tabs( $tabs, $activeTab, $activeTabContents, \IPS\Http\Url::internal( "app=core&module=membersettings&controller=reputation" ) );
        }
    }
   
   
/**
     * Add/Edit
     *
     * @return    void
     */
   
public function levelForm()
    {
       
$current = NULL;
        if ( \
IPS\Request::i()->id )
        {
           
$current = \IPS\Db::i()->select( '*', 'core_reputation_levels', array( 'level_id=?', \IPS\Request::i()->id ) )->first();
        }
   
       
$form = new \IPS\Helpers\Form();
       
       
$form->add( new \IPS\Helpers\Form\Translatable( 'rep_level_title', NULL, TRUE, array( 'app' => 'core', 'key' => ( $current ? "core_reputation_level_{$current['level_id']}" : NULL ) ) ) );
       
$form->add( new \IPS\Helpers\Form\Upload( 'rep_level_image', ( $current and $current['level_image'] ) ? \IPS\File::get( 'core_Theme', $current['level_image'] ) : NULL, FALSE, array( 'storageExtension' => 'core_Theme' ) ) );
       
$form->add( new \IPS\Helpers\Form\Number( 'rep_level_points', $current ? $current['level_points'] : 0, TRUE, array( 'min' => NULL ) ) );
       
        if (
$values = $form->values() )
        {
           
$save = array(
               
'level_image'    => $values['rep_level_image'] ? str_replace( \IPS\ROOT_PATH . '/uploads/', '', $values['rep_level_image'] ) : '',
               
'level_points'    => $values['rep_level_points'],
            );
       
            if (
$current )
            {
                \
IPS\Db::i()->update( 'core_reputation_levels', $save, array( 'level_id=?', $current['level_id'] ) );
               
$id = $current['level_id'];
                \
IPS\Session::i()->log( 'acplogs__rep_edited', array( $save['level_points'] => FALSE ) );
            }
            else
            {
               
$id = \IPS\Db::i()->insert( 'core_reputation_levels', $save );
                \
IPS\Session::i()->log( 'acplogs__rep_edited', array( $save['level_points'] => FALSE ) );
            }
           
            unset( \
IPS\Data\Store::i()->reputationLevels );

           
/* Clear guest page caches */
           
\IPS\Data\Cache::i()->clearAll();

            \
IPS\Lang::saveCustom( 'core', "core_reputation_level_{$id}", $values['rep_level_title'] );
           
            \
IPS\Output::i()->redirect( \IPS\Http\Url::internal( 'app=core&module=membersettings&controller=reputation&tab=levels' ), 'saved' );
        }

        \
IPS\Output::i()->output = \IPS\Theme::i()->getTemplate( 'global' )->block( $current ? "core_reputation_level_{$current['level_id']}" : 'add', $form, FALSE );
    }
   
   
/**
     * Delete
     *
     * @return    void
     */
   
public function deleteLevel()
    {
        \
IPS\Dispatcher::i()->checkAcpPermission( 'reps_delete' );

       
/* Make sure the user confirmed the deletion */
       
\IPS\Request::i()->confirmedDelete();

        try
        {
           
$current = \IPS\Db::i()->select( '*', 'core_reputation_levels', array( 'level_id=?', \IPS\Request::i()->id ) )->first();
           
            \
IPS\Session::i()->log( 'acplogs__rep_deleted', array( $current['level_points'] => FALSE ) );
            \
IPS\Db::i()->delete( 'core_reputation_levels', array( 'level_id=?', \IPS\Request::i()->id ) );
            unset( \
IPS\Data\Store::i()->reputationLevels );

           
/* Clear guest page caches */
           
\IPS\Data\Cache::i()->clearAll();

            \
IPS\Lang::deleteCustom( 'core', 'core_reputation_level_' . \IPS\Request::i()->id );
        }
        catch ( \
UnderflowExceptiobn $e ) { }
               
        \
IPS\Output::i()->redirect( \IPS\Http\Url::internal( 'app=core&module=membersettings&controller=reputation&tab=levels' ) );
    }
   
   
/**
     * Rebuild leaderboard
     *
     * @return void
     */
     
public function rebuildLeaderboard()
     {
        if ( ! isset( \
IPS\Request::i()->process ) )
        {
             \
IPS\Output::i()->title = \IPS\Member::loggedIn()->language()->addToStack('reputation_leaderboard_rebuild_title');
             \
IPS\Output::i()->output = \IPS\Theme::i()->getTemplate( 'settings' )->reputationLeaderboardRebuild();
        }
        else
        {
            \
IPS\Task::queue( 'core', 'RebuildReputationLeaderboard', array(), 4 );
            \
IPS\Db::i()->delete('core_reputation_leaderboard_history');
            \
IPS\Output::i()->redirect( \IPS\Http\Url::internal( 'app=core&module=membersettings&controller=reputation&tab=leaderboard' ), 'reputation_leaderboard_rebuilding' );
        }
     }
     
     
/**
     * Function to execute after nodes are reordered. Do nothing by default but plugins can extend.
     *
     * @param    array    $order    The new ordering that was saved
     * @return    void
     */
   
protected function _afterReorder( $order )
    {
        unset( \
IPS\Data\Store::i()->reactions );
    }
   
   
/**
     * Redirect after save
     *
     * @param    \IPS\Node\Model    $old            A clone of the node as it was before or NULL if this is a creation
     * @param    \IPS\Node\Model    $new            The node now
     * @param    string            $lastUsedTab    The tab last used in the form
     * @return    void
     */
   
protected function _afterSave( \IPS\Node\Model $old = NULL, \IPS\Node\Model $new, $lastUsedTab = FALSE )
    {
        if( \
IPS\Request::i()->isAjax() )
        {
            \
IPS\Output::i()->json( array() );
        }
        else
        {
            if( isset( \
IPS\Request::i()->save_and_reload ) )
            {
               
$buttons = $new->getButtons( $this->url, !( $new instanceof $this->nodeClass ) );

                \
IPS\Output::i()->redirect( ( $lastUsedTab ? $buttons['edit']['link']->setQueryString('activeTab', $lastUsedTab ) : $buttons['edit']['link'] ), 'saved' );
            }
            else
            {
                \
IPS\Output::i()->redirect( $this->url->setQueryString( array( 'root' => ( $new->parent() ? $new->parent()->_id : '' ), 'tab' => 'reactions' ) ), 'saved' );
            }
        }
    }
   
   
/**
     * Delete
     *
     * @return    void
     */
   
protected function delete()
    {
       
/* Get node */
       
$nodeClass = $this->nodeClass;
        if ( \
IPS\Request::i()->subnode )
        {
           
$nodeClass = $nodeClass::$subnodeClass;
        }
       
        try
        {
           
$node = call_user_func( "{$nodeClass}::load", \IPS\Request::i()->id );
        }
        catch ( \
OutOfRangeException $e )
        {
            \
IPS\Output::i()->error( 'node_error', '2S101/J', 404, '' );
        }
         
       
/* Permission check */
       
if( !$node->canDelete() )
        {
            \
IPS\Output::i()->error( 'node_noperm_delete', '2S101/H', 403, '' );
        }

       
/* Do we have any children or content? */
       
$form = new \IPS\Helpers\Form;
       
$form->add( new \IPS\Helpers\Form\Radio( 'existing_reactions', NULL, TRUE, array( 'options' => array( 'change' => 'change_new_reaction', 'delete' => 'delete' ), 'toggles' => array( 'change' => array( 'new_reaction' ) ) ) ) );
       
$form->add( new \IPS\Helpers\Form\Node( 'new_reaction', NULL, TRUE, array( 'class' => 'IPS\Content\Reaction', 'permissionCheck' => function( $row ) use ( $node ) {
            if (
$row->_id == $node->_id )
            {
                return
FALSE;
            }
           
            return
TRUE;
        } ),
NULL, NULL, NULL, 'new_reaction' ) );
       
        if (
$values = $form->values() )
        {
            if (
$values['existing_reactions'] == 'change' AND array_key_exists( 'new_reaction', $values ) )
            {
                \
IPS\Db::i()->update( 'core_reputation_index', array( 'reaction' => $values['new_reaction']->_id ), array( "reaction=?", $node->_id ) );
            }
            else
            {
                \
IPS\Db::i()->delete( 'core_reputation_index', array( "reaction=?", $node->_id ) );
            }
           
           
/* Delete it */
           
\IPS\Session::i()->log( 'acplog__node_deleted', array( $this->title => TRUE, $node->titleForLog() => FALSE ) );
           
$node->delete();
   
           
/* Boink */
           
if( \IPS\Request::i()->isAjax() )
            {
                \
IPS\Output::i()->json( "OK" );
            }
            else
            {
                \
IPS\Output::i()->redirect( $this->url->setQueryString( array( 'root' => ( $node->parent() ? $node->parent()->_id : '' ) ) ), 'deleted' );
            }
        }
       
        \
IPS\Output::i()->output = $form;
    }
}