windows-nt/Source/XPSP1/NT/multimedia/media/avi/avifile.16/aviopts.c

551 lines
18 KiB
C
Raw Normal View History

2020-09-26 03:20:57 -05:00
/****************************************************************************
*
* AVIOPTS.C
*
* routine for bringing up the compression options dialog
*
* AVISaveOptions()
*
* Copyright (c) 1992 Microsoft Corporation. All Rights Reserved.
*
* You have a royalty-free right to use, modify, reproduce and
* distribute the Sample Files (and/or any modified version) in
* any way you find useful, provided that you agree that
* Microsoft has no warranty obligations or liability for any
* Sample Application Files which are modified.
*
***************************************************************************/
#include <win32.h>
#include <mmreg.h>
#include <msacm.h>
#include <compman.h>
#include "avifile.h"
#include "aviopts.h"
#include "aviopts.dlg"
#ifdef WIN32
//!!! ACK the ACM does not work on NT
#define acmGetVersion() 0
#define acmFormatChoose(x) 1 // some error.
#define ICCompressorChoose(hwnd,a,b,c,d,e) 0
#define ICCompressorFree(x)
#endif
/****************************************************************************
***************************************************************************/
extern HINSTANCE ghMod;
LONG FAR PASCAL _export AVICompressOptionsDlgProc(HWND hwnd, unsigned msg, WORD wParam, LONG lParam);
/****************************************************************************
***************************************************************************/
int gnNumStreams = 0; // how many streams in array
int gnCurStream = 0; // which stream's options we're setting
PAVISTREAM FAR *gapAVI; // array of stream pointers
LPAVICOMPRESSOPTIONS FAR *gapOpt; // array of option structures to fill
UINT guiFlags;
COMPVARS gCompVars; // for ICCompressorChoose
/****************************************************************************
***************************************************************************/
/*************************************************************
* @doc EXTERNAL AVISaveOptions
*
* @api BOOL | AVISaveOptions | This function gets the save options for
* a file and returns them in a buffer.
*
* @parm HWND | hwnd | Specifies the parent window handle for the Compression Options
* dialog box.
*
* @parm UINT | uiFlags | Specifies the flags for displaying the
* Compression Options dialog box. The following flags are defined:
*
* @flag ICMF_CHOOSE_KEYFRAME | Displays a "Key frame every" box for
* the video options. This is the same flag used in <f ICCompressorChoose>.
*
* @flag ICMF_CHOOSE_DATARATE | Displays a "Data rate" box for the video
* options. This is the same flag used in <f ICCompressorChoose>.
*
* @flag ICMF_CHOOSE_PREVIEW | Displays a "Preview" button for
* the video options. This button previews the compression
* using a frame from the stream. This is the same flag
* used in <f ICCompressorChoose>.
*
* @parm int | nStreams | Specifies the number of streams
* that will have their options set by the dialog box.
*
* @parm PAVISTREAM FAR * | ppavi | Specifies a pointer to an
* array of stream interface pointers. The <p nStreams>
* parameter indicates the number of pointers in the array.
*
* @parm LPAVICOMPRESSOPTIONS FAR * | plpOptions | Specifies a pointer
* to an array of <t LPAVICOMPRESSOPTIONS> pointers
* to hold the compression options set by the dialog box. The
* <p nStreams> parameter indicates the number of
* pointers in the array.
*
* @rdesc Returns TRUE if the user pressed OK, FALSE for CANCEL or an error.
*
* @comm This function presents a standard Compression Options dialog
* box using <p hwnd> as the parent window handle. When the
* user is finished selecting the compression options for
* each stream, the options are returned in the <t AVICOMPRESSOPTIONS>
* structures in the array referenced by <p lpOptions>. The caller
* must pass the interface pointers for the streams
* in the array referenced by <p ppavi>.
*
******************************************************************/
STDAPI_(BOOL) AVISaveOptions(HWND hwnd, UINT uiFlags, int nStreams, PAVISTREAM FAR *ppavi, LPAVICOMPRESSOPTIONS FAR *plpOptions)
{
BOOL f;
AVICOMPRESSOPTIONS FAR *aOptions;
int i;
/* Save the stream pointer */
gnNumStreams = nStreams;
gnCurStream = -1;
gapAVI = ppavi;
gapOpt = plpOptions;
guiFlags = uiFlags;
//
// Remember the old compression options in case we cancel and need to
// restore them
//
aOptions = (AVICOMPRESSOPTIONS FAR *)GlobalAllocPtr(GMEM_MOVEABLE,
nStreams * sizeof(AVICOMPRESSOPTIONS));
if (!aOptions)
return FALSE;
for (i = 0; i < nStreams; i++)
aOptions[i] = *plpOptions[i];
f = DialogBox (ghMod, "AVICompressOptionsDialog", hwnd,
(DLGPROC)AVICompressOptionsDlgProc);
//
// The user cancelled... put the old compression options back.
//
if (f == 0)
for (i = 0; i < nStreams; i++)
*plpOptions[i] = aOptions[i];
// Couldn't bring up the dialog
if (f == -1)
f = 0;
GlobalFreePtr(aOptions);
// !!! Returning TRUE doesn't guarantee something actually changed...
return f;
}
/*************************************************************
* @doc EXTERNAL AVISaveOptionsFree
*
* @api LONG | AVISaveOptionsFree | This function frees the resources allocated
* by <f AVISaveOptions>.
*
* @parm int | nStreams | Specifies the number of <t AVICOMPRESSOPTIONS>
* structures in the array passed in as the next parameter.
*
* @parm LPAVICOMPRESSOPTIONS FAR * | plpOptions | Specifies a pointer
* to an array of <t LPAVICOMPRESSOPTIONS> pointers
* to hold the compression options set by the dialog box. The
* resources in each of these structures that were allocated by
* <f AVISaveOptions> will be freed.
*
* @rdesc This function always returns AVIERR_OK (zero)
*
* @comm This function frees the resources allocated by <f AVISaveOptions>.
**************************************************************/
STDAPI AVISaveOptionsFree(int nStreams, LPAVICOMPRESSOPTIONS FAR *plpOptions)
{
for (; nStreams > 0; nStreams--) {
if (plpOptions[nStreams-1]->lpParms)
GlobalFreePtr(plpOptions[nStreams-1]->lpParms);
plpOptions[nStreams-1]->lpParms = NULL;
if (plpOptions[nStreams-1]->lpFormat)
GlobalFreePtr(plpOptions[nStreams-1]->lpFormat);
plpOptions[nStreams-1]->lpFormat = NULL;
}
return AVIERR_OK;
}
/****************************************************************************
Bring up the compression options for the current stream
***************************************************************************/
BOOL StreamOptions(HWND hwnd) {
AVISTREAMINFO avis;
BOOL f = FALSE;
LONG lTemp;
UINT w;
// Get the stream type
if (AVIStreamInfo(gapAVI[gnCurStream], &avis, sizeof(avis)) != 0)
return FALSE;
//
// Video stream -- bring up the video compression dlg
//
if (avis.fccType == streamtypeVIDEO) {
// The structure we have now is not filled in ... init it
if (!(gapOpt[gnCurStream]->dwFlags & AVICOMPRESSF_VALID)) {
_fmemset(gapOpt[gnCurStream], 0,
sizeof(AVICOMPRESSOPTIONS));
gapOpt[gnCurStream]->fccHandler = comptypeDIB;
gapOpt[gnCurStream]->dwQuality = DEFAULT_QUALITY;
}
_fmemset(&gCompVars, 0, sizeof(gCompVars));
gCompVars.cbSize = sizeof(gCompVars);
gCompVars.dwFlags = ICMF_COMPVARS_VALID;
gCompVars.fccHandler = gapOpt[gnCurStream]->fccHandler;
gCompVars.lQ = gapOpt[gnCurStream]->dwQuality;
gCompVars.lpState = gapOpt[gnCurStream]->lpParms;
gCompVars.cbState = gapOpt[gnCurStream]->cbParms;
gCompVars.lKey =
(gapOpt[gnCurStream]->dwFlags & AVICOMPRESSF_KEYFRAMES)?
(gapOpt[gnCurStream]->dwKeyFrameEvery) : 0;
gCompVars.lDataRate =
(gapOpt[gnCurStream]->dwFlags & AVICOMPRESSF_DATARATE) ?
(gapOpt[gnCurStream]->dwBytesPerSecond / 1024) : 0;
// !!! Don't pass flags verbatim if others are defined!!!
f = ICCompressorChoose(hwnd, guiFlags, NULL,
gapAVI[gnCurStream], &gCompVars, NULL);
/* Set the options to our new values */
gapOpt[gnCurStream]->lpParms = gCompVars.lpState;
gapOpt[gnCurStream]->cbParms = gCompVars.cbState;
gCompVars.lpState = NULL; // so it won't be freed
gapOpt[gnCurStream]->fccHandler = gCompVars.fccHandler;
gapOpt[gnCurStream]->dwQuality = gCompVars.lQ;
gapOpt[gnCurStream]->dwKeyFrameEvery = gCompVars.lKey;
gapOpt[gnCurStream]->dwBytesPerSecond = gCompVars.lDataRate
* 1024;
if (gCompVars.lKey)
gapOpt[gnCurStream]->dwFlags |= AVICOMPRESSF_KEYFRAMES;
else
gapOpt[gnCurStream]->dwFlags &=~AVICOMPRESSF_KEYFRAMES;
if (gCompVars.lDataRate)
gapOpt[gnCurStream]->dwFlags |= AVICOMPRESSF_DATARATE;
else
gapOpt[gnCurStream]->dwFlags &=~AVICOMPRESSF_DATARATE;
// If they pressed OK, we have valid stuff in here now.
if (f)
gapOpt[gnCurStream]->dwFlags |= AVICOMPRESSF_VALID;
// Close the stuff opened by ICCompressorChoose
ICCompressorFree(&gCompVars);
//
// Bring up the ACM format dialog and stuff it in our
// compression options structure
//
} else if (avis.fccType == streamtypeAUDIO) {
ACMFORMATCHOOSE acf;
LONG lsizeF = 0;
if (acmGetVersion() < 0x02000000L) {
char achACM[160];
char achACMV[40];
LoadString(ghMod, IDS_BADACM, achACM, sizeof(achACM));
LoadString(ghMod, IDS_BADACMV, achACMV, sizeof(achACMV));
MessageBox(hwnd, achACM, achACMV, MB_OK | MB_ICONHAND);
return FALSE;
}
_fmemset(&acf, 0, sizeof(acf)); // or ACM blows up
acf.cbStruct = sizeof(ACMFORMATCHOOSE);
// If our options struct has valid data, use it to init
// the acm dialog with, otherwise pick a default.
acf.fdwStyle = (gapOpt[gnCurStream]->dwFlags & AVICOMPRESSF_VALID)
? ACMFORMATCHOOSE_STYLEF_INITTOWFXSTRUCT : 0;
acf.hwndOwner = hwnd;
// Make sure the AVICOMPRESSOPTIONS has a big enough lpFormat
acmMetrics(NULL, ACM_METRIC_MAX_SIZE_FORMAT, (LPVOID)&lTemp);
if ((gapOpt[gnCurStream]->cbFormat == 0 ||
gapOpt[gnCurStream]->lpFormat == NULL) && lTemp) {
gapOpt[gnCurStream]->lpFormat =
GlobalAllocPtr(GMEM_MOVEABLE, lTemp);
gapOpt[gnCurStream]->cbFormat = lTemp;
} else if (gapOpt[gnCurStream]->cbFormat < (DWORD)lTemp && lTemp) {
gapOpt[gnCurStream]->lpFormat =
GlobalReAllocPtr(gapOpt[gnCurStream]->lpFormat, lTemp,
GMEM_MOVEABLE);
gapOpt[gnCurStream]->cbFormat = lTemp;
}
if (!gapOpt[gnCurStream]->lpFormat)
return FALSE;
acf.pwfx = gapOpt[gnCurStream]->lpFormat;
acf.cbwfx = gapOpt[gnCurStream]->cbFormat;
//
// Only ask for choices that we can actually convert to
//
AVIStreamReadFormat(gapAVI[gnCurStream],
AVIStreamStart(gapAVI[gnCurStream]), NULL, &lsizeF);
// !!! Work around ACM bug by making sure our format is big enough
lsizeF = max(lsizeF, sizeof(WAVEFORMATEX));
acf.pwfxEnum = (LPWAVEFORMATEX)
GlobalAllocPtr(GMEM_MOVEABLE | GMEM_ZEROINIT, lsizeF);
if (acf.pwfxEnum) {
AVIStreamReadFormat(gapAVI[gnCurStream],
AVIStreamStart(gapAVI[gnCurStream]), acf.pwfxEnum, &lsizeF);
acf.fdwEnum |= ACM_FORMATENUMF_CONVERT;
}
// If they pressed OK, we now have valid stuff in here!
w = acmFormatChoose(&acf);
if (w == MMSYSERR_NOERROR)
gapOpt[gnCurStream]->dwFlags |= AVICOMPRESSF_VALID;
else if (w != ACMERR_CANCELED) {
MessageBeep(0); // !!! Should really be a message box!
}
if (acf.pwfxEnum)
GlobalFreePtr(acf.pwfxEnum);
f = (w == MMSYSERR_NOERROR);
}
return f;
}
void NEAR PASCAL NewStreamChosen(HWND hwnd)
{
BOOL f;
AVISTREAMINFO avis;
DWORD dw;
HIC hic;
ICINFO icinfo;
ACMFORMATDETAILS acmfmt;
ACMFORMATTAGDETAILS aftd;
LONG lsizeF;
LPBITMAPINFOHEADER lp = NULL;
char szFFDesc[80];
char szDesc[120];
// Set the interleave options for the selection we're leaving
// !!! This code also appears in the OK button
if (gnCurStream >= 0) { // there is a previous sel
if (IsDlgButtonChecked(hwnd, IDC_intINTERLEAVE)) {
dw = (DWORD)GetDlgItemInt(hwnd, IDC_intINTERLEAVEEDIT,
NULL, FALSE);
gapOpt[gnCurStream]->dwInterleaveEvery = dw;
gapOpt[gnCurStream]->dwFlags |= AVICOMPRESSF_INTERLEAVE;
} else {
dw = (DWORD)GetDlgItemInt(hwnd, IDC_intINTERLEAVEEDIT,
NULL, FALSE);
gapOpt[gnCurStream]->dwInterleaveEvery = dw;
gapOpt[gnCurStream]->dwFlags &=~AVICOMPRESSF_INTERLEAVE;
}
}
gnCurStream = (int)SendDlgItemMessage(hwnd, IDC_intCHOOSESTREAM,
CB_GETCURSEL, 0, 0L);
if (gnCurStream < 0)
return;
if (AVIStreamInfo(gapAVI[gnCurStream], &avis, sizeof(avis)) != 0)
return;
//
// Show a string describing the current format
//
szDesc[0] = '\0';
lsizeF = 0;
AVIStreamReadFormat(gapAVI[gnCurStream],
AVIStreamStart(gapAVI[gnCurStream]), NULL, &lsizeF);
if (lsizeF) {
lp = (LPBITMAPINFOHEADER)GlobalAllocPtr(GHND, lsizeF);
if (lp) {
if (AVIStreamReadFormat(gapAVI[gnCurStream],
AVIStreamStart(gapAVI[gnCurStream]),
lp, &lsizeF) == AVIERR_OK) {
if (avis.fccType == streamtypeVIDEO) {
wsprintf(szDesc, "%ldx%ldx%d\n", lp->biWidth,
lp->biHeight, lp->biBitCount);
if (lp->biCompression == BI_RGB) {
LoadString(ghMod, IDS_FFDESC, szFFDesc,
sizeof(szFFDesc));
lstrcat(szDesc, szFFDesc);
} else {
hic = ICDecompressOpen(ICTYPE_VIDEO,avis.fccHandler,
lp, NULL);
if (hic) {
if (ICGetInfo(hic, &icinfo,sizeof(icinfo)) != 0)
lstrcat(szDesc, icinfo.szDescription);
ICClose(hic);
}
}
} else if (avis.fccType == streamtypeAUDIO) {
_fmemset(&acmfmt, 0, sizeof(acmfmt));
acmfmt.pwfx = (LPWAVEFORMATEX) lp;
acmfmt.cbStruct = sizeof(ACMFORMATDETAILS);
acmfmt.dwFormatTag = acmfmt.pwfx->wFormatTag;
acmfmt.cbwfx = lsizeF;
aftd.cbStruct = sizeof(aftd);
aftd.dwFormatTag = acmfmt.pwfx->wFormatTag;
aftd.fdwSupport = 0;
if ((acmFormatTagDetails(NULL,
&aftd,
ACM_FORMATTAGDETAILSF_FORMATTAG) == 0) &&
(acmFormatDetails(NULL, &acmfmt,
ACM_FORMATDETAILSF_FORMAT) == 0)) {
wsprintf(szDesc, "%s %s", (LPSTR) acmfmt.szFormat,
(LPSTR) aftd.szFormatTag);
}
}
}
GlobalFreePtr(lp);
}
}
SetDlgItemText(hwnd, IDC_intFORMAT, szDesc);
//
// AUDIO and VIDEO streams have a compression dialog
//
if (avis.fccType == streamtypeAUDIO ||
avis.fccType == streamtypeVIDEO)
EnableWindow(GetDlgItem(hwnd, IDC_intOPTIONS), TRUE);
else
EnableWindow(GetDlgItem(hwnd, IDC_intOPTIONS), FALSE);
//
// Every stream but the first has an interleave options
//
if (gnCurStream > 0) {
EnableWindow(GetDlgItem(hwnd, IDC_intINTERLEAVE), TRUE);
EnableWindow(GetDlgItem(hwnd, IDC_intINTERLEAVEEDIT),
TRUE);
EnableWindow(GetDlgItem(hwnd, IDC_intINTERLEAVETEXT),
TRUE);
// Set the interleave situation for this stream
f = (gapOpt[gnCurStream]->dwFlags & AVICOMPRESSF_INTERLEAVE)
!= 0;
dw = gapOpt[gnCurStream]->dwInterleaveEvery;
CheckDlgButton(hwnd, IDC_intINTERLEAVE, f);
SetDlgItemInt(hwnd, IDC_intINTERLEAVEEDIT, (int)dw, FALSE);
} else {
EnableWindow(GetDlgItem(hwnd, IDC_intINTERLEAVE),FALSE);
EnableWindow(GetDlgItem(hwnd, IDC_intINTERLEAVEEDIT),
FALSE);
EnableWindow(GetDlgItem(hwnd, IDC_intINTERLEAVETEXT),
FALSE);
}
}
/*--------------------------------------------------------------+
* Dialog Proc for the main compression options dialog *
+--------------------------------------------------------------*/
LONG FAR PASCAL _export AVICompressOptionsDlgProc(HWND hwnd, unsigned msg, WORD wParam, LONG lParam)
{
int nVal;
AVISTREAMINFO avis;
DWORD dw;
switch(msg){
case WM_INITDIALOG:
//
// If we've only got one stream to set the options for, it seems
// strange to bring up a box to let you choose which stream you want.
// Let's skip straight to the proper options dlg box.
//
if (gnNumStreams == 1) {
gnCurStream = 0;
EndDialog(hwnd, StreamOptions(hwnd));
return TRUE;
}
/* Add the list of streams to the drop-down box */
for (nVal = 0; nVal < gnNumStreams; nVal++) {
// Get the name of this stream
AVIStreamInfo(gapAVI[nVal], &avis, sizeof(avis));
SendDlgItemMessage(hwnd, IDC_intCHOOSESTREAM, CB_ADDSTRING, 0,
(LONG) (LPSTR)avis.szName);
}
// Set our initial selection to the first item
SendDlgItemMessage(hwnd, IDC_intCHOOSESTREAM, CB_SETCURSEL, 0, 0L);
// Make sure we see it
SendMessage(hwnd, WM_COMMAND, IDC_intCHOOSESTREAM,
MAKELONG(GetDlgItem(hwnd, IDC_intCHOOSESTREAM), CBN_SELCHANGE));
return TRUE;
case WM_COMMAND:
switch(wParam){
case IDOK:
// Set the interleave options for the selection we're on
// !!! This code also appears in the SELCHANGE code
if (gnCurStream >= 0) { // there is a valid selection
if (IsDlgButtonChecked(hwnd, IDC_intINTERLEAVE)) {
dw = (DWORD)GetDlgItemInt(hwnd, IDC_intINTERLEAVEEDIT,
NULL, FALSE);
gapOpt[gnCurStream]->dwInterleaveEvery = dw;
gapOpt[gnCurStream]->dwFlags |= AVICOMPRESSF_INTERLEAVE;
} else {
// why not remember edit box entry anyway?
dw = (DWORD)GetDlgItemInt(hwnd, IDC_intINTERLEAVEEDIT,
NULL, FALSE);
gapOpt[gnCurStream]->dwInterleaveEvery = dw;
gapOpt[gnCurStream]->dwFlags &=~AVICOMPRESSF_INTERLEAVE;
}
}
// fall through (AAAAaaaahhhhh.....)
case IDCANCEL:
EndDialog(hwnd, wParam == IDOK);
break;
case IDC_intOPTIONS:
StreamOptions(hwnd);
break;
//
// Somebody chose a new stream. Do we need to grey InterleaveOpts?
// Set the current stream.
//
case IDC_intCHOOSESTREAM:
if (HIWORD(lParam) != CBN_SELCHANGE)
break;
NewStreamChosen(hwnd);
break;
case IDC_intINTERLEAVE:
break;
default:
break;
}
break;
default:
return FALSE;
}
return FALSE;
}