2020-06-27 09:48:55 +00:00
/*
* * screenjob . cpp
* *
* * Generic asynchronous screen display
* *
* * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* * Copyright 2020 Christoph Oelckers
* * All rights reserved .
* *
* * Redistribution and use in source and binary forms , with or without
* * modification , are permitted provided that the following conditions
* * are met :
* *
* * 1. Redistributions of source code must retain the above copyright
* * notice , this list of conditions and the following disclaimer .
* * 2. Redistributions in binary form must reproduce the above copyright
* * notice , this list of conditions and the following disclaimer in the
* * documentation and / or other materials provided with the distribution .
* * 3. The name of the author may not be used to endorse or promote products
* * derived from this software without specific prior written permission .
* *
* * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ` ` AS IS ' ' AND ANY EXPRESS OR
* * IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE IMPLIED WARRANTIES
* * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED .
* * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT , INDIRECT ,
* * INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL DAMAGES ( INCLUDING , BUT
* * NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES ; LOSS OF USE ,
* * DATA , OR PROFITS ; OR BUSINESS INTERRUPTION ) HOWEVER CAUSED AND ON ANY
* * THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT LIABILITY , OR TORT
* * ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY OUT OF THE USE OF
* * THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE .
* * - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
* *
*/
2020-06-20 07:46:41 +00:00
# include "types.h"
# include "build.h"
# include "screenjob.h"
2020-06-27 09:48:55 +00:00
# include "i_time.h"
# include "v_2ddrawer.h"
2020-06-27 22:32:28 +00:00
# include "animlib.h"
# include "v_draw.h"
# include "s_soundinternal.h"
# include "animtexture.h"
2020-07-19 10:48:31 +00:00
# include "gamestate.h"
2020-10-04 16:31:48 +00:00
# include "razemenu.h"
2020-07-26 16:02:24 +00:00
# include "raze_sound.h"
2020-07-29 21:18:08 +00:00
# include "SmackerDecoder.h"
2020-07-23 20:26:07 +00:00
# include "movie/playmve.h"
2020-07-28 19:05:14 +00:00
# include "gamecontrol.h"
2020-09-10 15:54:27 +00:00
# include <vpx/vpx_decoder.h>
# include <vpx/vp8dx.h>
2020-09-05 09:58:19 +00:00
# include "raze_music.h"
2020-06-20 07:46:41 +00:00
2020-06-27 09:48:55 +00:00
IMPLEMENT_CLASS ( DScreenJob , true , false )
2020-06-28 20:17:27 +00:00
IMPLEMENT_CLASS ( DImageScreen , true , false )
2020-07-29 21:18:08 +00:00
2021-04-15 22:02:09 +00:00
bool DSkippableScreenJob : : OnEvent ( event_t * evt )
{
2021-04-16 16:43:59 +00:00
if ( evt - > type = = EV_KeyDown )
{
state = skipped ;
Skipped ( ) ;
}
2021-04-15 22:02:09 +00:00
return true ;
}
2021-04-15 22:11:02 +00:00
void DBlackScreen : : OnTick ( )
{
if ( cleared )
{
int span = ticks * 1000 / GameTicRate ;
if ( span > wait ) state = finished ;
}
}
2021-04-15 22:02:09 +00:00
2021-04-15 22:11:02 +00:00
void DBlackScreen : : Draw ( double )
2020-07-29 21:18:08 +00:00
{
2021-04-15 22:11:02 +00:00
cleared = true ;
2020-07-29 21:18:08 +00:00
twod - > ClearScreen ( ) ;
}
2020-06-28 20:17:27 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
2021-04-15 22:11:02 +00:00
void DImageScreen : : OnTick ( )
2020-06-28 20:17:27 +00:00
{
2021-04-15 22:11:02 +00:00
if ( cleared )
2020-09-04 19:17:24 +00:00
{
2021-04-15 22:11:02 +00:00
int span = ticks * 1000 / GameTicRate ;
if ( span > waittime ) state = finished ;
2020-09-04 19:17:24 +00:00
}
2021-04-15 22:11:02 +00:00
}
void DImageScreen : : Draw ( double smoothratio )
{
if ( tilenum > 0 ) tex = tileGetTexture ( tilenum , true ) ;
2020-06-28 20:17:27 +00:00
twod - > ClearScreen ( ) ;
2021-04-15 22:11:02 +00:00
if ( tex ) DrawTexture ( twod , tex , 0 , 0 , DTA_FullscreenEx , FSMode_ScaleToFit43 , DTA_LegacyRenderStyle , STYLE_Normal , DTA_TranslationIndex , trans , TAG_DONE ) ;
cleared = true ;
2020-06-28 20:17:27 +00:00
}
2020-07-19 09:57:00 +00:00
//---------------------------------------------------------------------------
//
//
//
//---------------------------------------------------------------------------
class ScreenJobRunner
{
enum
{
State_Clear ,
State_Run ,
State_Fadeout
} ;
TArray < JobDesc > jobs ;
CompletionFunc completion ;
int index = - 1 ;
float screenfade ;
bool clearbefore ;
2020-09-03 22:20:32 +00:00
int64_t startTime = - 1 ;
int64_t lastTime = - 1 ;
2020-07-19 09:57:00 +00:00
int actionState ;
int terminateState ;
2020-09-05 14:21:53 +00:00
uint64_t clock = 0 ;
2020-07-19 09:57:00 +00:00
public :
ScreenJobRunner ( JobDesc * jobs_ , int count , CompletionFunc completion_ , bool clearbefore_ )
: completion ( std : : move ( completion_ ) ) , clearbefore ( clearbefore_ )
{
jobs . Resize ( count ) ;
memcpy ( jobs . Data ( ) , jobs_ , count * sizeof ( JobDesc ) ) ;
// Release all jobs from the garbage collector - the code as it is cannot deal with them getting collected. This should be removed later once the GC is working.
for ( int i = 0 ; i < count ; i + + )
{
jobs [ i ] . job - > Release ( ) ;
}
AdvanceJob ( false ) ;
}
2020-07-19 10:48:31 +00:00
~ ScreenJobRunner ( )
{
DeleteJobs ( ) ;
}
void DeleteJobs ( )
{
for ( auto & job : jobs )
{
job . job - > ObjectFlags | = OF_YesReallyDelete ;
delete job . job ;
}
jobs . Clear ( ) ;
}
2020-07-19 09:57:00 +00:00
void AdvanceJob ( bool skip )
{
2020-07-29 21:18:08 +00:00
if ( index > = 0 )
{
if ( jobs [ index ] . postAction ) jobs [ index ] . postAction ( ) ;
jobs [ index ] . job - > Destroy ( ) ;
}
2020-07-19 09:57:00 +00:00
index + + ;
2020-07-29 21:18:08 +00:00
while ( index < jobs . Size ( ) & & ( jobs [ index ] . job = = nullptr | | ( skip & & jobs [ index ] . ignoreifskipped ) ) )
{
if ( jobs [ index ] . job ! = nullptr ) jobs [ index ] . job - > Destroy ( ) ;
index + + ;
}
2020-07-19 09:57:00 +00:00
actionState = clearbefore ? State_Clear : State_Run ;
if ( index < jobs . Size ( ) ) screenfade = jobs [ index ] . job - > fadestyle & DScreenJob : : fadein ? 0.f : 1.f ;
2020-09-05 15:37:37 +00:00
lastTime = startTime = - 1 ;
clock = 0 ;
2020-07-26 17:55:06 +00:00
inputState . ClearAllInput ( ) ;
2020-07-19 09:57:00 +00:00
}
int DisplayFrame ( )
{
auto & job = jobs [ index ] ;
2020-09-05 18:31:45 +00:00
auto now = I_GetTimeNS ( ) ;
2020-08-21 20:30:51 +00:00
bool processed = job . job - > ProcessInput ( ) ;
2020-09-05 14:21:53 +00:00
if ( startTime = = - 1 )
{
lastTime = startTime = now ;
}
else if ( ! M_Active ( ) )
2020-07-21 22:42:50 +00:00
{
2020-09-05 14:21:53 +00:00
clock + = now - lastTime ;
if ( clock = = 0 ) clock = 1 ;
2020-07-21 22:42:50 +00:00
}
2021-04-08 16:01:42 +00:00
bool skiprequest = clock > 100'000'000 & & inputState . CheckAllInput ( ) & & ! processed & & job . job - > fadestate ! = DScreenJob : : fadeout ;
2020-07-21 22:42:50 +00:00
lastTime = now ;
2020-09-16 17:11:36 +00:00
if ( screenfade < 1.f & & ! M_Active ( ) )
2020-07-19 09:57:00 +00:00
{
float ms = ( clock / 1'000'000 ) / job . job - > fadetime ;
screenfade = clamp ( ms , 0.f , 1.f ) ;
2020-09-16 17:11:36 +00:00
twod - > SetScreenFade ( screenfade ) ;
2021-04-08 16:01:42 +00:00
if ( job . job - > fadestate ! = DScreenJob : : fadeout )
job . job - > fadestate = DScreenJob : : fadein ;
2020-07-19 09:57:00 +00:00
}
2020-09-16 17:11:36 +00:00
else
{
job . job - > fadestate = DScreenJob : : visible ;
screenfade = 1.f ;
}
2020-07-19 09:57:00 +00:00
job . job - > SetClock ( clock ) ;
2021-04-15 22:11:02 +00:00
int state = job . job - > Frame ( clock , skiprequest , M_Active ( ) ? 1. : I_GetTimeFrac ( ) ) ;
2020-09-05 14:21:53 +00:00
clock = job . job - > GetClock ( ) ;
if ( clock = = 0 ) clock = 1 ;
2020-07-19 09:57:00 +00:00
return state ;
}
int FadeoutFrame ( )
{
2020-09-05 18:31:45 +00:00
auto now = I_GetTimeNS ( ) ;
2020-07-21 22:42:50 +00:00
2020-09-11 18:55:58 +00:00
if ( startTime = = - 1 )
{
lastTime = startTime = now ;
}
else if ( ! M_Active ( ) )
2020-07-21 22:42:50 +00:00
{
2020-09-05 14:21:53 +00:00
clock + = now - lastTime ;
if ( clock = = 0 ) clock = 1 ;
2020-07-21 22:42:50 +00:00
}
lastTime = now ;
2020-07-19 09:57:00 +00:00
float ms = ( clock / 1'000'000 ) / jobs [ index ] . job - > fadetime ;
float screenfade2 = clamp ( screenfade - ms , 0.f , 1.f ) ;
2020-07-21 22:42:50 +00:00
if ( ! M_Active ( ) ) twod - > SetScreenFade ( screenfade2 ) ;
2020-07-19 09:57:00 +00:00
if ( screenfade2 < = 0.f )
{
twod - > Unlock ( ) ; // must unlock before displaying.
return 0 ;
}
return 1 ;
}
2021-04-15 22:50:13 +00:00
bool OnEvent ( event_t * ev )
{
2021-04-15 22:02:09 +00:00
if ( index > = jobs . Size ( ) ) return false ;
if ( jobs [ index ] . job - > state ! = DScreenJob : : running ) return false ;
2021-04-15 22:50:13 +00:00
return jobs [ index ] . job - > OnEvent ( ev ) ;
}
void OnFinished ( )
{
if ( completion ) completion ( false ) ;
completion = nullptr ; // only finish once.
}
void OnTick ( )
{
if ( index > = jobs . Size ( ) )
{
//DeleteJobs();
//twod->SetScreenFade(1);
//twod->ClearScreen(); // This must not leave the 2d buffer empty.
//if (gamestate == GS_INTRO) OnFinished();
//else Net_WriteByte(DEM_ENDSCREENJOB); // intermissions must be terminated synchronously.
}
else
{
2021-04-15 22:02:09 +00:00
if ( jobs [ index ] . job - > state ! = DScreenJob : : running ) return ;
2021-04-15 22:50:13 +00:00
jobs [ index ] . job - > ticks + + ;
jobs [ index ] . job - > OnTick ( ) ;
}
}
2020-07-19 09:57:00 +00:00
bool RunFrame ( )
{
if ( index > = jobs . Size ( ) )
{
2020-07-19 10:48:31 +00:00
DeleteJobs ( ) ;
2020-07-19 09:57:00 +00:00
twod - > SetScreenFade ( 1 ) ;
2020-09-05 13:59:32 +00:00
twod - > ClearScreen ( ) ; // This must not leave the 2d buffer empty.
2020-07-19 09:57:00 +00:00
if ( completion ) completion ( false ) ;
return false ;
}
if ( actionState = = State_Clear )
{
actionState = State_Run ;
twod - > ClearScreen ( ) ;
}
else if ( actionState = = State_Run )
{
terminateState = DisplayFrame ( ) ;
if ( terminateState < 1 )
{
// Must lock before displaying.
if ( jobs [ index ] . job - > fadestyle & DScreenJob : : fadeout )
{
twod - > Lock ( ) ;
2020-09-11 18:55:58 +00:00
startTime = - 1 ;
clock = 0 ;
2020-07-19 09:57:00 +00:00
jobs [ index ] . job - > fadestate = DScreenJob : : fadeout ;
2021-04-15 22:02:09 +00:00
jobs [ index ] . job - > state = DScreenJob : : stopping ;
2021-04-07 17:39:48 +00:00
gamestate = GS_INTRO ; // block menu and console during fadeout - this can cause timing problems.
2020-07-19 09:57:00 +00:00
actionState = State_Fadeout ;
}
else
{
AdvanceJob ( terminateState < 0 ) ;
}
}
}
else if ( actionState = = State_Fadeout )
{
int ended = FadeoutFrame ( ) ;
if ( ended < 1 )
{
2021-04-15 22:02:09 +00:00
jobs [ index ] . job - > state = DScreenJob : : stopped ;
2020-07-19 09:57:00 +00:00
AdvanceJob ( terminateState < 0 ) ;
}
}
return true ;
}
} ;
2020-07-19 10:48:31 +00:00
ScreenJobRunner * runner ;
2020-07-21 22:42:50 +00:00
void RunScreenJob ( JobDesc * jobs , int count , CompletionFunc completion , bool clearbefore , bool blockingui )
2020-07-19 09:57:00 +00:00
{
2020-07-19 10:48:31 +00:00
assert ( completion ! = nullptr ) ;
2020-08-07 20:00:43 +00:00
videoclearFade ( ) ;
2020-07-19 10:48:31 +00:00
if ( count )
2020-07-19 09:57:00 +00:00
{
2020-07-19 10:48:31 +00:00
runner = new ScreenJobRunner ( jobs , count , completion , clearbefore ) ;
2020-09-05 13:43:34 +00:00
gameaction = blockingui ? ga_intro : ga_intermission ;
}
else
{
completion ( false ) ;
2020-07-19 09:57:00 +00:00
}
}
2020-07-19 10:48:31 +00:00
void DeleteScreenJob ( )
2020-07-19 09:57:00 +00:00
{
2020-07-19 10:48:31 +00:00
if ( runner )
2020-07-19 09:57:00 +00:00
{
2020-07-19 10:48:31 +00:00
delete runner ;
runner = nullptr ;
2020-07-19 09:57:00 +00:00
}
2021-04-07 17:39:48 +00:00
twod - > SetScreenFade ( 1 ) ;
2020-07-19 10:48:31 +00:00
}
2020-07-19 09:57:00 +00:00
2021-04-15 22:50:13 +00:00
void EndScreenJob ( )
{
if ( runner ) runner - > OnFinished ( ) ;
DeleteScreenJob ( ) ;
}
bool ScreenJobResponder ( event_t * ev )
{
if ( runner ) return runner - > OnEvent ( ev ) ;
return false ;
}
void ScreenJobTick ( )
{
if ( runner ) runner - > OnTick ( ) ;
}
void ScreenJobDraw ( )
2020-07-19 10:48:31 +00:00
{
// we cannot recover from this because we have no completion callback to call.
2020-09-04 18:46:44 +00:00
if ( ! runner )
{
// We can get here before a gameaction has been processed. In that case just draw a black screen and wait.
if ( gameaction = = ga_nothing ) I_Error ( " Trying to run a non-existent screen job " ) ;
twod - > ClearScreen ( ) ;
return ;
}
2020-07-19 10:48:31 +00:00
auto res = runner - > RunFrame ( ) ;
2020-07-26 10:43:32 +00:00
if ( ! res )
{
2020-09-04 18:46:44 +00:00
assert ( ( gamestate ! = GS_INTERMISSION & & gamestate ! = GS_INTRO ) | | gameaction ! = ga_nothing ) ;
2020-07-26 10:43:32 +00:00
DeleteScreenJob ( ) ;
}
2020-07-19 09:57:00 +00:00
}