2014-05-04 00:36:20 +02:00
< ? php
namespace MCTeam\Dedimania ;
use FML\Controls\Control ;
use FML\Controls\Frame ;
use FML\Controls\Label ;
use FML\Controls\Quad ;
use FML\Controls\Quads\Quad_BgsPlayerCard ;
use FML\ManiaLink ;
use FML\Script\Features\Paging ;
use ManiaControl\Callbacks\CallbackListener ;
use ManiaControl\Callbacks\CallbackManager ;
use ManiaControl\Callbacks\Callbacks ;
2014-05-24 20:29:32 +02:00
use ManiaControl\Callbacks\Models\RecordCallback ;
2014-05-04 00:36:20 +02:00
use ManiaControl\Callbacks\TimerListener ;
use ManiaControl\Commands\CommandListener ;
use ManiaControl\ManiaControl ;
use ManiaControl\Manialinks\ManialinkManager ;
use ManiaControl\Players\Player ;
use ManiaControl\Players\PlayerManager ;
use ManiaControl\Plugins\Plugin ;
2014-05-13 16:40:05 +02:00
use ManiaControl\Utils\Formatter ;
2014-05-04 00:36:20 +02:00
/**
* ManiaControl Dedimania Plugin
*
* @ author ManiaControl Team < mail @ maniacontrol . com >
* @ copyright 2014 ManiaControl Team
* @ license http :// www . gnu . org / licenses / GNU General Public License , Version 3
*/
class DedimaniaPlugin implements CallbackListener , CommandListener , TimerListener , Plugin {
/*
* Constants
*/
const ID = 8 ;
const VERSION = 0.1 ;
const AUTHOR = 'MCTeam' ;
const NAME = 'Dedimania Plugin' ;
const MLID_DEDIMANIA = 'Dedimania.ManialinkId' ;
const XMLRPC_MULTICALL = 'system.multicall' ;
const DEDIMANIA_URL = 'http://dedimania.net:8081/Dedimania' ;
const DEDIMANIA_OPENSESSION = 'dedimania.OpenSession' ;
const DEDIMANIA_CHECKSESSION = 'dedimania.CheckSession' ;
const DEDIMANIA_GETRECORDS = 'dedimania.GetChallengeRecords' ;
const DEDIMANIA_PLAYERCONNECT = 'dedimania.PlayerConnect' ;
const DEDIMANIA_PLAYERDISCONNECT = 'dedimania.PlayerDisconnect' ;
const DEDIMANIA_UPDATESERVERPLAYERS = 'dedimania.UpdateServerPlayers' ;
const DEDIMANIA_SETCHALLENGETIMES = 'dedimania.SetChallengeTimes' ;
const DEDIMANIA_WARNINGSANDTTR2 = 'dedimania.WarningsAndTTR2' ;
const SETTING_WIDGET_ENABLE = 'Enable Dedimania Widget' ;
const SETTING_WIDGET_TITLE = 'Widget Title' ;
const SETTING_WIDGET_POSX = 'Widget Position: X' ;
const SETTING_WIDGET_POSY = 'Widget Position: Y' ;
const SETTING_WIDGET_WIDTH = 'Widget Width' ;
const SETTING_WIDGET_LINESCOUNT = 'Widget Displayed Lines Count' ;
const SETTING_WIDGET_LINEHEIGHT = 'Widget Line Height' ;
const SETTING_DEDIMANIA_CODE = '$l[http://dedimania.net/tm2stats/?do=register]Dedimania Code for ' ;
const CB_DEDIMANIA_CHANGED = 'Dedimania.Changed' ;
const CB_DEDIMANIA_UPDATED = 'Dedimania.Updated' ;
const ACTION_SHOW_DEDIRECORDSLIST = 'Dedimania.ShowDediRecordsList' ;
/*
* Private Properties
*/
/** @var ManiaControl $maniaControl */
private $maniaControl = null ;
/** @var DedimaniaData $dedimaniaData */
private $dedimaniaData = null ;
private $updateManialink = false ;
private $checkpoints = array ();
private $init = false ;
/**
* @ see \ManiaControl\Plugins\Plugin :: prepare ()
*/
public static function prepare ( ManiaControl $maniaControl ) {
$servers = $maniaControl -> server -> getAllServers ();
foreach ( $servers as $server ) {
$maniaControl -> settingManager -> initSetting ( get_class (), self :: SETTING_DEDIMANIA_CODE . $server -> login . '$l' , '' );
}
}
/**
* @ see \ManiaControl\Plugins\Plugin :: getId ()
*/
public static function getId () {
return self :: ID ;
}
/**
* @ see \ManiaControl\Plugins\Plugin :: getName ()
*/
public static function getName () {
return self :: NAME ;
}
/**
* @ see \ManiaControl\Plugins\Plugin :: getVersion ()
*/
public static function getVersion () {
return self :: VERSION ;
}
/**
* @ see \ManiaControl\Plugins\Plugin :: getAuthor ()
*/
public static function getAuthor () {
return self :: AUTHOR ;
}
/**
* @ see \ManiaControl\Plugins\Plugin :: getDescription ()
*/
public static function getDescription () {
2014-05-24 20:19:41 +02:00
return " Dedimania Plugin for TrackMania " ;
2014-05-04 00:36:20 +02:00
}
/**
* @ see \ManiaControl\Plugins\Plugin :: load ()
*/
public function load ( ManiaControl $maniaControl ) {
$this -> maniaControl = $maniaControl ;
if ( ! extension_loaded ( 'xmlrpc' )) {
throw new \Exception ( " You need to activate the PHP extension xmlrpc to run this Plugin! " );
}
$this -> maniaControl -> settingManager -> initSetting ( $this , self :: SETTING_WIDGET_ENABLE , true );
$this -> maniaControl -> settingManager -> initSetting ( $this , self :: SETTING_WIDGET_TITLE , 'Dedimania' );
$this -> maniaControl -> settingManager -> initSetting ( $this , self :: SETTING_WIDGET_POSX , - 139 );
$this -> maniaControl -> settingManager -> initSetting ( $this , self :: SETTING_WIDGET_POSY , 7 );
$this -> maniaControl -> settingManager -> initSetting ( $this , self :: SETTING_WIDGET_WIDTH , 40 );
$this -> maniaControl -> settingManager -> initSetting ( $this , self :: SETTING_WIDGET_LINEHEIGHT , 4 );
$this -> maniaControl -> settingManager -> initSetting ( $this , self :: SETTING_WIDGET_LINESCOUNT , 12 );
$this -> maniaControl -> callbackManager -> registerCallbackListener ( Callbacks :: BEGINMAP , $this , 'handleBeginMap' );
$this -> maniaControl -> callbackManager -> registerCallbackListener ( Callbacks :: ENDMAP , $this , 'handleMapEnd' );
2014-05-24 20:29:32 +02:00
$this -> maniaControl -> callbackManager -> registerCallbackListener ( CallbackManager :: CB_MP_PLAYERMANIALINKPAGEANSWER , $this , 'handleManialinkPageAnswer' );
2014-05-04 00:36:20 +02:00
$this -> maniaControl -> callbackManager -> registerCallbackListener ( PlayerManager :: CB_PLAYERCONNECT , $this , 'handlePlayerConnect' );
$this -> maniaControl -> callbackManager -> registerCallbackListener ( PlayerManager :: CB_PLAYERDISCONNECT , $this , 'handlePlayerDisconnect' );
2014-05-24 20:29:32 +02:00
$this -> maniaControl -> callbackManager -> registerCallbackListener ( RecordCallback :: CHECKPOINT , $this , 'handleCheckpointCallback' );
$this -> maniaControl -> callbackManager -> registerCallbackListener ( RecordCallback :: LAPFINISH , $this , 'handleLapFinishCallback' );
$this -> maniaControl -> callbackManager -> registerCallbackListener ( RecordCallback :: FINISH , $this , 'handleFinishCallback' );
2014-05-04 00:36:20 +02:00
$this -> maniaControl -> timerManager -> registerTimerListening ( $this , 'updateEverySecond' , 1000 );
$this -> maniaControl -> timerManager -> registerTimerListening ( $this , 'handleEveryMinute' , 1000 * 60 );
$this -> maniaControl -> timerManager -> registerTimerListening ( $this , 'updatePlayerList' , 1000 * 60 * 3 );
2014-05-24 20:19:41 +02:00
2014-05-04 00:36:20 +02:00
$this -> maniaControl -> commandManager -> registerCommandListener ( array ( 'dedirecs' , 'dedirecords' ), $this , 'showDediRecordsList' , false , 'Shows a list of Dedimania records of the current map.' );
// Open session
$serverInfo = $this -> maniaControl -> server -> getInfo ();
$serverVersion = $this -> maniaControl -> client -> getVersion ();
$packMask = $this -> maniaControl -> server -> titleId ;
if ( $packMask != 'Trackmania_2@nadeolabs' ) {
$packMask = substr ( $this -> maniaControl -> server -> titleId , 2 );
}
2014-05-13 16:03:26 +02:00
$dedimaniaCode = $this -> maniaControl -> settingManager -> getSettingValue ( $this , self :: SETTING_DEDIMANIA_CODE . $serverInfo -> login . '$l' );
2014-05-24 20:19:41 +02:00
if ( ! $dedimaniaCode ) {
2014-05-04 00:36:20 +02:00
throw new \Exception ( " No Dedimania Code Specified, check the settings! " );
}
$this -> dedimaniaData = new DedimaniaData ( $serverInfo -> login , $dedimaniaCode , $serverInfo -> path , $packMask , $serverVersion );
$this -> openDedimaniaSession ();
}
/**
* Opens the Dedimania Session
*/
private function openDedimaniaSession () {
$content = $this -> encode_request ( self :: DEDIMANIA_OPENSESSION , array ( $this -> dedimaniaData -> toArray ()));
2014-05-24 20:19:41 +02:00
$this -> maniaControl -> fileReader -> postData ( self :: DEDIMANIA_URL , function ( $data , $error ) {
$this -> maniaControl -> log ( " Try to connect on Dedimania " );
2014-05-04 00:36:20 +02:00
2014-05-09 18:42:17 +02:00
if ( ! $data || $error ) {
2014-05-24 20:19:41 +02:00
$this -> maniaControl -> log ( " Dedimania Error: ' { $error } ' " );
2014-05-04 00:36:20 +02:00
}
2014-05-24 20:19:41 +02:00
$data = $this -> decode ( $data );
2014-05-04 00:36:20 +02:00
if ( is_array ( $data )) {
foreach ( $data as $index => $methodResponse ) {
if ( xmlrpc_is_fault ( $methodResponse )) {
2014-05-24 20:19:41 +02:00
$this -> handleXmlRpcFault ( $methodResponse , self :: DEDIMANIA_OPENSESSION );
2014-05-04 00:36:20 +02:00
} else if ( $index <= 0 ) {
$responseData = $methodResponse [ 0 ];
2014-05-24 20:19:41 +02:00
$this -> dedimaniaData -> sessionId = $responseData [ 'SessionId' ];
if ( $this -> dedimaniaData -> sessionId ) {
$this -> maniaControl -> log ( " Dedimania connection successfully established. " );
$this -> fetchDedimaniaRecords ();
$this -> init = true ;
2014-05-04 00:36:20 +02:00
} else {
2014-05-24 20:19:41 +02:00
$this -> maniaControl -> log ( " Error while opening Dedimania Connection " );
2014-05-04 00:36:20 +02:00
}
}
}
}
}, $content , true );
}
/**
* Encode the given xml rpc method and params
*
* @ param string $method
* @ param array $params
* @ return string
*/
private function encode_request ( $method , $params ) {
$paramArray = array ( array ( 'methodName' => $method , 'params' => $params ), array ( 'methodName' => self :: DEDIMANIA_WARNINGSANDTTR2 , 'params' => array ()));
return xmlrpc_encode_request ( self :: XMLRPC_MULTICALL , array ( $paramArray ), array ( 'encoding' => 'UTF-8' , 'escaping' => 'markup' ));
}
/**
* Decodes xml rpc response
*
* @ param string $response
* @ return mixed
*/
private function decode ( $response ) {
return xmlrpc_decode ( $response , 'utf-8' );
}
/**
* Handle xml rpc fault
*
2014-05-13 16:40:05 +02:00
* @ param array $fault
* @ param string $method
2014-05-04 00:36:20 +02:00
*/
2014-05-13 16:40:05 +02:00
private function handleXmlRpcFault ( array $fault , $method ) {
trigger_error ( " XmlRpc Fault on ' { $method } ': ' { $fault [ 'faultString' ] } ( { $fault [ 'faultCode' ] } )! " );
2014-05-04 00:36:20 +02:00
}
/**
* Fetch Dedimania Records
*
* @ param bool $reset
2014-05-09 17:31:29 +02:00
* @ return bool
2014-05-04 00:36:20 +02:00
*/
private function fetchDedimaniaRecords ( $reset = true ) {
if ( ! $this -> dedimaniaData || $this -> dedimaniaData -> sessionId == '' ) {
return false ;
}
// Reset records
if ( $reset ) {
$this -> dedimaniaData -> records = array ();
}
$serverInfo = $this -> getServerInfo ();
$playerInfo = $this -> getPlayerList ();
$mapInfo = $this -> getMapInfo ();
$gameMode = $this -> getGameModeString ();
if ( ! $serverInfo || ! $playerInfo || ! $mapInfo || ! $gameMode ) {
return false ;
}
$data = array ( $this -> dedimaniaData -> sessionId , $mapInfo , $gameMode , $serverInfo , $playerInfo );
$content = $this -> encode_request ( self :: DEDIMANIA_GETRECORDS , $data );
$self = $this ;
$this -> maniaControl -> fileReader -> postData ( self :: DEDIMANIA_URL , function ( $data , $error ) use ( & $self ) {
if ( $error != '' ) {
$self -> maniaControl -> log ( " Dedimania Error: " . $error );
}
$data = $self -> decode ( $data );
if ( is_array ( $data )) {
foreach ( $data as $index => $methodResponse ) {
if ( xmlrpc_is_fault ( $methodResponse )) {
2014-05-04 14:40:27 +02:00
$self -> handleXmlRpcFault ( $methodResponse , self :: DEDIMANIA_GETRECORDS );
2014-05-04 00:36:20 +02:00
return false ;
} else if ( $index <= 0 ) {
$responseData = $methodResponse [ 0 ];
$self -> dedimaniaData -> serverMaxRank = $responseData [ 'ServerMaxRank' ];
foreach ( $responseData [ 'Players' ] as $player ) {
2014-05-24 20:19:41 +02:00
$dediPlayer = new DedimaniaPlayer ( $player );
2014-05-04 00:36:20 +02:00
$self -> dedimaniaData -> addPlayer ( $dediPlayer );
}
foreach ( $responseData [ 'Records' ] as $key => $record ) {
$self -> dedimaniaData -> records [ $key ] = new RecordData ( $record );
}
}
}
}
$self -> updateManialink = true ;
2014-05-04 14:40:27 +02:00
$self -> maniaControl -> callbackManager -> triggerCallback ( self :: CB_DEDIMANIA_UPDATED , $self -> dedimaniaData -> records );
2014-05-04 00:36:20 +02:00
return true ;
}, $content , true );
return true ;
}
/**
* Build server info Structure for callbacks
*/
private function getServerInfo () {
$server = $this -> maniaControl -> client -> getServerOptions ();
if ( ! $server ) {
return null ;
}
2014-05-27 22:44:22 +02:00
if ( $this -> maniaControl -> playerManager -> getPlayerCount ( false ) <= 0 ) {
2014-05-04 00:36:20 +02:00
return null ;
}
$playerCount = $this -> maniaControl -> playerManager -> getPlayerCount ();
$spectatorCount = $this -> maniaControl -> playerManager -> getSpectatorCount ();
return array ( 'SrvName' => $server -> name , 'Comment' => $server -> comment , 'Private' => ( strlen ( $server -> password ) > 0 ), 'NumPlayers' => $playerCount , 'MaxPlayers' => $server -> currentMaxPlayers , 'NumSpecs' => $spectatorCount , 'MaxSpecs' => $server -> currentMaxSpectators );
}
/**
* Build simple player list for callbacks
*/
private function getPlayerList () {
$players = $this -> maniaControl -> playerManager -> getPlayers ();
if ( count ( $players ) == 0 ) {
return null ;
}
$playerInfo = array ();
foreach ( $players as $player ) {
array_push ( $playerInfo , array ( 'Login' => $player -> login , 'IsSpec' => $player -> isSpectator ));
}
return $playerInfo ;
}
/**
2014-05-13 18:26:38 +02:00
* Build Map Info Array for Dedimania Requests
*
* @ return array
2014-05-04 00:36:20 +02:00
*/
private function getMapInfo () {
$map = $this -> maniaControl -> mapManager -> getCurrentMap ();
if ( ! $map ) {
return null ;
}
$mapInfo = array ();
$mapInfo [ 'UId' ] = $map -> uid ;
$mapInfo [ 'Name' ] = $map -> rawName ;
$mapInfo [ 'Author' ] = $map -> authorLogin ;
$mapInfo [ 'Environment' ] = $map -> environment ;
$mapInfo [ 'NbCheckpoints' ] = $map -> nbCheckpoints ;
$mapInfo [ 'NbLaps' ] = $map -> nbLaps ;
return $mapInfo ;
}
/**
* Get Dedimania string representation of the current game mode
*
* @ return String
*/
private function getGameModeString () {
2014-05-11 16:02:29 +02:00
$gameMode = $this -> maniaControl -> server -> getGameMode ();
2014-05-04 00:36:20 +02:00
$scriptNameResponse = $this -> maniaControl -> client -> getScriptName ();
$scriptName = str_replace ( '.Script.txt' , '' , $scriptNameResponse [ " CurrentValue " ]);
if ( $gameMode === null ) {
trigger_error ( " Couldn't retrieve game mode. " );
return null ;
}
switch ( $gameMode ) {
case 0 :
{
if ( $scriptName == 'Rounds' || $scriptName == 'Cup' || $scriptName == 'Team' ) {
return 'Rounds' ;
} else if ( $scriptName == 'TimeAttack' || $scriptName == 'Laps' || $scriptName == 'TeamAttack' || $scriptName == 'TimeAttackPlus' ) {
return 'TA' ;
}
break ;
}
case 1 :
case 3 :
case 5 :
{
return 'Rounds' ;
}
case 2 :
case 4 :
{
return 'TA' ;
}
}
return null ;
}
/**
* Handle 1 Second callback
*/
public function updateEverySecond ( $time ) {
if ( ! $this -> updateManialink ) {
return ;
}
if ( ! $this -> dedimaniaData -> records ) {
return ;
}
$this -> updateManialink = false ;
2014-05-13 16:03:26 +02:00
if ( $this -> maniaControl -> settingManager -> getSettingValue ( $this , self :: SETTING_WIDGET_ENABLE )) {
2014-05-04 00:36:20 +02:00
$manialink = $this -> buildManialink ();
$this -> maniaControl -> manialinkManager -> sendManialink ( $manialink );
}
}
/**
* Build Manialink
*
* @ return \FML\ManiaLink
*/
private function buildManialink () {
if ( ! $this -> dedimaniaData -> records ) {
return null ;
}
$records = $this -> dedimaniaData -> records ;
2014-05-13 16:03:26 +02:00
$title = $this -> maniaControl -> settingManager -> getSettingValue ( $this , self :: SETTING_WIDGET_TITLE );
$pos_x = $this -> maniaControl -> settingManager -> getSettingValue ( $this , self :: SETTING_WIDGET_POSX );
$pos_y = $this -> maniaControl -> settingManager -> getSettingValue ( $this , self :: SETTING_WIDGET_POSY );
$width = $this -> maniaControl -> settingManager -> getSettingValue ( $this , self :: SETTING_WIDGET_WIDTH );
$lines = $this -> maniaControl -> settingManager -> getSettingValue ( $this , self :: SETTING_WIDGET_LINESCOUNT );
$lineHeight = $this -> maniaControl -> settingManager -> getSettingValue ( $this , self :: SETTING_WIDGET_LINEHEIGHT );
2014-05-04 00:36:20 +02:00
$labelStyle = $this -> maniaControl -> manialinkManager -> styleManager -> getDefaultLabelStyle ();
$quadStyle = $this -> maniaControl -> manialinkManager -> styleManager -> getDefaultQuadStyle ();
$quadSubstyle = $this -> maniaControl -> manialinkManager -> styleManager -> getDefaultQuadSubstyle ();
$manialink = new ManiaLink ( self :: MLID_DEDIMANIA );
$frame = new Frame ();
$manialink -> add ( $frame );
$frame -> setPosition ( $pos_x , $pos_y );
$backgroundQuad = new Quad ();
$frame -> add ( $backgroundQuad );
$backgroundQuad -> setVAlign ( Control :: TOP );
$height = 7. + $lines * $lineHeight ;
$backgroundQuad -> setSize ( $width * 1.05 , $height );
$backgroundQuad -> setStyles ( $quadStyle , $quadSubstyle );
$titleLabel = new Label ();
$frame -> add ( $titleLabel );
$titleLabel -> setPosition ( 0 , $lineHeight * - 0.9 );
$titleLabel -> setWidth ( $width );
$titleLabel -> setStyle ( $labelStyle );
$titleLabel -> setTextSize ( 2 );
$titleLabel -> setText ( $title );
$titleLabel -> setTranslate ( true );
foreach ( $records as $index => $record ) {
/** @var RecordData $record */
if ( $index >= $lines ) {
break ;
}
$y = - 8. - $index * $lineHeight ;
$recordFrame = new Frame ();
$frame -> add ( $recordFrame );
$recordFrame -> setPosition ( 0 , $y );
/* $backgroundQuad = new Quad ();
$recordFrame -> add ( $backgroundQuad );
$backgroundQuad -> setSize ( $width * 1.04 , $lineHeight * 1.4 );
$backgroundQuad -> setStyles ( $quadStyle , $quadSubstyle ); */
//Rank
$rankLabel = new Label ();
$recordFrame -> add ( $rankLabel );
$rankLabel -> setHAlign ( Control :: LEFT );
$rankLabel -> setX ( $width * - 0.47 );
$rankLabel -> setSize ( $width * 0.06 , $lineHeight );
$rankLabel -> setTextSize ( 1 );
$rankLabel -> setTextPrefix ( '$o' );
$rankLabel -> setText ( $record -> rank );
$rankLabel -> setTextEmboss ( true );
//Name
$nameLabel = new Label ();
$recordFrame -> add ( $nameLabel );
$nameLabel -> setHAlign ( Control :: LEFT );
$nameLabel -> setX ( $width * - 0.4 );
$nameLabel -> setSize ( $width * 0.6 , $lineHeight );
$nameLabel -> setTextSize ( 1 );
$nameLabel -> setText ( $record -> nickName );
$nameLabel -> setTextEmboss ( true );
//Time
$timeLabel = new Label ();
$recordFrame -> add ( $timeLabel );
$timeLabel -> setHAlign ( Control :: RIGHT );
$timeLabel -> setX ( $width * 0.47 );
$timeLabel -> setSize ( $width * 0.25 , $lineHeight );
$timeLabel -> setTextSize ( 1 );
$timeLabel -> setText ( Formatter :: formatTime ( $record -> best ));
$timeLabel -> setTextEmboss ( true );
}
return $manialink ;
}
/**
2014-05-13 16:40:05 +02:00
* Handle 1 Minute Callback
2014-05-04 00:36:20 +02:00
*/
2014-05-13 16:40:05 +02:00
public function handleEveryMinute () {
2014-05-04 00:36:20 +02:00
if ( ! $this -> init ) {
return ;
}
$this -> checkDedimaniaSession ();
}
/**
* Checks If a Dedimania Session exists , if not create a new oen
*/
private function checkDedimaniaSession () {
2014-05-24 20:19:41 +02:00
if ( ! $this -> dedimaniaData -> sessionId ) {
2014-05-04 00:36:20 +02:00
$this -> openDedimaniaSession ();
return ;
}
$content = $this -> encode_request ( self :: DEDIMANIA_CHECKSESSION , array ( $this -> dedimaniaData -> sessionId ));
$self = $this ;
$this -> maniaControl -> fileReader -> postData ( self :: DEDIMANIA_URL , function ( $data , $error ) use ( & $self ) {
if ( $error != '' ) {
$self -> maniaControl -> log ( " Dedimania Error: " . $error );
}
$data = $self -> decode ( $data );
if ( is_array ( $data )) {
foreach ( $data as $methodResponse ) {
if ( xmlrpc_is_fault ( $methodResponse )) {
2014-05-04 14:40:27 +02:00
$self -> handleXmlRpcFault ( $methodResponse , self :: DEDIMANIA_CHECKSESSION );
2014-05-04 00:36:20 +02:00
} else {
$responseData = $methodResponse [ 0 ];
if ( is_bool ( $responseData )) {
if ( ! $responseData ) {
$self -> openDedimaniaSession ();
}
}
}
}
}
}, $content , true );
return ;
}
/**
* Handle PlayerConnect callback
*
2014-05-24 20:19:41 +02:00
* @ param Player $player
2014-05-04 00:36:20 +02:00
*/
public function handlePlayerConnect ( Player $player ) {
// Send Dedimania request
$data = array ( $this -> dedimaniaData -> sessionId , $player -> login , $player -> rawNickname , $player -> path , $player -> isSpectator );
$content = $this -> encode_request ( self :: DEDIMANIA_PLAYERCONNECT , $data );
$self = $this ;
$this -> maniaControl -> fileReader -> postData ( self :: DEDIMANIA_URL , function ( $data , $error ) use ( & $self , & $player ) {
2014-05-24 20:19:41 +02:00
if ( $error ) {
2014-05-04 00:36:20 +02:00
$self -> maniaControl -> log ( " Dedimania Error: " . $error );
}
$data = $self -> decode ( $data );
if ( is_array ( $data )) {
foreach ( $data as $index => $methodResponse ) {
if ( xmlrpc_is_fault ( $methodResponse )) {
2014-05-04 14:40:27 +02:00
$self -> handleXmlRpcFault ( $methodResponse , self :: DEDIMANIA_PLAYERCONNECT );
2014-05-04 00:36:20 +02:00
} else if ( $index <= 0 ) {
$responseData = $methodResponse [ 0 ];
2014-05-24 20:19:41 +02:00
$dediPlayer = new DedimaniaPlayer ( $responseData );
$self -> dedimaniaData -> addPlayer ( $dediPlayer );
2014-05-04 00:36:20 +02:00
2014-05-24 20:19:41 +02:00
// Fetch records if he is the first who joined the server
if ( $self -> maniaControl -> playerManager -> getPlayerCount ( false ) === 1 ) {
2014-05-04 00:36:20 +02:00
$self -> fetchDedimaniaRecords ( true );
}
}
2014-05-13 16:03:26 +02:00
if ( $self -> maniaControl -> settingManager -> getSettingValue ( $self , self :: SETTING_WIDGET_ENABLE )) {
2014-05-04 00:36:20 +02:00
$manialink = $self -> buildManialink ();
$self -> maniaControl -> manialinkManager -> sendManialink ( $manialink , $player -> login );
}
}
} else {
if ( ! $data ) {
trigger_error ( 'XmlRpc Error.' );
var_dump ( $data );
}
}
return true ;
}, $content , true );
}
/**
* Handle PlayerDisconnect callback
*
* @ param \ManiaControl\Players\Player $player
*/
public function handlePlayerDisconnect ( Player $player ) {
$this -> dedimaniaData -> removePlayer ( $player -> login );
// Send Dedimania request
$data = array ( $this -> dedimaniaData -> sessionId , $player -> login , '' );
$content = $this -> encode_request ( self :: DEDIMANIA_PLAYERDISCONNECT , $data );
$self = $this ;
$this -> maniaControl -> fileReader -> postData ( self :: DEDIMANIA_URL , function ( $data , $error ) use ( & $self ) {
if ( $error != '' ) {
$self -> maniaControl -> log ( " Dedimania Error: " . $error );
}
$data = $self -> decode ( $data );
if ( is_array ( $data )) {
foreach ( $data as $methodResponse ) {
if ( xmlrpc_is_fault ( $methodResponse )) {
2014-05-04 14:40:27 +02:00
$self -> handleXmlRpcFault ( $methodResponse , self :: DEDIMANIA_PLAYERDISCONNECT );
2014-05-04 00:36:20 +02:00
}
}
} else {
if ( ! $data ) {
trigger_error ( 'XmlRpc Error.' );
var_dump ( $data );
}
}
return true ;
}, $content , true );
}
/**
2014-05-13 16:40:05 +02:00
* Handle Begin Map Callback
2014-05-04 00:36:20 +02:00
*/
2014-05-13 16:40:05 +02:00
public function handleBeginMap () {
2014-05-04 00:36:20 +02:00
unset ( $this -> dedimaniaData -> records );
$this -> fetchDedimaniaRecords ( true );
}
/**
2014-05-13 16:40:05 +02:00
* Handle EndMap Callback
2014-05-04 00:36:20 +02:00
*/
2014-05-13 16:40:05 +02:00
public function handleMapEnd () {
2014-05-04 00:36:20 +02:00
if ( ! $this -> dedimaniaData || ! $this -> dedimaniaData -> records ) {
return ;
}
// Send dedimania records
$gameMode = $this -> getGameModeString ();
$times = array ();
$replays = array ();
foreach ( $this -> dedimaniaData -> records as $record ) {
/** @var RecordData $record */
if ( $record -> rank > $this -> dedimaniaData -> serverMaxRank ) {
break ;
}
if ( $record -> newRecord == false ) {
continue ;
}
array_push ( $times , array ( 'Login' => $record -> login , 'Best' => $record -> best , 'Checks' => $record -> checkpoints ));
if ( ! isset ( $replays [ 'VReplay' ])) {
$replays [ 'VReplay' ] = $record -> vReplay ;
}
if ( ! isset ( $replays [ 'Top1GReplay' ])) {
$replays [ 'Top1GReplay' ] = $record -> top1GReplay ;
}
if ( ! isset ( $replays [ 'VReplayChecks' ])) {
$replays [ 'VReplayChecks' ] = '' ;
// TODO: VReplayChecks
}
}
xmlrpc_set_type ( $replays [ 'VReplay' ], 'base64' );
xmlrpc_set_type ( $replays [ 'Top1GReplay' ], 'base64' );
//var_dump($replays);
$data = array ( $this -> dedimaniaData -> sessionId , $this -> getMapInfo (), $gameMode , $times , $replays );
//var_dump($data);
$content = $this -> encode_request ( self :: DEDIMANIA_SETCHALLENGETIMES , $data );
$self = $this ;
$maniaControl = $this -> maniaControl ;
$this -> maniaControl -> fileReader -> postData ( self :: DEDIMANIA_URL , function ( $data , $error ) use ( & $self , & $maniaControl ) {
if ( $error != '' ) {
$maniaControl -> log ( " Dedimania Error: " . $error );
}
$data = $self -> decode ( $data );
if ( is_array ( $data )) {
foreach ( $data as $index => $methodResponse ) {
if ( xmlrpc_is_fault ( $methodResponse )) {
2014-05-04 14:40:27 +02:00
$self -> handleXmlRpcFault ( $methodResponse , self :: DEDIMANIA_SETCHALLENGETIMES );
2014-05-04 00:36:20 +02:00
} else {
if ( $index <= 0 ) {
// Called method response
$responseData = $methodResponse [ 0 ];
if ( ! $responseData ) {
trigger_error ( " Records Plugin: Submitting dedimania records failed. " );
}
continue ;
}
}
}
}
}, $content , false );
}
/**
2014-05-13 16:40:05 +02:00
* Update the PlayerList every 3 Minutes
2014-05-04 00:36:20 +02:00
*/
2014-05-13 16:40:05 +02:00
public function updatePlayerList () {
2014-05-04 00:36:20 +02:00
$serverInfo = $this -> getServerInfo ();
$playerList = $this -> getPlayerList ();
$votesInfo = $this -> getVotesInfo ();
if ( ! $serverInfo || ! $votesInfo || ! $playerList || ! isset ( $this -> dedimaniaData ) || $this -> dedimaniaData -> sessionId == '' ) {
return ;
}
// Send Dedimania request
$data = array ( $this -> dedimaniaData -> sessionId , $serverInfo , $votesInfo , $playerList );
$content = $this -> encode_request ( self :: DEDIMANIA_UPDATESERVERPLAYERS , $data );
$self = $this ;
$this -> maniaControl -> fileReader -> postData ( self :: DEDIMANIA_URL , function ( $data , $error ) use ( & $self ) {
if ( $error != '' ) {
$self -> maniaControl -> log ( " Dedimania Error: " . $error );
}
$data = $self -> decode ( $data );
if ( is_array ( $data )) {
foreach ( $data as $methodResponse ) {
if ( xmlrpc_is_fault ( $methodResponse )) {
2014-05-04 14:40:27 +02:00
$self -> handleXmlRpcFault ( $methodResponse , self :: DEDIMANIA_UPDATESERVERPLAYERS );
2014-05-04 00:36:20 +02:00
}
}
} else {
if ( ! $data ) {
trigger_error ( 'XmlRpc Error.' );
var_dump ( $data );
}
}
return true ;
}, $content , true );
}
/**
2014-05-13 16:03:26 +02:00
* Build Votes Info Array for Callbacks
2014-05-04 00:36:20 +02:00
*/
private function getVotesInfo () {
$map = $this -> maniaControl -> mapManager -> getCurrentMap ();
if ( ! $map ) {
return null ;
}
$gameMode = $this -> getGameModeString ();
if ( ! $gameMode ) {
return null ;
}
return array ( 'UId' => $map -> uid , 'GameMode' => $gameMode );
}
/**
2014-05-24 20:29:32 +02:00
* Handle Checkpoint Callback
2014-05-04 00:36:20 +02:00
*
2014-05-24 20:29:32 +02:00
* @ param RecordCallback $callback
*/
public function handleCheckpointCallback ( RecordCallback $callback ) {
if ( $callback -> isLegacyCallback ) {
return ;
}
if ( ! isset ( $this -> checkpoints [ $callback -> login ])) {
$this -> checkpoints [ $callback -> login ] = array ();
}
$this -> checkpoints [ $callback -> login ][ $callback -> lapCheckpoint ] = $callback -> lapTime ;
2014-05-04 00:36:20 +02:00
}
/**
2014-05-24 20:29:32 +02:00
* Handle LapFinish Callback
2014-05-04 00:36:20 +02:00
*
2014-05-24 20:29:32 +02:00
* @ param RecordCallback $callback
2014-05-04 00:36:20 +02:00
*/
2014-05-24 20:29:32 +02:00
public function handleLapFinishCallback ( RecordCallback $callback ) {
$this -> handleFinishCallback ( $callback );
}
/**
* Handle Finish Callback
*
* @ param RecordCallback $callback
*/
public function handleFinishCallback ( RecordCallback $callback ) {
if ( $callback -> isLegacyCallback ) {
return ;
}
if ( $callback -> time <= 0 ) {
// Invalid time
2014-05-04 00:36:20 +02:00
return ;
}
2014-05-24 20:29:32 +02:00
$map = $this -> maniaControl -> mapManager -> getCurrentMap ();
2014-05-04 00:36:20 +02:00
if ( ! $map ) {
return ;
}
2014-05-24 20:29:32 +02:00
$ oldRecord = $this -> getDedimaniaRecord ( $callback -> login );
if ( $oldRecord -> nullRecord || $oldRecord && $oldRecord -> best > $callback -> lapTime ) {
2014-05-04 00:36:20 +02:00
// Save time
$newRecord = new RecordData ( null );
2014-05-24 20:29:32 +02:00
$newRecord -> constructNewRecord ( $callback -> login , $callback -> player -> nickname , $callback -> lapTime , $this -> getCheckpoints ( $callback -> login ), true );
2014-05-04 00:36:20 +02:00
if ( $this -> insertDedimaniaRecord ( $newRecord , $oldRecord )) {
// Get newly saved record
foreach ( $this -> dedimaniaData -> records as & $record ) {
/** @var RecordData $record */
if ( $record -> login !== $newRecord -> login ) {
continue ;
}
$newRecord = $record ;
break ;
}
$this -> maniaControl -> callbackManager -> triggerCallback ( self :: CB_DEDIMANIA_CHANGED , $newRecord );
// Announce record
if ( $oldRecord -> nullRecord || $newRecord -> rank < $oldRecord -> rank ) {
// Gained rank
$improvement = 'gained the' ;
} else {
// Only improved time
$improvement = 'improved his/her' ;
}
2014-05-24 20:29:32 +02:00
$message = '$390$<$fff' . $callback -> player -> nickname . '$> ' . $improvement . ' $<$ff0' . $newRecord -> rank . '.$> Dedimania Record: $<$fff' . Formatter :: formatTime ( $newRecord -> best ) . '$>' ;
2014-05-04 00:36:20 +02:00
if ( ! $oldRecord -> nullRecord ) {
2014-05-24 20:29:32 +02:00
$message .= ' ($<$ff0' . $oldRecord -> rank . '.$> $<$fff-' . Formatter :: formatTime (( $oldRecord -> best - $callback -> lapTime )) . '$>)' ;
2014-05-04 00:36:20 +02:00
}
$this -> maniaControl -> chat -> sendInformation ( $message . '!' );
$this -> updateManialink = true ;
}
}
}
/**
* Get the dedimania record of the given login
*
* @ param string $login
* @ return RecordData $record
*/
private function getDedimaniaRecord ( $login ) {
if ( ! $this -> dedimaniaData -> records ) {
return new RecordData ( null );
}
$records = $this -> dedimaniaData -> records ;
foreach ( $records as & $record ) {
/** @var RecordData $record */
if ( $record -> login === $login ) {
return $record ;
}
}
return new RecordData ( null );
}
/**
* Get current checkpoint string for dedimania record
*
* @ param string $login
* @ return string
*/
private function getCheckpoints ( $login ) {
if ( ! $login || ! isset ( $this -> checkpoints [ $login ])) {
return null ;
}
$string = '' ;
$count = count ( $this -> checkpoints [ $login ]);
foreach ( $this -> checkpoints [ $login ] as $index => $check ) {
$string .= $check ;
if ( $index < $count - 1 ) {
$string .= ',' ;
}
}
return $string ;
}
/**
* Inserts the given new Dedimania record at the proper position
*
2014-05-09 17:31:29 +02:00
* @ param RecordData $newRecord
* @ param RecordData $oldRecord
2014-05-04 00:36:20 +02:00
* @ return bool
*/
private function insertDedimaniaRecord ( RecordData & $newRecord , RecordData $oldRecord ) {
if ( $newRecord -> nullRecord ) {
return false ;
}
$insert = false ;
// Get max possible rank
$maxRank = $this -> dedimaniaData -> getPlayerMaxRank ( $newRecord -> login );
// Loop through existing records
foreach ( $this -> dedimaniaData -> records as $key => & $record ) {
/** @var RecordData $record */
if ( $record -> rank > $maxRank ) {
// Max rank reached
return false ;
}
if ( $record -> login === $newRecord -> login ) {
// Old record of the same player
if ( $record -> best <= $newRecord -> best ) {
// It's better - Do nothing
return false ;
}
// Replace old record
unset ( $this -> dedimaniaData -> records [ $key ]);
$insert = true ;
break ;
}
// Other player's record
if ( $record -> best <= $newRecord -> best ) {
// It's better - Skip
continue ;
}
// New record is better - Insert it
$insert = true ;
if ( $oldRecord ) {
// Remove old record
foreach ( $this -> dedimaniaData -> records as $key2 => $record2 ) {
/** @var RecordData $record2 */
if ( $record2 -> login !== $oldRecord -> login ) {
continue ;
}
unset ( $this -> dedimaniaData -> records [ $key2 ]);
break ;
}
}
break ;
}
if ( ! $insert && count ( $this -> dedimaniaData -> records ) < $maxRank ) {
// Records list not full - Append new record
$insert = true ;
}
if ( $insert ) {
// Insert new record
array_push ( $this -> dedimaniaData -> records , $newRecord );
// Update ranks
$this -> updateDedimaniaRecordRanks ();
// Save replays
foreach ( $this -> dedimaniaData -> records as & $record ) {
if ( $record -> login !== $newRecord -> login ) {
continue ;
}
$this -> setRecordReplays ( $record );
break ;
}
// Record inserted
return true ;
}
// No new record
return false ;
}
/**
* Update the sorting and the ranks of all dedimania records
*/
private function updateDedimaniaRecordRanks () {
if ( $this -> dedimaniaData -> getRecordCount () == 0 ) {
$this -> maniaControl -> callbackManager -> triggerCallback ( self :: CB_DEDIMANIA_UPDATED , $this -> dedimaniaData -> records );
return ;
}
//TODO move into class dedimania data
// Sort records
usort ( $this -> dedimaniaData -> records , array ( $this , 'compareRecords' ));
// Update ranks
$rank = 1 ;
foreach ( $this -> dedimaniaData -> records as & $record ) {
/** @var RecordData $record */
$record -> rank = $rank ;
$rank ++ ;
}
$this -> maniaControl -> callbackManager -> triggerCallback ( self :: CB_DEDIMANIA_UPDATED , $this -> dedimaniaData -> records );
}
/**
2014-05-09 17:31:29 +02:00
* Update the replay values for the given record
2014-05-04 00:36:20 +02:00
*
2014-05-09 17:31:29 +02:00
* @ param RecordData $record
2014-05-04 00:36:20 +02:00
*/
private function setRecordReplays ( RecordData & $record ) {
// Set validation replay
$validationReplay = $this -> maniaControl -> server -> getValidationReplay ( $record -> login );
if ( $validationReplay ) {
$record -> vReplay = $validationReplay ;
}
// Set ghost replay
if ( $record -> rank <= 1 ) {
$dataDirectory = $this -> maniaControl -> server -> getDataDirectory ();
if ( ! isset ( $this -> dedimaniaData -> directoryAccessChecked )) {
$access = $this -> maniaControl -> server -> checkAccess ( $dataDirectory );
if ( ! $access ) {
trigger_error ( " No access to the servers data directory. Can't retrieve ghost replays. " );
}
$this -> dedimaniaData -> directoryAccessChecked = $access ;
}
if ( $this -> dedimaniaData -> directoryAccessChecked ) {
$ghostReplay = $this -> maniaControl -> server -> getGhostReplay ( $record -> login );
if ( $ghostReplay ) {
$record -> top1GReplay = $ghostReplay ;
}
}
}
}
/**
* Handle PlayerManialinkPageAnswer callback
*
* @ param array $callback
*/
public function handleManialinkPageAnswer ( array $callback ) {
$actionId = $callback [ 1 ][ 2 ];
$login = $callback [ 1 ][ 1 ];
$player = $this -> maniaControl -> playerManager -> getPlayer ( $login );
if ( $actionId == self :: ACTION_SHOW_DEDIRECORDSLIST ) {
$this -> showDediRecordsList ( array (), $player );
}
}
/**
* Shows a ManiaLink list with the local records .
*
* @ param array $chat
* @ param Player $player
*/
public function showDediRecordsList ( array $chat , Player $player ) {
$width = $this -> maniaControl -> manialinkManager -> styleManager -> getListWidgetsWidth ();
$height = $this -> maniaControl -> manialinkManager -> styleManager -> getListWidgetsHeight ();
// get PlayerList
$records = $this -> dedimaniaData -> records ;
if ( ! $records ) {
$this -> maniaControl -> chat -> sendInformation ( 'There are no Dedimania records on this map!' );
return ;
}
//create manialink
$maniaLink = new ManiaLink ( ManialinkManager :: MAIN_MLID );
$script = $maniaLink -> getScript ();
$paging = new Paging ();
$script -> addFeature ( $paging );
// Main frame
$frame = $this -> maniaControl -> manialinkManager -> styleManager -> getDefaultListFrame ( $script , $paging );
$maniaLink -> add ( $frame );
// Start offsets
$x = - $width / 2 ;
$y = $height / 2 ;
// Predefine Description Label
$descriptionLabel = $this -> maniaControl -> manialinkManager -> styleManager -> getDefaultDescriptionLabel ();
$frame -> add ( $descriptionLabel );
// Headline
$headFrame = new Frame ();
$frame -> add ( $headFrame );
$headFrame -> setY ( $y - 5 );
$array = array ( " Rank " => $x + 5 , " Nickname " => $x + 18 , " Login " => $x + 70 , " Time " => $x + 101 );
$this -> maniaControl -> manialinkManager -> labelLine ( $headFrame , $array );
2014-05-15 17:45:08 +02:00
$i = 0 ;
$y = $height / 2 - 10 ;
$pageFrame = null ;
2014-05-04 00:36:20 +02:00
foreach ( $records as $listRecord ) {
2014-05-15 17:45:08 +02:00
if ( $i % 15 === 0 ) {
2014-05-04 00:36:20 +02:00
$pageFrame = new Frame ();
$frame -> add ( $pageFrame );
$y = $height / 2 - 10 ;
$paging -> addPage ( $pageFrame );
}
$recordFrame = new Frame ();
$pageFrame -> add ( $recordFrame );
if ( $i % 2 != 0 ) {
$lineQuad = new Quad_BgsPlayerCard ();
$recordFrame -> add ( $lineQuad );
$lineQuad -> setSize ( $width , 4 );
$lineQuad -> setSubStyle ( $lineQuad :: SUBSTYLE_BgPlayerCardBig );
$lineQuad -> setZ ( 0.001 );
}
if ( strlen ( $listRecord -> nickName ) < 2 ) {
$listRecord -> nickName = $listRecord -> login ;
}
$array = array ( $listRecord -> rank => $x + 5 , '$fff' . $listRecord -> nickName => $x + 18 , $listRecord -> login => $x + 70 , Formatter :: formatTime ( $listRecord -> best ) => $x + 101 );
$this -> maniaControl -> manialinkManager -> labelLine ( $recordFrame , $array );
$recordFrame -> setY ( $y );
$y -= 4 ;
$i ++ ;
}
// Render and display xml
$this -> maniaControl -> manialinkManager -> displayWidget ( $maniaLink , $player , 'DediRecordsList' );
}
/**
* Function to retrieve the dedimania records on the current map
*
* @ return array | RecordData
*/
public function getDedimaniaRecords () {
2014-05-27 23:00:39 +02:00
if ( $this -> dedimaniaData -> records ) {
return $this -> dedimaniaData -> records ;
2014-05-04 00:36:20 +02:00
}
2014-05-27 23:00:39 +02:00
return null ;
2014-05-04 00:36:20 +02:00
}
/**
* @ see \ManiaControl\Plugins\Plugin :: unload ()
*/
public function unload () {
}
/**
* Compare function for sorting dedimania records
*
2014-05-04 14:40:27 +02:00
* @ param \MCTeam\Dedimania\RecordData $first
* @ param \MCTeam\Dedimania\RecordData $second
2014-05-04 00:36:20 +02:00
* @ return int
*/
private function compareRecords ( RecordData $first , RecordData $second ) {
//TODO move into class dedimania data
if ( $first -> best < $second -> best ) {
return - 1 ;
} else if ( $first -> best > $second -> best ) {
return 1 ;
} else {
if ( $first -> rank < $second -> rank ) {
return - 1 ;
} else {
return 1 ;
}
}
}
}