This repository has been archived on 2023-10-09. You can view files and clone it. You cannot open issues or pull requests or push a commit.
Files
blender-archive/source/blender/imbuf/intern/anim.c

965 lines
22 KiB
C
Raw Normal View History

2002-10-12 11:37:38 +00:00
/**
* anim.c
*
* $Id$
*
* ***** BEGIN GPL/BL DUAL LICENSE BLOCK *****
*
* This program 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. The Blender
* Foundation also sells licenses for use in proprietary software under
* the Blender License. See http://www.blender.org/BL/ for information
* about this.
*
* This program 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 this program; if not, write to the Free Software Foundation,
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* The Original Code is Copyright (C) 2001-2002 by NaN Holding BV.
* All rights reserved.
*
* The Original Code is: all of this file.
*
* Contributor(s): none yet.
*
* ***** END GPL/BL DUAL LICENSE BLOCK *****
*/
#ifdef _WIN32
#define INC_OLE2
#include <windows.h>
#include <windowsx.h>
#include <mmsystem.h>
#include <memory.h>
#include <commdlg.h>
#ifndef FREE_WINDOWS
#include <vfw.h>
#endif
2002-10-12 11:37:38 +00:00
#undef AVIIF_KEYFRAME // redefined in AVI_avi.h
#undef AVIIF_LIST // redefined in AVI_avi.h
#define FIXCC(fcc) if (fcc == 0) fcc = mmioFOURCC('N', 'o', 'n', 'e'); \
if (fcc == BI_RLE8) fcc = mmioFOURCC('R', 'l', 'e', '8');
2002-10-12 11:37:38 +00:00
#endif
#include <sys/types.h>
#include <ctype.h>
#include <stdlib.h>
#include <stdio.h>
#ifndef _WIN32
#include <dirent.h>
#else
#include <io.h>
#endif
#include "BLI_blenlib.h" /* BLI_remlink BLI_filesize BLI_addtail
BLI_countlist BLI_stringdec */
#include "DNA_userdef_types.h"
#include "BKE_global.h"
2002-10-12 11:37:38 +00:00
#include "imbuf.h"
#include "imbuf_patch.h"
#include "AVI_avi.h"
#ifdef WITH_QUICKTIME
#if defined(_WIN32) || defined(__APPLE__)
#include "quicktime_import.h"
#endif /* _WIN32 || __APPLE__ */
#endif /* WITH_QUICKTIME */
2002-10-12 11:37:38 +00:00
#include "IMB_imbuf_types.h"
#include "IMB_imbuf.h"
#include "IMB_allocimbuf.h"
#include "IMB_bitplanes.h"
#include "IMB_anim.h"
#include "IMB_anim5.h"
2002-10-12 11:37:38 +00:00
#ifdef WITH_FFMPEG
#include <ffmpeg/avformat.h>
#include <ffmpeg/avcodec.h>
#include <ffmpeg/rational.h>
/* #define FFMPEG_SEEK_DEBUG 1 */
#if LIBAVFORMAT_VERSION_INT < (49 << 16)
#define FFMPEG_OLD_FRAME_RATE 1
#else
#define FFMPEG_CODEC_IS_POINTER 1
#define FFMPEG_HAVE_SKIP_OPTS 1
#endif
#ifndef FF_ER_CAREFUL
#define FF_ER_CAREFUL FF_ER_CAREFULL
#endif
#endif
2002-10-12 11:37:38 +00:00
/****/
#ifdef __sgi
#include <dmedia/moviefile.h>
static void movie_printerror(char * str) {
const char * errstr = mvGetErrorStr(mvGetErrno());
if (str) {
if (errstr) printf("%s: %s\n", str, errstr);
else printf("%s: returned error\n", str);
} else printf("%s\n", errstr);
}
static int startmovie(struct anim * anim) {
if (anim == 0) return(-1);
if ( mvOpenFile (anim->name, O_BINARY|O_RDONLY, &anim->movie ) != DM_SUCCESS ) {
printf("Can't open movie: %s\n", anim->name);
return(-1);
}
if ( mvFindTrackByMedium (anim->movie, DM_IMAGE, &anim->track) != DM_SUCCESS ) {
printf("No image track in movie: %s\n", anim->name);
mvClose(anim->movie);
return(-1);
}
anim->duration = mvGetTrackLength (anim->track);
anim->params = mvGetParams( anim->track );
anim->x = dmParamsGetInt( anim->params, DM_IMAGE_WIDTH);
anim->y = dmParamsGetInt( anim->params, DM_IMAGE_HEIGHT);
anim->interlacing = dmParamsGetEnum (anim->params, DM_IMAGE_INTERLACING);
anim->orientation = dmParamsGetEnum (anim->params, DM_IMAGE_ORIENTATION);
anim->framesize = dmImageFrameSize(anim->params);
anim->curposition = 0;
anim->preseek = 0;
2002-10-12 11:37:38 +00:00
/*printf("x:%d y:%d size:%d interl:%d dur:%d\n", anim->x, anim->y, anim->framesize, anim->interlacing, anim->duration);*/
return (0);
}
static ImBuf * movie_fetchibuf(struct anim * anim, int position) {
ImBuf * ibuf;
/* extern rectcpy(); */
int size;
unsigned int *rect1, *rect2;
if (anim == 0) return (0);
ibuf = IMB_allocImBuf(anim->x, anim->y, 24, IB_rect, 0);
if ( mvReadFrames(anim->track, position, 1, ibuf->x * ibuf->y *
sizeof(int), ibuf->rect ) != DM_SUCCESS ) {
2002-10-12 11:37:38 +00:00
movie_printerror("mvReadFrames");
IMB_freeImBuf(ibuf);
return(0);
}
/*
if (anim->interlacing == DM_IMAGE_INTERLACED_EVEN) {
rect1 = ibuf->rect + (ibuf->x * ibuf->y) - 1;
rect2 = rect1 - ibuf->x;
2002-10-12 11:37:38 +00:00
for (size = ibuf->x * (ibuf->y - 1); size > 0; size--){
*rect1-- = *rect2--;
}
}
*/
2002-10-12 11:37:38 +00:00
if (anim->interlacing == DM_IMAGE_INTERLACED_EVEN)
{
rect1 = ibuf->rect;
rect2 = rect1 + ibuf->x;
for (size = ibuf->x * (ibuf->y - 1); size > 0; size--){
*rect1++ = *rect2++;
}
}
/*if (anim->orientation == DM_TOP_TO_BOTTOM) IMB_flipy(ibuf);*/
return(ibuf);
}
static void free_anim_movie(struct anim * anim) {
if (anim == NULL) return;
if (anim->movie) {
mvClose(anim->movie);
anim->movie = NULL;
}
anim->duration = 0;
}
int ismovie(char *name) {
2002-10-12 11:37:38 +00:00
return (mvIsMovieFile(name) == DM_TRUE);
}
#else
int ismovie(char *name) {
2002-10-12 11:37:38 +00:00
return 0;
}
/* never called, just keep the linker happy */
static int startmovie(struct anim * anim) { return 1; }
static ImBuf * movie_fetchibuf(struct anim * anim, int position) { return NULL; }
static void free_anim_movie(struct anim * anim) { ; }
#endif
static int an_stringdec(char *string, char* kop, char *staart,unsigned short *numlen) {
unsigned short len,nume,nums=0;
2002-10-12 11:37:38 +00:00
short i,found=FALSE;
len=strlen(string);
nume = len;
2002-10-12 11:37:38 +00:00
for(i=len-1;i>=0;i--){
if (string[i]=='/') break;
if (isdigit(string[i])) {
if (found){
nums=i;
} else{
nume=i;
nums=i;
found=TRUE;
}
} else{
if (found) break;
}
}
if (found){
strcpy(staart,&string[nume+1]);
strcpy(kop,string);
kop[nums]=0;
*numlen=nume-nums+1;
return ((int)atoi(&(string[nums])));
}
staart[0]=0;
strcpy(kop,string);
*numlen=0;
return (1);
}
static void an_stringenc(char *string, char *kop, char *staart,
unsigned short numlen, int pic) {
char numstr[10];
unsigned short len,i;
len=sprintf(numstr,"%d",pic);
strcpy(string,kop);
for(i=len;i<numlen;i++){
strcat(string,"0");
}
strcat(string,numstr);
strcat(string,staart);
}
static void free_anim_avi (struct anim *anim) {
#if defined(_WIN32) && !defined(FREE_WINDOWS)
2002-10-12 11:37:38 +00:00
int i;
#endif
2002-10-12 11:37:38 +00:00
if (anim == NULL) return;
if (anim->avi == NULL) return;
AVI_close (anim->avi);
MEM_freeN (anim->avi);
anim->avi = NULL;
#if defined(_WIN32) && !defined(FREE_WINDOWS)
2002-10-12 11:37:38 +00:00
if (anim->pgf) {
AVIStreamGetFrameClose(anim->pgf);
anim->pgf = NULL;
}
for (i = 0; i < anim->avistreams; i++){
AVIStreamRelease(anim->pavi[i]);
}
anim->avistreams = 0;
if (anim->pfileopen) {
AVIFileRelease(anim->pfile);
anim->pfileopen = 0;
AVIFileExit();
}
#endif
anim->duration = 0;
}
void IMB_free_anim_ibuf(struct anim * anim) {
2002-10-12 11:37:38 +00:00
if (anim == NULL) return;
if (anim->ibuf1) IMB_freeImBuf(anim->ibuf1);
if (anim->ibuf2) IMB_freeImBuf(anim->ibuf2);
anim->ibuf1 = anim->ibuf2 = NULL;
}
#ifdef WITH_FFMPEG
static void free_anim_ffmpeg(struct anim * anim);
#endif
2002-10-12 11:37:38 +00:00
void IMB_free_anim(struct anim * anim) {
if (anim == NULL) {
printf("free anim, anim == NULL\n");
return;
}
IMB_free_anim_ibuf(anim);
2002-10-12 11:37:38 +00:00
free_anim_anim5(anim);
free_anim_movie(anim);
free_anim_avi(anim);
#ifdef WITH_QUICKTIME
free_anim_quicktime(anim);
#endif
#ifdef WITH_FFMPEG
free_anim_ffmpeg(anim);
#endif
2002-10-12 11:37:38 +00:00
free(anim);
}
void IMB_close_anim(struct anim * anim) {
if (anim == 0) return;
IMB_free_anim(anim);
}
struct anim * IMB_open_anim(char * name, int ib_flags) {
struct anim * anim;
anim = (struct anim*)MEM_callocN(sizeof(struct anim), "anim struct");
if (anim != NULL) {
strcpy(anim->name, name);
anim->ib_flags = ib_flags;
}
return(anim);
}
static int startavi (struct anim *anim) {
AviError avierror;
#if defined(_WIN32) && !defined(FREE_WINDOWS)
2002-10-12 11:37:38 +00:00
HRESULT hr;
int i, firstvideo = -1;
BYTE abFormat[1024];
LONG l;
LPBITMAPINFOHEADER lpbi;
AVISTREAMINFO avis;
#endif
anim->avi = MEM_callocN (sizeof(AviMovie),"animavi");
if (anim->avi == NULL) {
printf("Can't open avi: %s\n", anim->name);
return -1;
}
avierror = AVI_open_movie (anim->name, anim->avi);
#if defined(_WIN32) && !defined(FREE_WINDOWS)
2002-10-12 11:37:38 +00:00
if (avierror == AVI_ERROR_COMPRESSION) {
AVIFileInit();
hr = AVIFileOpen(&anim->pfile, anim->name, OF_READ, 0L);
if (hr == 0) {
anim->pfileopen = 1;
for (i = 0; i < MAXNUMSTREAMS; i++) {
if (AVIFileGetStream(anim->pfile, &anim->pavi[i], 0L, i) != AVIERR_OK) {
break;
}
AVIStreamInfo(anim->pavi[i], &avis, sizeof(avis));
if ((avis.fccType == streamtypeVIDEO) && (firstvideo == -1)) {
anim->pgf = AVIStreamGetFrameOpen(anim->pavi[i], NULL);
if (anim->pgf) {
firstvideo = i;
// get stream length
anim->avi->header->TotalFrames = AVIStreamLength(anim->pavi[i]);
// get information about images inside the stream
l = sizeof(abFormat);
AVIStreamReadFormat(anim->pavi[i], 0, &abFormat, &l);
lpbi = (LPBITMAPINFOHEADER)abFormat;
anim->avi->header->Height = lpbi->biHeight;
anim->avi->header->Width = lpbi->biWidth;
} else {
FIXCC(avis.fccHandler);
FIXCC(avis.fccType);
printf("Can't find AVI decoder for type : %4.4hs/%4.4hs\n",
(LPSTR)&avis.fccType,
(LPSTR)&avis.fccHandler);
}
}
}
// register number of opened avistreams
anim->avistreams = i;
//
// Couldn't get any video streams out of this file
//
if ((anim->avistreams == 0) || (firstvideo == -1)) {
avierror = AVI_ERROR_FORMAT;
} else {
avierror = AVI_ERROR_NONE;
anim->firstvideo = firstvideo;
}
} else {
AVIFileExit();
}
}
#endif
if (avierror != AVI_ERROR_NONE) {
AVI_print_error(avierror);
printf ("Error loading avi: %s\n", anim->name);
free_anim_avi(anim);
return -1;
}
anim->duration = anim->avi->header->TotalFrames;
anim->params = 0;
anim->x = anim->avi->header->Width;
anim->y = anim->avi->header->Height;
anim->interlacing = 0;
anim->orientation = 0;
anim->framesize = anim->x * anim->y * 4;
anim->curposition = 0;
anim->preseek = 0;
2002-10-12 11:37:38 +00:00
/* printf("x:%d y:%d size:%d interl:%d dur:%d\n", anim->x, anim->y, anim->framesize, anim->interlacing, anim->duration);*/
return 0;
}
static ImBuf * avi_fetchibuf (struct anim *anim, int position) {
ImBuf *ibuf = NULL;
int *tmp;
int y;
if (anim == NULL) return (NULL);
#if defined(_WIN32) && !defined(FREE_WINDOWS)
2002-10-12 11:37:38 +00:00
if (anim->avistreams) {
LPBITMAPINFOHEADER lpbi;
if (anim->pgf) {
lpbi = AVIStreamGetFrame(anim->pgf, position + AVIStreamStart(anim->pavi[anim->firstvideo]));
2002-10-12 11:37:38 +00:00
if (lpbi) {
ibuf = IMB_ibImageFromMemory((int *) lpbi, 100, IB_rect);
//Oh brother...
2002-10-12 11:37:38 +00:00
}
}
} else {
#else
if (1) {
#endif
ibuf = IMB_allocImBuf (anim->x, anim->y, 24, IB_rect, 0);
tmp = AVI_read_frame (anim->avi, AVI_FORMAT_RGB32, position,
AVI_get_stream(anim->avi, AVIST_VIDEO, 0));
if (tmp == NULL) {
printf ("Error reading frame from AVI");
IMB_freeImBuf (ibuf);
return NULL;
}
for (y=0; y < anim->y; y++) {
memcpy (&(ibuf->rect)[((anim->y-y)-1)*anim->x], &tmp[y*anim->x],
anim->x * 4);
}
MEM_freeN (tmp);
}
return ibuf;
}
#ifdef WITH_FFMPEG
extern void do_init_ffmpeg();
#ifdef FFMPEG_CODEC_IS_POINTER
static AVCodecContext* get_codec_from_stream(AVStream* stream)
{
return stream->codec;
}
#else
static AVCodecContext* get_codec_from_stream(AVStream* stream)
{
return &stream->codec;
}
#endif
static int startffmpeg(struct anim * anim) {
int i, videoStream;
AVCodec *pCodec;
AVFormatContext *pFormatCtx;
AVCodecContext *pCodecCtx;
if (anim == 0) return(-1);
do_init_ffmpeg();
if(av_open_input_file(&pFormatCtx, anim->name, NULL, 0, NULL)!=0) {
return -1;
}
if(av_find_stream_info(pFormatCtx)<0) {
av_close_input_file(pFormatCtx);
return -1;
}
dump_format(pFormatCtx, 0, anim->name, 0);
/* Find the first video stream */
videoStream=-1;
for(i=0; i<pFormatCtx->nb_streams; i++)
if(get_codec_from_stream(pFormatCtx->streams[i])->codec_type
== CODEC_TYPE_VIDEO) {
videoStream=i;
break;
}
if(videoStream==-1) {
av_close_input_file(pFormatCtx);
return -1;
}
pCodecCtx = get_codec_from_stream(pFormatCtx->streams[videoStream]);
/* Find the decoder for the video stream */
pCodec=avcodec_find_decoder(pCodecCtx->codec_id);
if(pCodec==NULL) {
av_close_input_file(pFormatCtx);
return -1;
}
pCodecCtx->workaround_bugs = 1;
pCodecCtx->lowres = 0;
pCodecCtx->idct_algo= FF_IDCT_AUTO;
#ifdef FFMPEG_HAVE_SKIP_OPTS
pCodecCtx->skip_frame= AVDISCARD_DEFAULT;
pCodecCtx->skip_idct= AVDISCARD_DEFAULT;
pCodecCtx->skip_loop_filter= AVDISCARD_DEFAULT;
#endif
pCodecCtx->error_resilience= FF_ER_CAREFUL;
pCodecCtx->error_concealment= 3;
if(avcodec_open(pCodecCtx, pCodec)<0) {
av_close_input_file(pFormatCtx);
return -1;
}
#ifdef FFMPEG_OLD_FRAME_RATE
if(pCodecCtx->frame_rate>1000 && pCodecCtx->frame_rate_base==1)
pCodecCtx->frame_rate_base=1000;
anim->duration = pFormatCtx->duration * pCodecCtx->frame_rate
/ pCodecCtx->frame_rate_base / AV_TIME_BASE;
#else
anim->duration = pFormatCtx->duration
* av_q2d(pFormatCtx->streams[videoStream]->r_frame_rate)
/ AV_TIME_BASE;
#endif
anim->params = 0;
anim->x = pCodecCtx->width;
anim->y = pCodecCtx->height;
anim->interlacing = 0;
anim->orientation = 0;
anim->framesize = anim->x * anim->y * 4;
anim->curposition = -1;
anim->pFormatCtx = pFormatCtx;
anim->pCodecCtx = pCodecCtx;
anim->pCodec = pCodec;
anim->videoStream = videoStream;
anim->pFrame = avcodec_alloc_frame();
anim->pFrameRGB = avcodec_alloc_frame();
if (avpicture_get_size(PIX_FMT_RGBA32, anim->x, anim->y)
!= anim->x * anim->y * 4) {
fprintf (stderr,
"ffmpeg has changed alloc scheme ... ARGHHH!\n");
avcodec_close(anim->pCodecCtx);
av_close_input_file(anim->pFormatCtx);
av_free(anim->pFrameRGB);
av_free(anim->pFrame);
return -1;
}
if (pCodecCtx->has_b_frames) {
anim->preseek = 18; /* FIXME: detect gopsize ... */
} else {
anim->preseek = 0;
}
return (0);
}
static ImBuf * ffmpeg_fetchibuf(struct anim * anim, int position) {
ImBuf * ibuf;
int frameFinished;
AVPacket packet;
offset_t pos_to_match = AV_NOPTS_VALUE;
int pos_found = 1;
if (anim == 0) return (0);
ibuf = IMB_allocImBuf(anim->x, anim->y, 24, IB_rect, 0);
avpicture_fill((AVPicture *)anim->pFrameRGB,
(unsigned char*) ibuf->rect,
PIX_FMT_RGBA32, anim->x, anim->y);
if (position != anim->curposition + 1) {
if (position > anim->curposition + 1
&& anim->preseek
&& position - (anim->curposition + 1) < anim->preseek) {
while(av_read_frame(anim->pFormatCtx, &packet)>=0) {
if (packet.stream_index == anim->videoStream) {
avcodec_decode_video(
anim->pCodecCtx,
anim->pFrame, &frameFinished,
packet.data, packet.size);
if (frameFinished) {
anim->curposition++;
}
}
av_free_packet(&packet);
if (position == anim->curposition+1) {
break;
}
}
}
}
if (position != anim->curposition + 1) {
int keyframe_found = 0;
#ifdef FFMPEG_OLD_FRAME_RATE
double frame_rate =
(double) anim->pCodecCtx->frame_rate
/ (double) anim->pCodecCtx->frame_rate_base;
#else
double frame_rate =
av_q2d(anim->pFormatCtx->streams[anim->videoStream]
->r_frame_rate);
#endif
double time_base =
av_q2d(anim->pFormatCtx->streams[anim->videoStream]
->time_base);
long long pos = (long long) position * AV_TIME_BASE
/ frame_rate;
long long st_time = anim->pFormatCtx
->streams[anim->videoStream]->start_time;
if (st_time != AV_NOPTS_VALUE) {
pos += st_time * AV_TIME_BASE * time_base;
}
av_seek_frame(anim->pFormatCtx, -1,
pos, AVSEEK_FLAG_ANY | AVSEEK_FLAG_BACKWARD );
while(av_read_frame(anim->pFormatCtx, &packet)>=0) {
if(packet.stream_index == anim->videoStream) {
pos_to_match
= url_ftell(&anim->pFormatCtx->pb);
if ((packet.flags & PKT_FLAG_KEY) != 0) {
keyframe_found = 1;
}
av_free_packet(&packet);
break;
}
av_free_packet(&packet);
}
if (!keyframe_found) {
int scan_pos = position - anim->preseek;
if (scan_pos < 0) {
scan_pos = 0;
}
pos = (long long) scan_pos * AV_TIME_BASE / frame_rate;
if (st_time != AV_NOPTS_VALUE) {
pos += st_time * AV_TIME_BASE * time_base;
}
}
av_seek_frame(anim->pFormatCtx, -1,
pos, AVSEEK_FLAG_ANY | AVSEEK_FLAG_BACKWARD);
pos_found = 0;
avcodec_flush_buffers(anim->pCodecCtx);
}
while(av_read_frame(anim->pFormatCtx, &packet)>=0) {
if (packet.stream_index == anim->videoStream
&& !pos_found) {
if (url_ftell(&anim->pFormatCtx->pb) == pos_to_match) {
pos_found = 1;
if (anim->pCodecCtx->has_b_frames) {
pos_found++;
/* account for delay caused by
decoding pipeline */
}
}
}
if(packet.stream_index == anim->videoStream) {
avcodec_decode_video(anim->pCodecCtx,
anim->pFrame, &frameFinished,
packet.data, packet.size);
if (frameFinished && pos_found == 2) {
pos_found = 1;
} else if(frameFinished && pos_found == 1) {
unsigned char * p =(unsigned char*) ibuf->rect;
unsigned char * e = p + anim->x * anim->y * 4;
img_convert((AVPicture *)anim->pFrameRGB,
PIX_FMT_RGBA32,
(AVPicture*)anim->pFrame,
anim->pCodecCtx->pix_fmt,
anim->pCodecCtx->width,
anim->pCodecCtx->height);
IMB_flipy(ibuf);
if (G.order == L_ENDIAN) {
/* BGRA -> RGBA */
while (p != e) {
unsigned char a = p[0];
p[0] = p[2];
p[2] = a;
p += 4;
}
} else {
/* ARGB -> RGBA */
while (p != e) {
unsigned long a =
*(unsigned long*) p;
a = (a << 8) | p[0];
*(unsigned long*) p = a;
p += 4;
}
}
av_free_packet(&packet);
break;
}
}
av_free_packet(&packet);
}
return(ibuf);
}
static void free_anim_ffmpeg(struct anim * anim) {
if (anim == NULL) return;
if (anim->pCodecCtx) {
avcodec_close(anim->pCodecCtx);
av_close_input_file(anim->pFormatCtx);
av_free(anim->pFrameRGB);
av_free(anim->pFrame);
}
anim->duration = 0;
}
#endif
2002-10-12 11:37:38 +00:00
/* probeer volgende plaatje te lezen */
/* Geen plaatje, probeer dan volgende animatie te openen */
/* gelukt, haal dan eerste plaatje van animatie */
static struct ImBuf * anim_getnew(struct anim * anim) {
struct ImBuf *ibuf = 0;
if (anim == NULL) return(0);
free_anim_anim5(anim);
free_anim_movie(anim);
free_anim_avi(anim);
#ifdef WITH_QUICKTIME
free_anim_quicktime(anim);
#endif
#ifdef WITH_FFMPEG
free_anim_ffmpeg(anim);
#endif
2002-10-12 11:37:38 +00:00
if (anim->curtype != 0) return (0);
anim->curtype = imb_get_anim_type(anim->name);
switch (anim->curtype) {
case ANIM_ANIM5:
if (startanim5(anim)) return (0);
ibuf = anim5_fetchibuf(anim);
break;
case ANIM_SEQUENCE:
ibuf = IMB_loadiffname(anim->name, anim->ib_flags);
if (ibuf) {
strcpy(anim->first, anim->name);
anim->duration = 1;
}
break;
case ANIM_MOVIE:
if (startmovie(anim)) return (0);
ibuf = IMB_allocImBuf (anim->x, anim->y, 24, 0, 0); /* fake */
break;
case ANIM_AVI:
if (startavi(anim)) {
printf("couldnt start avi\n");
return (0);
}
ibuf = IMB_allocImBuf (anim->x, anim->y, 24, 0, 0);
break;
#ifdef WITH_QUICKTIME
case ANIM_QTIME:
if (startquicktime(anim)) return (0);
2002-10-12 11:37:38 +00:00
ibuf = IMB_allocImBuf (anim->x, anim->y, 24, 0, 0);
break;
#endif
#ifdef WITH_FFMPEG
case ANIM_FFMPEG:
if (startffmpeg(anim)) return (0);
ibuf = IMB_allocImBuf (anim->x, anim->y, 24, 0, 0);
break;
#endif
2002-10-12 11:37:38 +00:00
}
return(ibuf);
}
struct ImBuf * IMB_anim_absolute(struct anim * anim, int position) {
struct ImBuf * ibuf = 0;
char head[256], tail[256];
unsigned short digits;
int pic;
if (anim == NULL) return(0);
if (anim->curtype == 0) {
ibuf = anim_getnew(anim);
if (ibuf == NULL) {
return (0);
}
2002-10-12 11:37:38 +00:00
IMB_freeImBuf(ibuf); /* ???? */
}
if (position < 0) return(0);
if (position >= anim->duration) return(0);
switch(anim->curtype) {
case ANIM_ANIM5:
if (anim->curposition > position) rewindanim5(anim);
while (anim->curposition < position) {
if (nextanim5(anim)) return (0);
}
ibuf = anim5_fetchibuf(anim);
break;
case ANIM_SEQUENCE:
pic = an_stringdec(anim->first, head, tail, &digits);
pic += position;
an_stringenc(anim->name, head, tail, digits, pic);
ibuf = IMB_loadiffname(anim->name, LI_rect);
if (ibuf) {
anim->curposition = position;
/* patch... by freeing the cmap you prevent a double apply cmap... */
/* probably the IB_CMAP option isn't working proper
* after the abgr->rgba reconstruction
*/
IMB_freecmapImBuf(ibuf);
}
break;
case ANIM_MOVIE:
ibuf = movie_fetchibuf(anim, position);
if (ibuf) {
anim->curposition = position;
IMB_convert_rgba_to_abgr(ibuf);
2002-10-12 11:37:38 +00:00
}
break;
case ANIM_AVI:
ibuf = avi_fetchibuf(anim, position);
if (ibuf) anim->curposition = position;
break;
#ifdef WITH_QUICKTIME
case ANIM_QTIME:
ibuf = qtime_fetchibuf(anim, position);
if (ibuf) anim->curposition = position;
break;
#endif
#ifdef WITH_FFMPEG
case ANIM_FFMPEG:
ibuf = ffmpeg_fetchibuf(anim, position);
if (ibuf) anim->curposition = position;
break;
#endif
}
2002-10-12 11:37:38 +00:00
if (ibuf) {
if (anim->ib_flags & IB_ttob) IMB_flipy(ibuf);
sprintf(ibuf->name, "%s.%04d", anim->name, anim->curposition + 1);
}
return(ibuf);
}
struct ImBuf * IMB_anim_nextpic(struct anim * anim) {
struct ImBuf * ibuf = 0;
if (anim == 0) return(0);
ibuf = IMB_anim_absolute(anim, anim->curposition + 1);
return(ibuf);
}
/***/
int IMB_anim_get_duration(struct anim *anim) {
return anim->duration;
}
void IMB_anim_set_preseek(struct anim * anim, int preseek)
{
anim->preseek = preseek;
}
int IMB_anim_get_preseek(struct anim * anim)
{
return anim->preseek;
}