MFC没有提供CD刻录的类,但WindowsXp内置了写入CD的支持。如果只是复制文件和目录,可以用shell's ICDBurn接口。如果要刻录音频或更近一步的实现对音频的控制,那么下面会介绍一个专用的API。
用ICDBurn有三个步骤,HasRecordableDrive扫描系统内可写的CD驱动器,找到的话返回TRUE。GetRecorderDriveLetter返回可写驱动器的盘符。最后刻录指令通知Windows从“集结区”向可写CD拷贝数据。“集结区”是一个专用的文件夹,通常是"%userprofile%\Local Settings\Application Data\Microsoft\CD Burning", 但还是应该调用SHGetFolderPath和参数CSIDL_CDBURN_AREA获得准确的目录名。开发者会经常用到SHGetFolderPath,因为用户会经常手动或用PowerTools/TweakUI等工具改变刻录的目录。
笔者写了一个很短的类CCDBurn来封装ICDBurn.这个结构用CLSID_CDBurn调用CoCreateInstance,读者可调试运行。
CCDBurn burner;
if (!burner.HasRecordableDrive()) {
printf("Oops—No recordable drive!\n");
} else {
CString dl =
burner.GetRecorderDriveLetter();
printf("Default Recorder drive letter =
%s\n", (LPCTSTR)dl);
}
驱动器盘符是在驱动器记录属性中启动CD记录功能的那个驱动器。只有一个驱动器可以设置这种属性。假定HasRecordableDrive 返回TRUE,也就是电脑至少有一个可记录的CD驱动器,你要做的就是把文件拷贝到集结区内,这就是刻录。
由于笔者很严谨,我又加入了另一个步骤GetBurnFolderPath,调用SHGetSpecialFolderPath 得到一个CString形式的刻录文件目录:
CString path = burner.GetBurnFolderPath();
如果要写入音乐,或找到其他刻录驱动器,或者获得更详细的信息:如驱动器是否是可写的。对此,windows xp提供了IMAPI,它是Image Mastering API的缩写,不要和 MAPI——用于e-mail的Messaging API 混淆。IMAPI提供了COM接口可以寻找可刻录驱动器和写入数据或者指向你最近用过的光驱。
由于与COM的冲突会引起较大的问题,笔者写了一个小的类库,IMAPITools,解决了大多数问题。为了说明怎么用这个类库笔者还写了一个程序CDINFO。CDINFO在控制台窗口中显示CD记录体的信息。
只要掌握了IMAPI,CD刻录就不难。但IMAPI很庞大,在讲完基础后再简要的说一下这个问题。
首先,CDINFO创建了一个对象显示驱动器盘符和刻录路径。接着,创建CDiscMaster 打开IMAPI session:
CDiscMaster dm; // create IDiscMaster
if (!dm.Open()) {
printf("Oops: ...");
return;
CDiscMaster 封装了第一个主IMAPI 接口IDiscMaster。它调用CoCreateInstance创建 Microsoft MSDiscMasterObj 对象然后得到IDiscMaster 接口。IDiscMaster 例举出格式和记录器,选择活动记录器等。
CD分为两种:记录音频的CD-Audio和存储文件的CD-ROM。它们的格式分别为Redbook和Joliet。RedBook是Philips和Sony在80年代根据“redbook" 标准制定的;Joliet是微软扩展ISO-9660制定的一种CD-ROM文件格式。微软在win95时代制定Joliet是为了扩展ISO-9660从而支持长文件名和多层目录。
IDiscMaster::EnumDiscMasterFormats可以例举出刻录器支持哪种格式,但是这种方法很耗费资源,我用一个简单的方法实现了这个功能,CDiscMaster::GetSupportedFormats在数组IID中返回格式:
const MAXNFORMATS = 2;
IID fmts[MAXNFORMATS];
int nFormats = dm.GetSupportedFormats(fmts,MAXNFORMATS);
fmts数组中包含了支持的格式,IID_IRedbookDiscMaster和IID_IJoiletDiscMaster,而再也不用使用IEnumDiscMasterFormats了。有经验的程序员可能会提出疑问:为什么IMAPI的设计者选择一个这么复杂的API来获得只有两种的支持格式,一行简单的代码就可以提供足够的带宽来传送信息。这只有设计者知道答案,也许他们希望有人会用VB写一个音频记录器。不管怎样,只要你使用了IMAPITools,你就会忘记COM。
一旦打开一个session,就可以实现查询单个的刻录驱动器。但IMAPI会再一次使用COM例举刻录器,而笔者又用一个类隐藏了这个结构。
CDiscRecorder dr;
CDiscRecorderIterator itr(dm); // dm=CDiscMaster
while (itr.Next(dr)) {
// do something
}
程序每次调用下一步,累加器就会抓取下一段记录到CDiscRecorder中。CDiscRecorder封装了其他大的IMAPI接口,CDiscRecorder代表了可记录CD设备。CDiscRecorder提供了打开记录器的方法,询问它的类型(CD-R or CD-RW)和路径,得到设备属性,弹出CD等等。CDINFO演示了如何用CDiscRecorder获得记录器的所有信息。
要把数据写入光盘,就要使用IJolietDiscMaster或IRedbookDiscMaster,或者也可以用IMAPITools:
dm.SetActiveDiscRecorder(dr); // select recorder
CJolietDiscMaster jdm(dm); // get joliet interface
jdm保存了IJolietDiscMaster接口,可以调用任何IJolietDiscMaster方法。AddData是写入数据方法的一种;它需要一个COM IStorage指针。写入音频也是一样的,除非是用IRedbookDiscMaster和AddAudioTrackBlocks 添加未处理的音频数据(44.1 KHz, 16-bit RAW, WAV 文件也相同)。创建多音轨可以用Create/CloseAudioTrack。AddData 和AddAudioTrackBlocks实际上不往光盘中写数据,而是写到集结区中。如果要真正的移动数据,还需要调用RecordDisc:
BOOL bSimulate=FALSE;
BOOL bEjectAfterBurn=TRUE;
dm.RecordDisc(bSimulate, bEjectAfterBurn);
dm.Close();
bSimulate=TRUE会调用RecordDisc模拟刻录但实际上并未写入。Windows检查全部预刻录列表并刻录,但实际上并没有写入。这可以让开发者测试和调试软件而并不需要花很多时间真正的刻录光盘。
以上简单介绍了一下IMAPI,但这足够你开始刻录编程了。大多数人都会调用IMAPI,除非要写一个复杂的备份程序或音频纪录器。对于普通的拷贝文件,要用到的就是ICDBurn。即使你要显示出可记录驱动器列表,用IMAPI也足够了。
文中代码下载:
http://download.microsoft.com/download/1/6/4/164c2a20-aeb0-460f-907d-985d83e86bd4/CQA0404.exe