1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257
|
{
Copyright 2008-2018 Michalis Kamburelis.
This file is part of "view3dscene".
"view3dscene" is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
"view3dscene" 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with "view3dscene"; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
----------------------------------------------------------------------------
}
{ Saving screenshots / movies. }
unit V3DSceneScreenShot;
interface
uses Classes, Generics.Collections,
CastleUtils, CastleClassUtils, CastleTimeUtils, CastleParameters;
type
EInvalidScreenShotURL = class(EInvalidParams);
{ One screenshot, that is one --screenshot or --screenshot-range option
occurrence.
Count describes how many actual image screen shots we have to do
for this TScreenShot instance. For each Index between 0 and Count - 1,
you can call UseTime and UseURL to actually get time of this
capture and the image URL where to save this.
Note that UseURL will do the substitution and actually
increment the counter for URLs with @@counter(<padding>) (or %d) inside.
So always call UseURL only once, and in correct order. }
TScreenShot = class
URLPattern: string;
function Count: Cardinal; virtual; abstract;
function UseTime(const Index: Integer): TFloatTime; virtual; abstract;
function UseURL(const Index: Integer): string; virtual; abstract;
{ Call BeginCapture, EndCapture around saving all images for this
TScreenShot instance. They allow for some tasks, e.g.
this is the place when TRangeScreenShot may convert images to a single
movie file.
Success = @true for EndCapture means that we successfully did
all the images, @false means that exception occurred and this
is called only to finalize some things. }
procedure BeginCapture; virtual;
procedure EndCapture(Success: boolean); virtual;
end;
TSingleScreenShot = class(TScreenShot)
Time: TFloatTime;
function Count: Cardinal; override;
function UseTime(const Index: Integer): TFloatTime; override;
function UseURL(const Index: Integer): string; override;
end;
TRangeScreenShot = class(TScreenShot)
private
{ Are we making single movie file? Will be calculated in BeginCapture. }
SingleMovieFile: boolean;
TemporaryImagesPattern: string;
TemporaryImagesCounter: Cardinal;
public
TimeBegin, TimeStep: TFloatTime;
FramesCount: Cardinal;
function Count: Cardinal; override;
function UseTime(const Index: Integer): TFloatTime; override;
function UseURL(const Index: Integer): string; override;
{ @raises(EInvalidScreenShotURL When invalid URLPattern
detected, like an image file with no @@counter(<padding>) or %d.) }
procedure BeginCapture; override;
procedure EndCapture(Success: boolean); override;
end;
TScreenShotList = class(specialize TObjectList<TScreenShot>)
private
ScreenShotCounter: Cardinal;
public
procedure BeginCapture;
end;
var
{ List of screenshots to take.
Created / destroyed in init / fini of this unit. }
ScreenShotsList: TScreenShotList;
{ Are we currently making a screenshot? Just a shortcut for
ScreenShotsList.Count <> 0. }
function MakingScreenShot: boolean;
implementation
uses SysUtils, CastleStringUtils, CastleProgress, CastleFilesUtils,
CastleVideos, CastleURIUtils, CastleLog;
function MakingScreenShot: boolean;
begin
Result := ScreenShotsList.Count <> 0;
end;
function MakeURL(const URLPattern: string;
var Counter: Cardinal): string;
var
ReplacementsDone: Cardinal;
begin
Result := FormatNameCounter(URLPattern, Counter, true, ReplacementsDone);
if ReplacementsDone > 0 then
Inc(Counter);
end;
{ TScreenShot ---------------------------------------------------------------- }
procedure TScreenShot.BeginCapture;
begin
end;
procedure TScreenShot.EndCapture(Success: boolean);
begin
end;
{ TSingleScreenShot ---------------------------------------------------------- }
function TSingleScreenShot.Count: Cardinal;
begin
Result := 1;
end;
function TSingleScreenShot.UseTime(const Index: Integer): TFloatTime;
begin
Result := Time;
end;
function TSingleScreenShot.UseURL(const Index: Integer): string;
begin
Result := MakeURL(URLPattern, ScreenShotsList.ScreenShotCounter);
end;
{ TRangeScreenShot ---------------------------------------------------------- }
function TRangeScreenShot.Count: Cardinal;
begin
Result := FramesCount;
end;
function TRangeScreenShot.UseTime(const Index: Integer): TFloatTime;
begin
Result := TimeBegin + Index * TimeStep;
end;
function TRangeScreenShot.UseURL(const Index: Integer): string;
begin
if SingleMovieFile then
Result := MakeURL(TemporaryImagesPattern, TemporaryImagesCounter) else
Result := MakeURL(URLPattern, ScreenShotsList.ScreenShotCounter);
Progress.Step;
end;
procedure TRangeScreenShot.BeginCapture;
var
ReplacementsDone: Cardinal;
TemporaryImagesPrefix: string;
begin
{ calculate SingleMovieFile }
SingleMovieFile := FfmpegVideoMimeType(URIMimeType(URLPattern), true);
if SingleMovieFile then
begin
{ initialize TemporaryImagesPrefix, TemporaryImagesPattern,
TemporaryImagesCounter }
TemporaryImagesPrefix := GetTempFileNamePrefix;
TemporaryImagesPattern := TemporaryImagesPrefix + '%d.png';
TemporaryImagesCounter := 1;
end;
if not SingleMovieFile then
begin
{ Check that we have some @counter(<padding>) (or %d) in our filename.
Just call FormatNameCounter and ignore result,
to get and check ReplacementsDone. }
FormatNameCounter(URLPattern, -1, true, ReplacementsDone);
if ReplacementsDone = 0 then
raise EInvalidScreenShotURL.CreateFmt('--screenshot-range invalid filename "%s": not recognized as movie filename (so assuming image filename), and no @counter(<padding>) or %%d pattern found', [URLPattern]);
end;
Progress.Init(Count, Format('Screenshot range from time %f (%d frames)',
[TimeBegin, FramesCount]));
end;
procedure TRangeScreenShot.EndCapture(Success: boolean);
var
Executable, OutputMovieFileName, TempFile: string;
I: Integer;
begin
Progress.Fini;
if SingleMovieFile and Success then
begin
Executable := FfmpegExecutable(false);
if Executable = '' then
begin
WritelnWarning('Video', Format('You must have "ffmpeg" program from ' +
'[http://ffmpeg.mplayerhq.hu/] installed and available on $PATH to be able to ' +
'create movie files". Leaving generated temporary images "%s"',
[TemporaryImagesPattern]));
end else
begin
OutputMovieFileName := URIToFilenameSafe(MakeURL(URLPattern, ScreenShotsList.ScreenShotCounter));
FfmpegExecute(Executable,
[ '-f', 'image2', '-i', TemporaryImagesPattern, '-y', '-qscale', '1', OutputMovieFileName ]);
WritelnLog('Removing temporary image files "' + TemporaryImagesPattern + '" ...');
TemporaryImagesCounter := 1;
for I := 1 to FramesCount do
begin
TempFile := URIToFilenameSafe(MakeURL(TemporaryImagesPattern, TemporaryImagesCounter));
CheckDeleteFile(TempFile, true);
end;
WritelnLog('Done removing temporary image files.');
end;
end;
end;
{ TScreenShotList ----------------------------------------------------------- }
procedure TScreenShotList.BeginCapture;
begin
ScreenShotCounter := 1;
end;
{ unit initialization / finalization ---------------------------------------- }
initialization
ScreenShotsList := TScreenShotList.Create(true);
finalization
FreeAndNil(ScreenShotsList);
end.
|