IBackgroundCopyCallback2 インターフェイス (bits3_0.h)
このインターフェイスを実装して、ファイルのダウンロードが完了したことを 示す通知を受け取ります 。 クライアントは、ファイルのダウンロード状態をポーリングする代わりに、このインターフェイスを使用します。
通知を受信するには、 IBackgroundCopyJob::SetNotifyInterface メソッドを呼び出して 、IBackgroundCopyCallback 実装へのインターフェイス ポインターを指定します。 受信する通知を指定するには、 IBackgroundCopyJob::SetNotifyFlags メソッドを 呼び出します。
このインターフェイスと IBackgroundCopyCallback インターフェイスのすべてのメソッドを実装する必要があります。 たとえば、ファイル転送コールバックに登録しない場合、 FileTransferred メソッドは引き続き S_OKを返す必要があります。 ファイル転送コールバックを受け取りたくない場合は、代わりに IBackgroundCopyCallback を 実装するだけです。
継承
IBackgroundCopyCallback2 インターフェイスは、IBackgroundCopyCallback から継承されます。 IBackgroundCopyCallback2 には、次の種類のメンバーもあります。
メソッド
IBackgroundCopyCallback2 インターフェイスには、これらのメソッドがあります。
IBackgroundCopyCallback2::FileTransferred BITS は、BITS がファイルの転送を正常に完了したときに FileTransferred メソッドの実装を呼び出します。 |
注釈
このインターフェイスの実装の詳細については、 IBackgroundCopyCallback インターフェイスを参照してください。
例
次の例は、 IBackgroundCopyCallback2 の実装を示しています。 この例では、シングル スレッド アパートメント モデルの JobModification コールバックで再入可能な呼び出しを処理する方法も示します。
#define TWO_GB 2147483648 // 2GB
class CNotifyInterface : public IBackgroundCopyCallback2
{
LONG m_lRefCount;
LONG m_PendingJobModificationCount;
public:
//Constructor, Destructor
CNotifyInterface() {m_lRefCount = 1; m_PendingJobModificationCount = 0;};
~CNotifyInterface() {};
//IUnknown
HRESULT __stdcall QueryInterface(REFIID riid, LPVOID *ppvObj);
ULONG __stdcall AddRef();
ULONG __stdcall Release();
//IBackgroundCopyCallback methods
HRESULT __stdcall JobTransferred(IBackgroundCopyJob* pJob);
HRESULT __stdcall JobError(IBackgroundCopyJob* pJob, IBackgroundCopyError* pError);
HRESULT __stdcall JobModification(IBackgroundCopyJob* pJob, DWORD dwReserved);
HRESULT __stdcall FileTransferred(IBackgroundCopyJob* pJob, IBackgroundCopyFile* pFile);
};
HRESULT CNotifyInterface::QueryInterface(REFIID riid, LPVOID* ppvObj)
{
if (riid == __uuidof(IUnknown) ||
riid == __uuidof(IBackgroundCopyCallback) ||
riid == __uuidof(IBackgroundCopyCallback2))
{
*ppvObj = this;
}
else
{
*ppvObj = NULL;
return E_NOINTERFACE;
}
AddRef();
return NOERROR;
}
ULONG CNotifyInterface::AddRef()
{
return InterlockedIncrement(&m_lRefCount);
}
ULONG CNotifyInterface::Release()
{
ULONG ulCount = InterlockedDecrement(&m_lRefCount);
if(0 == ulCount)
{
delete this;
}
return ulCount;
}
HRESULT CNotifyInterface::JobTransferred(IBackgroundCopyJob* pJob)
{
HRESULT hr;
//Add logic that will not block the callback thread. If you need to perform
//extensive logic at this time, consider creating a separate thread to perform
//the work.
hr = pJob->Complete();
if (FAILED(hr))
{
//Handle error. BITS probably was unable to rename one or more of the
//temporary files. See the Remarks section of the IBackgroundCopyJob::Complete
//method for more details.
}
//If you do not return S_OK, BITS continues to call this callback.
return S_OK;
}
HRESULT CNotifyInterface::JobError(IBackgroundCopyJob* pJob, IBackgroundCopyError* pError)
{
HRESULT hr;
BG_FILE_PROGRESS Progress;
BG_ERROR_CONTEXT Context;
HRESULT ErrorCode = S_OK;
WCHAR* pszJobName = NULL;
WCHAR* pszErrorDescription = NULL;
BOOL IsError = TRUE;
//Use pJob and pError to retrieve information of interest. For example,
//if the job is an upload reply, call the IBackgroundCopyError::GetError method
//to determine the context in which the job failed. If the context is
//BG_JOB_CONTEXT_REMOTE_APPLICATION, the server application that received the
//upload file failed.
hr = pError->GetError(&Context, &ErrorCode);
//If the proxy or server does not support the Content-Range header or if
//antivirus software removes the range requests, BITS returns BG_E_INSUFFICIENT_RANGE_SUPPORT.
//This implementation tries to switch the job to foreground priority, so
//the content has a better chance of being successfully downloaded.
if (BG_E_INSUFFICIENT_RANGE_SUPPORT == ErrorCode)
{
hr = pError->GetFile(&pFile);
hr = pFile->GetProgress(&Progress);
if (BG_SIZE_UNKNOWN == Progress.BytesTotal)
{
//The content is dynamic, do not change priority. Handle as an error.
}
else if (Progress.BytesTotal > TWO_GB)
{
//BITS does not use range requests if the content is less than 2 GB.
//However, if the content is greater than 2 GB, BITS
//uses 2 GB ranges to download the file, so switching to foreground
//priority will not help.
}
else
{
hr = pJob->SetPriority(BG_JOB_PRIORITY_FOREGROUND);
hr = pJob->Resume();
IsError = FALSE;
}
pFile->Release();
}
if (TRUE == IsError)
{
hr = pJob->GetDisplayName(&pszJobName);
hr = pError->GetErrorDescription(LANGIDFROMLCID(GetThreadLocale()), &pszErrorDescription);
if (pszJobName && pszErrorDescription)
{
//Do something with the job name and description.
}
CoTaskMemFree(pszJobName);
CoTaskMemFree(pszErrorDescription);
}
//If you do not return S_OK, BITS continues to call this callback.
return S_OK;
}
HRESULT CNotifyInterface::JobModification(IBackgroundCopyJob* pJob, DWORD dwReserved)
{
HRESULT hr;
WCHAR* pszJobName = NULL;
BG_JOB_PROGRESS Progress;
BG_JOB_STATE State;
//If you are already processing a callback, ignore this notification.
if (InterlockedCompareExchange(&m_PendingJobModificationCount, 1, 0) == 1)
{
return S_OK;
}
hr = pJob->GetDisplayName(&pszJobName);
if (SUCCEEDED(hr))
{
hr = pJob->GetProgress(&Progress);
if (SUCCEEDED(hr))
{
hr = pJob->GetState(&State);
if (SUCCEEDED(hr))
{
//Do something with the progress and state information.
//BITS generates a high volume of modification
//callbacks. Use this callback with discretion. Consider creating a timer and
//polling for state and progress information.
}
}
CoTaskMemFree(pszJobName);
}
m_PendingJobModificationCount = 0;
return S_OK;
}
HRESULT CNotifyInterface::FileTransferred(IBackgroundCopyJob* pJob, IBackgroundCopyFile* pFile)
{
HRESULT hr = S_OK;
IBackgroundCopyFile3* pFile3 = NULL;
BOOL IsValid = FALSE;
hr = pFile->QueryInterface(__uuidof(IBackgroundCopyFile3), (void**)&pFile3);
if (SUCCEEDED(hr))
{
// Add code to validate downloaded content and set IsValid.
hr = pFile3->SetValidationState(IsValid);
if (FAILED(hr))
{
// Handle error
}
pFile3->Release();
}
return S_OK;
}
要件
要件 | 値 |
---|---|
サポートされている最小のクライアント | Windows Vista |
サポートされている最小のサーバー | Windows Server 2008 |
対象プラットフォーム | Windows |
ヘッダー | bits3_0.h (Bits.h を含む) |