酱坛子

专注C++技术 在这里写下自己的学习心得 感悟 和大家讨论 共同进步(欢迎批评!!!)

  C++博客 :: 首页 :: 联系 :: 聚合  :: 管理
  66 Posts :: 16 Stories :: 236 Comments :: 0 Trackbacks

公告

王一伟 湖南商学院毕业 电子信息工程专业

常用链接

留言簿(19)

我参与的团队

搜索

  •  

积分与排名

  • 积分 - 382160
  • 排名 - 63

最新随笔

最新评论

阅读排行榜

评论排行榜

小弟想实现将硬盘某个文件夹下的文件刻录到光盘上

我不想使用第三方的刻录软件

不知道XP系统是否有提供这样的接口,哪位大虾知道请给小弟一点提示:)给个思考的方向就可以了

谢过了
posted on 2006-10-30 11:41 @王一伟 阅读(1645) 评论(5)  编辑 收藏 引用

Feedback

# re: 求实现方法!! 2006-10-30 12:07 CornerZhang
XP是自带这个功能的,我的是English版,只要“IMAPI CD-Burning COM Service”这个系统服务开着,就可以的。怎么用你google一下吧  回复  更多评论
  

# re: 求实现方法!! 2006-10-30 12:22 王一伟
<转:在网上看见的别人写的东西>


关于使用IMAPI的一些问题
工作中要求使用IMAPI写一个小的刻录测试程序,苦于资料较少,但最后大体理出一些内容,并初步测试通过,把一些心得和问题记录在此。希望交流一下。程序基于MSDN (April 2004)和微软工程师Paul DiLascia的一个例程,并作了一定修改,环境是XPsp2和VS.net2003(MFC)。

1.IDiscMaster::ProgressAdvise()的使用
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// 声明一个类,从接口继承
class CDiscMasterProgressEvents : public CComQIPtr<IDiscMasterProgressEvents>
{
public:
HRESULT m_hr;
int m_nRefCount;

// 进度条控件
CProgressCtrl * m_pCtlProgress;

// 构造和析构
CDiscMasterProgressEvents(/*CWnd * pWnd*/);

ULONG __stdcall AddRef(void);
ULONG __stdcall Release(void);
HRESULT __stdcall QueryInterface(REFIID riid, void **ppv);

// 以下为一些消息的回调函数
// 刻录结束后被IMAPI调用(Reports that the burn is fully complete.IDiscMaster::RecordDisc()结束时调用)
HRESULT __stdcall NotifyBurnComplete(HRESULT status);

.............................
};

/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
//类实现代码
ULONG __stdcall CDiscMasterProgressEvents::AddRef(void)
{
return ++m_nRefCount;
}

ULONG __stdcall CDiscMasterProgressEvents::Release(void)
{
if (--m_nRefCount)
return m_nRefCount;
delete this;
return 0L;
}

HRESULT __stdcall CDiscMasterProgressEvents::QueryInterface(REFIID riid, void **ppv)
{
if(riid == IID_IUnknown)
{
*ppv = (LPUNKNOWN)(IDiscMasterProgressEvents*)this;
m_nRefCount++;
return NOERROR;
}
else if(riid == IID_IDiscMasterProgressEvents)
{
*ppv = (IDiscMasterProgressEvents*)this;
m_nRefCount++;
return NOERROR;
}
else
{
*ppv = NULL;
return E_NOINTERFACE;
}
}

// 刻录结束后被IMAPI调用
HRESULT __stdcall CDiscMasterProgressEvents::NotifyBurnComplete(HRESULT status)
{
if (status == S_OK)
{
AfxMessageBox("刻录完毕");
if (m_pCtlProgress)
{
m_pCtlProgress->SetPos(0);
}
}
return status ;
}
/////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// 类的使用
..........

CDiscMasterProgressEvents * pEvent = new CDiscMasterProgressEvents;
pEvent->m_pCtlProgress = &(pDlg->m_progBurning);
UINT_PTR m_nCookie = 0;

// dm为CDIscMater(class CDiscMaster : public CComQIPtr<IDiscMaster>)
hr = dm.ProgressAdvise(pEvent, &m_nCookie); // ok

..........
hr = dm.RecodeDisc(0, 0);

2. IJolietDiscMaster::AddData()前,所有IStream.Release()和子ISubStorage.Release();
如果没有释放,我遇到的问题是数据刻录到光盘中,但却看不到;根IStorage在AddData后Release()

3.文件名和文件夹名的问题
pStorage->CreateStream(const WCHAR* pwcsName, ....); // 参数对应刻录后的文件名
pStorage->CreateStorage(const WCHAR* pwcsName, .....); // 参数对应刻录后的子文件夹名
上面的两个参数都不能超过31个字符,不能使用某些特殊字符(!等),否则会造成创建失败;没有查到资料如何处理超过31字符的文件名,如果有解决问题的方法,希望能和大家交流。

4.光盘的Volume(没有测试)
通过IJolietDiscMaster::GetJolietProperties()得到;对应的一个IJolietDiscMaster::SetJolietProperties()

5.IDiscRecorder::GetPath(BSTR* pbstrPath)(未解决)
得到刻录机路径,MSDN说明:
Parameters
pbstrPath
[out] Path to the disc recorder. This can be a drive letter such as 'E:\', or a full path such as '\Device\CdRom0'.
测试程序中得到的是后者,还没有找到合理的方式来把两者挂钩,特别是多光驱情况下
  回复  更多评论
  

# re: 求实现方法!! 2006-10-30 12:23 王一伟
@CornerZhang
多谢了,我查到的也是这个  回复  更多评论
  

# re: 求实现方法!! 2006-10-30 12:25 王一伟
http://download.microsoft.com/download/1/6/4/164c2a20-aeb0-460f-907d-985d83e86bd4/CQA0404.exe

用IMAPI实现CD刻录和设备查找

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也足够了。
  回复  更多评论
  

# re: 求实现方法!! 2006-10-30 13:53 王一伟
注意必须在服务管理器中开启IMAPI CD-Burning COM Service服务  回复  更多评论
  


只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理