mirror of
https://github.com/mozilla/gecko-dev.git
synced 2025-01-01 05:43:46 +00:00
744 lines
24 KiB
C++
744 lines
24 KiB
C++
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#ifndef DOM_CAMERA_CAMERACONTROLIMPL_H
|
|
#define DOM_CAMERA_CAMERACONTROLIMPL_H
|
|
|
|
#include "mozilla/Attributes.h"
|
|
#include "nsDOMFile.h"
|
|
#include "nsProxyRelease.h"
|
|
#include "DictionaryHelpers.h"
|
|
#include "nsIDOMDeviceStorage.h"
|
|
#include "DOMCameraManager.h"
|
|
#include "DOMCameraPreview.h"
|
|
#include "ICameraControl.h"
|
|
#include "CameraCommon.h"
|
|
|
|
namespace mozilla {
|
|
|
|
class GetPreviewStreamTask;
|
|
class StartPreviewTask;
|
|
class StopPreviewTask;
|
|
class AutoFocusTask;
|
|
class TakePictureTask;
|
|
class StartRecordingTask;
|
|
class StopRecordingTask;
|
|
class SetParameterTask;
|
|
class GetParameterTask;
|
|
class GetPreviewStreamVideoModeTask;
|
|
class ReleaseHardwareTask;
|
|
|
|
class DOMCameraPreview;
|
|
class RecorderProfileManager;
|
|
|
|
class CameraControlImpl : public ICameraControl
|
|
{
|
|
friend class GetPreviewStreamTask;
|
|
friend class StartPreviewTask;
|
|
friend class StopPreviewTask;
|
|
friend class AutoFocusTask;
|
|
friend class TakePictureTask;
|
|
friend class StartRecordingTask;
|
|
friend class StopRecordingTask;
|
|
friend class SetParameterTask;
|
|
friend class GetParameterTask;
|
|
friend class GetPreviewStreamVideoModeTask;
|
|
friend class ReleaseHardwareTask;
|
|
|
|
public:
|
|
CameraControlImpl(uint32_t aCameraId, nsIThread* aCameraThread, uint64_t aWindowId);
|
|
|
|
nsresult GetPreviewStream(idl::CameraSize aSize, nsICameraPreviewStreamCallback* onSuccess, nsICameraErrorCallback* onError);
|
|
nsresult StartPreview(DOMCameraPreview* aDOMPreview);
|
|
void StopPreview();
|
|
nsresult AutoFocus(nsICameraAutoFocusCallback* onSuccess, nsICameraErrorCallback* onError);
|
|
nsresult TakePicture(const idl::CameraSize& aSize, int32_t aRotation, const nsAString& aFileFormat, idl::CameraPosition aPosition, uint64_t aDateTime, nsICameraTakePictureCallback* onSuccess, nsICameraErrorCallback* onError);
|
|
nsresult StartRecording(idl::CameraStartRecordingOptions* aOptions, nsIFile* aFolder, const nsAString& aFilename, nsICameraStartRecordingCallback* onSuccess, nsICameraErrorCallback* onError);
|
|
nsresult StopRecording();
|
|
nsresult GetPreviewStreamVideoMode(idl::CameraRecorderOptions* aOptions, nsICameraPreviewStreamCallback* onSuccess, nsICameraErrorCallback* onError);
|
|
nsresult ReleaseHardware(nsICameraReleaseCallback* onSuccess, nsICameraErrorCallback* onError);
|
|
|
|
nsresult Set(uint32_t aKey, const nsAString& aValue);
|
|
nsresult Get(uint32_t aKey, nsAString& aValue);
|
|
nsresult Set(uint32_t aKey, double aValue);
|
|
nsresult Get(uint32_t aKey, double* aValue);
|
|
nsresult Set(JSContext* aCx, uint32_t aKey, const JS::Value& aValue, uint32_t aLimit);
|
|
nsresult Get(JSContext* aCx, uint32_t aKey, JS::Value* aValue);
|
|
nsresult Set(nsICameraShutterCallback* aOnShutter);
|
|
nsresult Get(nsICameraShutterCallback** aOnShutter);
|
|
nsresult Set(nsICameraClosedCallback* aOnClosed);
|
|
nsresult Get(nsICameraClosedCallback** aOnClosed);
|
|
nsresult Set(nsICameraRecorderStateChange* aOnRecorderStateChange);
|
|
nsresult Get(nsICameraRecorderStateChange** aOnRecorderStateChange);
|
|
nsresult Set(nsICameraPreviewStateChange* aOnPreviewStateChange);
|
|
nsresult Get(nsICameraPreviewStateChange** aOnPreviewStateChange);
|
|
nsresult Set(uint32_t aKey, const idl::CameraSize& aSize);
|
|
nsresult Get(uint32_t aKey, idl::CameraSize& aSize);
|
|
nsresult Get(uint32_t aKey, int32_t* aValue);
|
|
|
|
nsresult SetFocusAreas(JSContext* aCx, const JS::Value& aValue)
|
|
{
|
|
return Set(aCx, CAMERA_PARAM_FOCUSAREAS, aValue, mMaxFocusAreas);
|
|
}
|
|
|
|
nsresult SetMeteringAreas(JSContext* aCx, const JS::Value& aValue)
|
|
{
|
|
return Set(aCx, CAMERA_PARAM_METERINGAREAS, aValue, mMaxMeteringAreas);
|
|
}
|
|
|
|
already_AddRefed<RecorderProfileManager> GetRecorderProfileManager();
|
|
uint32_t GetCameraId() { return mCameraId; }
|
|
|
|
virtual const char* GetParameter(const char* aKey) = 0;
|
|
virtual const char* GetParameterConstChar(uint32_t aKey) = 0;
|
|
virtual double GetParameterDouble(uint32_t aKey) = 0;
|
|
virtual int32_t GetParameterInt32(uint32_t aKey) = 0;
|
|
virtual void GetParameter(uint32_t aKey, nsTArray<idl::CameraRegion>& aRegions) = 0;
|
|
virtual void GetParameter(uint32_t aKey, idl::CameraSize& aSize) = 0;
|
|
virtual void SetParameter(const char* aKey, const char* aValue) = 0;
|
|
virtual void SetParameter(uint32_t aKey, const char* aValue) = 0;
|
|
virtual void SetParameter(uint32_t aKey, double aValue) = 0;
|
|
virtual void SetParameter(uint32_t aKey, const nsTArray<idl::CameraRegion>& aRegions) = 0;
|
|
virtual void SetParameter(uint32_t aKey, const idl::CameraSize& aSize) = 0;
|
|
virtual nsresult GetVideoSizes(nsTArray<idl::CameraSize>& aVideoSizes) = 0;
|
|
virtual nsresult PushParameters() = 0;
|
|
virtual void Shutdown();
|
|
|
|
bool ReceiveFrame(void* aBuffer, ImageFormat aFormat, FrameBuilder aBuilder);
|
|
void OnShutter();
|
|
void OnClosed();
|
|
void OnRecorderStateChange(const nsString& aStateMsg, int32_t aStatus, int32_t aTrackNumber);
|
|
|
|
enum PreviewState {
|
|
PREVIEW_STOPPED,
|
|
PREVIEW_STARTED
|
|
};
|
|
void OnPreviewStateChange(PreviewState aNewState);
|
|
|
|
uint64_t GetWindowId()
|
|
{
|
|
return mWindowId;
|
|
}
|
|
|
|
protected:
|
|
virtual ~CameraControlImpl();
|
|
|
|
virtual nsresult GetPreviewStreamImpl(GetPreviewStreamTask* aGetPreviewStream) = 0;
|
|
virtual nsresult StartPreviewImpl(StartPreviewTask* aStartPreview) = 0;
|
|
virtual nsresult StopPreviewImpl(StopPreviewTask* aStopPreview) = 0;
|
|
virtual nsresult AutoFocusImpl(AutoFocusTask* aAutoFocus) = 0;
|
|
virtual nsresult TakePictureImpl(TakePictureTask* aTakePicture) = 0;
|
|
virtual nsresult StartRecordingImpl(StartRecordingTask* aStartRecording) = 0;
|
|
virtual nsresult StopRecordingImpl(StopRecordingTask* aStopRecording) = 0;
|
|
virtual nsresult PushParametersImpl() = 0;
|
|
virtual nsresult PullParametersImpl() = 0;
|
|
virtual nsresult GetPreviewStreamVideoModeImpl(GetPreviewStreamVideoModeTask* aGetPreviewStreamVideoMode) = 0;
|
|
virtual nsresult ReleaseHardwareImpl(ReleaseHardwareTask* aReleaseHardware) = 0;
|
|
virtual already_AddRefed<RecorderProfileManager> GetRecorderProfileManagerImpl() = 0;
|
|
|
|
void OnShutterInternal();
|
|
void OnClosedInternal();
|
|
|
|
uint32_t mCameraId;
|
|
nsCOMPtr<nsIThread> mCameraThread;
|
|
uint64_t mWindowId;
|
|
nsString mFileFormat;
|
|
uint32_t mMaxMeteringAreas;
|
|
uint32_t mMaxFocusAreas;
|
|
PreviewState mPreviewState;
|
|
|
|
/**
|
|
* 'mDOMPreview' is a raw pointer to the object that will receive incoming
|
|
* preview frames. This is guaranteed to be valid, or null.
|
|
*
|
|
* It is set by a call to StartPreview(), and set to null on StopPreview().
|
|
* It is up to the caller to ensure that the object will not disappear
|
|
* out from under this pointer--usually by calling NS_ADDREF().
|
|
*/
|
|
DOMCameraPreview* mDOMPreview;
|
|
|
|
nsMainThreadPtrHandle<nsICameraAutoFocusCallback> mAutoFocusOnSuccessCb;
|
|
nsMainThreadPtrHandle<nsICameraErrorCallback> mAutoFocusOnErrorCb;
|
|
nsMainThreadPtrHandle<nsICameraTakePictureCallback> mTakePictureOnSuccessCb;
|
|
nsMainThreadPtrHandle<nsICameraErrorCallback> mTakePictureOnErrorCb;
|
|
nsMainThreadPtrHandle<nsICameraShutterCallback> mOnShutterCb;
|
|
nsMainThreadPtrHandle<nsICameraClosedCallback> mOnClosedCb;
|
|
nsMainThreadPtrHandle<nsICameraRecorderStateChange> mOnRecorderStateChangeCb;
|
|
nsMainThreadPtrHandle<nsICameraPreviewStateChange> mOnPreviewStateChangeCb;
|
|
|
|
private:
|
|
CameraControlImpl(const CameraControlImpl&) MOZ_DELETE;
|
|
CameraControlImpl& operator=(const CameraControlImpl&) MOZ_DELETE;
|
|
};
|
|
|
|
// Error result runnable
|
|
class CameraErrorResult : public nsRunnable
|
|
{
|
|
public:
|
|
CameraErrorResult(nsMainThreadPtrHandle<nsICameraErrorCallback> onError, const nsString& aErrorMsg, uint64_t aWindowId)
|
|
: mOnErrorCb(onError)
|
|
, mErrorMsg(aErrorMsg)
|
|
, mWindowId(aWindowId)
|
|
{ }
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (mOnErrorCb.get() && nsDOMCameraManager::IsWindowStillActive(mWindowId)) {
|
|
mOnErrorCb->HandleEvent(mErrorMsg);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
protected:
|
|
nsMainThreadPtrHandle<nsICameraErrorCallback> mOnErrorCb;
|
|
const nsString mErrorMsg;
|
|
uint64_t mWindowId;
|
|
};
|
|
|
|
// Return the resulting preview stream to JS. Runs on the main thread.
|
|
class GetPreviewStreamResult : public nsRunnable
|
|
{
|
|
public:
|
|
GetPreviewStreamResult(CameraControlImpl* aCameraControl, uint32_t aWidth, uint32_t aHeight, uint32_t aFramesPerSecond, nsMainThreadPtrHandle<nsICameraPreviewStreamCallback>& onSuccess, uint64_t aWindowId)
|
|
: mCameraControl(aCameraControl)
|
|
, mWidth(aWidth)
|
|
, mHeight(aHeight)
|
|
, mFramesPerSecond(aFramesPerSecond)
|
|
, mOnSuccessCb(onSuccess)
|
|
, mWindowId(aWindowId)
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~GetPreviewStreamResult()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
// Run() method is implementation specific.
|
|
NS_IMETHOD Run() MOZ_OVERRIDE;
|
|
|
|
protected:
|
|
nsRefPtr<CameraControlImpl> mCameraControl;
|
|
uint32_t mWidth;
|
|
uint32_t mHeight;
|
|
uint32_t mFramesPerSecond;
|
|
nsMainThreadPtrHandle<nsICameraPreviewStreamCallback> mOnSuccessCb;
|
|
uint64_t mWindowId;
|
|
};
|
|
|
|
// Get the desired preview stream.
|
|
class GetPreviewStreamTask : public nsRunnable
|
|
{
|
|
public:
|
|
GetPreviewStreamTask(CameraControlImpl* aCameraControl, idl::CameraSize aSize, nsICameraPreviewStreamCallback* onSuccess, nsICameraErrorCallback* onError)
|
|
: mSize(aSize)
|
|
, mCameraControl(aCameraControl)
|
|
, mOnSuccessCb(new nsMainThreadPtrHolder<nsICameraPreviewStreamCallback>(onSuccess))
|
|
, mOnErrorCb(new nsMainThreadPtrHolder<nsICameraErrorCallback>(onError))
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~GetPreviewStreamTask()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
nsresult rv = mCameraControl->GetPreviewStreamImpl(this);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
nsCOMPtr<nsIRunnable> cameraErrorResult = new CameraErrorResult(mOnErrorCb, NS_LITERAL_STRING("FAILURE"), mCameraControl->GetWindowId());
|
|
rv = NS_DispatchToMainThread(cameraErrorResult);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
idl::CameraSize mSize;
|
|
nsRefPtr<CameraControlImpl> mCameraControl;
|
|
nsMainThreadPtrHandle<nsICameraPreviewStreamCallback> mOnSuccessCb;
|
|
nsMainThreadPtrHandle<nsICameraErrorCallback> mOnErrorCb;
|
|
};
|
|
|
|
// Return the autofocus status to JS. Runs on the main thread.
|
|
class AutoFocusResult : public nsRunnable
|
|
{
|
|
public:
|
|
AutoFocusResult(bool aSuccess, nsMainThreadPtrHandle<nsICameraAutoFocusCallback> onSuccess, uint64_t aWindowId)
|
|
: mSuccess(aSuccess)
|
|
, mOnSuccessCb(onSuccess)
|
|
, mWindowId(aWindowId)
|
|
{ }
|
|
|
|
virtual ~AutoFocusResult() { }
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (mOnSuccessCb.get() && nsDOMCameraManager::IsWindowStillActive(mWindowId)) {
|
|
mOnSuccessCb->HandleEvent(mSuccess);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
protected:
|
|
bool mSuccess;
|
|
nsMainThreadPtrHandle<nsICameraAutoFocusCallback> mOnSuccessCb;
|
|
uint64_t mWindowId;
|
|
};
|
|
|
|
// Autofocus the camera.
|
|
class AutoFocusTask : public nsRunnable
|
|
{
|
|
public:
|
|
AutoFocusTask(CameraControlImpl* aCameraControl, bool aCancel, nsICameraAutoFocusCallback* onSuccess, nsICameraErrorCallback* onError)
|
|
: mCameraControl(aCameraControl)
|
|
, mCancel(aCancel)
|
|
, mOnSuccessCb(new nsMainThreadPtrHolder<nsICameraAutoFocusCallback>(onSuccess))
|
|
, mOnErrorCb(new nsMainThreadPtrHolder<nsICameraErrorCallback>(onError))
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~AutoFocusTask()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
nsresult rv = mCameraControl->AutoFocusImpl(this);
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
nsCOMPtr<nsIRunnable> cameraErrorResult = new CameraErrorResult(mOnErrorCb, NS_LITERAL_STRING("FAILURE"), mCameraControl->GetWindowId());
|
|
rv = NS_DispatchToMainThread(cameraErrorResult);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
nsRefPtr<CameraControlImpl> mCameraControl;
|
|
bool mCancel;
|
|
nsMainThreadPtrHandle<nsICameraAutoFocusCallback> mOnSuccessCb;
|
|
nsMainThreadPtrHandle<nsICameraErrorCallback> mOnErrorCb;
|
|
};
|
|
|
|
// Return the captured picture to JS. Runs on the main thread.
|
|
class TakePictureResult : public nsRunnable
|
|
{
|
|
public:
|
|
TakePictureResult(uint8_t* aData, uint64_t aLength, const nsAString& aMimeType, nsMainThreadPtrHandle<nsICameraTakePictureCallback> onSuccess, uint64_t aWindowId)
|
|
: mData(aData)
|
|
, mLength(aLength)
|
|
, mMimeType(aMimeType)
|
|
, mOnSuccessCb(onSuccess)
|
|
, mWindowId(aWindowId)
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~TakePictureResult()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
if (mOnSuccessCb.get() && nsDOMCameraManager::IsWindowStillActive(mWindowId)) {
|
|
nsCOMPtr<nsIDOMBlob> image = new nsDOMMemoryFile(static_cast<void*>(mData), static_cast<uint64_t>(mLength), mMimeType);
|
|
mOnSuccessCb->HandleEvent(image);
|
|
} else {
|
|
delete[] mData;
|
|
}
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
return NS_OK;
|
|
}
|
|
|
|
protected:
|
|
uint8_t* mData;
|
|
uint64_t mLength;
|
|
nsString mMimeType;
|
|
nsMainThreadPtrHandle<nsICameraTakePictureCallback> mOnSuccessCb;
|
|
uint64_t mWindowId;
|
|
};
|
|
|
|
// Capture a still image with the camera.
|
|
class TakePictureTask : public nsRunnable
|
|
{
|
|
public:
|
|
TakePictureTask(CameraControlImpl* aCameraControl, bool aCancel, const idl::CameraSize& aSize, int32_t aRotation, const nsAString& aFileFormat, idl::CameraPosition aPosition, uint64_t aDateTime, nsICameraTakePictureCallback* onSuccess, nsICameraErrorCallback* onError)
|
|
: mCameraControl(aCameraControl)
|
|
, mCancel(aCancel)
|
|
, mSize(aSize)
|
|
, mRotation(aRotation)
|
|
, mFileFormat(aFileFormat)
|
|
, mPosition(aPosition)
|
|
, mDateTime(aDateTime)
|
|
, mOnSuccessCb(new nsMainThreadPtrHolder<nsICameraTakePictureCallback>(onSuccess))
|
|
, mOnErrorCb(new nsMainThreadPtrHolder<nsICameraErrorCallback>(onError))
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~TakePictureTask()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
nsresult rv = mCameraControl->TakePictureImpl(this);
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
nsCOMPtr<nsIRunnable> cameraErrorResult = new CameraErrorResult(mOnErrorCb, NS_LITERAL_STRING("FAILURE"), mCameraControl->GetWindowId());
|
|
rv = NS_DispatchToMainThread(cameraErrorResult);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
nsRefPtr<CameraControlImpl> mCameraControl;
|
|
bool mCancel;
|
|
idl::CameraSize mSize;
|
|
int32_t mRotation;
|
|
nsString mFileFormat;
|
|
idl::CameraPosition mPosition;
|
|
uint64_t mDateTime;
|
|
nsMainThreadPtrHandle<nsICameraTakePictureCallback> mOnSuccessCb;
|
|
nsMainThreadPtrHandle<nsICameraErrorCallback> mOnErrorCb;
|
|
};
|
|
|
|
// Return the result of starting recording. Runs on the main thread.
|
|
class StartRecordingResult : public nsRunnable
|
|
{
|
|
public:
|
|
StartRecordingResult(nsMainThreadPtrHandle<nsICameraStartRecordingCallback> onSuccess, uint64_t aWindowId)
|
|
: mOnSuccessCb(onSuccess)
|
|
, mWindowId(aWindowId)
|
|
{ }
|
|
|
|
virtual ~StartRecordingResult() { }
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (mOnSuccessCb.get() && nsDOMCameraManager::IsWindowStillActive(mWindowId)) {
|
|
mOnSuccessCb->HandleEvent();
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
protected:
|
|
nsMainThreadPtrHandle<nsICameraStartRecordingCallback> mOnSuccessCb;
|
|
uint64_t mWindowId;
|
|
};
|
|
|
|
// Start video recording.
|
|
class StartRecordingTask : public nsRunnable
|
|
{
|
|
public:
|
|
StartRecordingTask(CameraControlImpl* aCameraControl, idl::CameraStartRecordingOptions aOptions, nsIFile* aFolder, const nsAString& aFilename, nsICameraStartRecordingCallback* onSuccess, nsICameraErrorCallback* onError, uint64_t aWindowId)
|
|
: mCameraControl(aCameraControl)
|
|
, mOptions(aOptions)
|
|
, mFolder(aFolder)
|
|
, mFilename(aFilename)
|
|
, mOnSuccessCb(new nsMainThreadPtrHolder<nsICameraStartRecordingCallback>(onSuccess))
|
|
, mOnErrorCb(new nsMainThreadPtrHolder<nsICameraErrorCallback>(onError))
|
|
, mWindowId(aWindowId)
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~StartRecordingTask()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
nsresult rv = mCameraControl->StartRecordingImpl(this);
|
|
DOM_CAMERA_LOGT("%s:%d : result %d\n", __func__, __LINE__, rv);
|
|
|
|
// dispatch the callback
|
|
nsCOMPtr<nsIRunnable> startRecordingResult;
|
|
if (NS_SUCCEEDED(rv)) {
|
|
startRecordingResult = new StartRecordingResult(mOnSuccessCb, mWindowId);
|
|
} else {
|
|
startRecordingResult = new CameraErrorResult(mOnErrorCb, NS_LITERAL_STRING("FAILURE"), mWindowId);
|
|
}
|
|
rv = NS_DispatchToMainThread(startRecordingResult);
|
|
if (NS_FAILED(rv)) {
|
|
DOM_CAMERA_LOGE("Failed to dispatch start recording result to main thread (%d)!", rv);
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
nsRefPtr<CameraControlImpl> mCameraControl;
|
|
idl::CameraStartRecordingOptions mOptions;
|
|
nsCOMPtr<nsIFile> mFolder;
|
|
nsString mFilename;
|
|
nsMainThreadPtrHandle<nsICameraStartRecordingCallback> mOnSuccessCb;
|
|
nsMainThreadPtrHandle<nsICameraErrorCallback> mOnErrorCb;
|
|
uint64_t mWindowId;
|
|
};
|
|
|
|
// Stop video recording.
|
|
class StopRecordingTask : public nsRunnable
|
|
{
|
|
public:
|
|
StopRecordingTask(CameraControlImpl* aCameraControl)
|
|
: mCameraControl(aCameraControl)
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~StopRecordingTask()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
nsresult rv = mCameraControl->StopRecordingImpl(this);
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsRefPtr<CameraControlImpl> mCameraControl;
|
|
};
|
|
|
|
// Start the preview.
|
|
class StartPreviewTask : public nsRunnable
|
|
{
|
|
public:
|
|
StartPreviewTask(CameraControlImpl* aCameraControl, DOMCameraPreview* aDOMPreview)
|
|
: mCameraControl(aCameraControl)
|
|
, mDOMPreview(aDOMPreview)
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~StartPreviewTask()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
nsresult rv = mCameraControl->StartPreviewImpl(this);
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
return NS_OK;
|
|
}
|
|
|
|
nsRefPtr<CameraControlImpl> mCameraControl;
|
|
DOMCameraPreview* mDOMPreview; // DOMCameraPreview NS_ADDREFs itself for us
|
|
};
|
|
|
|
// Stop the preview.
|
|
class StopPreviewTask : public nsRunnable
|
|
{
|
|
public:
|
|
StopPreviewTask(CameraControlImpl* aCameraControl)
|
|
: mCameraControl(aCameraControl)
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~StopPreviewTask()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
mCameraControl->StopPreviewImpl(this);
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
nsRefPtr<CameraControlImpl> mCameraControl;
|
|
};
|
|
|
|
// Get the video mode preview stream.
|
|
class GetPreviewStreamVideoModeTask : public nsRunnable
|
|
{
|
|
public:
|
|
GetPreviewStreamVideoModeTask(CameraControlImpl* aCameraControl, idl::CameraRecorderOptions aOptions, nsICameraPreviewStreamCallback* onSuccess, nsICameraErrorCallback* onError)
|
|
: mCameraControl(aCameraControl)
|
|
, mOptions(aOptions)
|
|
, mOnSuccessCb(new nsMainThreadPtrHolder<nsICameraPreviewStreamCallback>(onSuccess))
|
|
, mOnErrorCb(new nsMainThreadPtrHolder<nsICameraErrorCallback>(onError))
|
|
{ }
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
DOM_CAMERA_LOGI("%s:%d -- BEFORE IMPL\n", __func__, __LINE__);
|
|
nsresult rv = mCameraControl->GetPreviewStreamVideoModeImpl(this);
|
|
DOM_CAMERA_LOGI("%s:%d -- AFTER IMPL : rv = %d\n", __func__, __LINE__, rv);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
nsCOMPtr<nsIRunnable> cameraErrorResult = new CameraErrorResult(mOnErrorCb, NS_LITERAL_STRING("FAILURE"), mCameraControl->GetWindowId());
|
|
rv = NS_DispatchToMainThread(cameraErrorResult);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
nsRefPtr<CameraControlImpl> mCameraControl;
|
|
idl::CameraRecorderOptions mOptions;
|
|
nsMainThreadPtrHandle<nsICameraPreviewStreamCallback> mOnSuccessCb;
|
|
nsMainThreadPtrHandle<nsICameraErrorCallback> mOnErrorCb;
|
|
};
|
|
|
|
// Return the result of releasing the camera hardware. Runs on the main thread.
|
|
class ReleaseHardwareResult : public nsRunnable
|
|
{
|
|
public:
|
|
ReleaseHardwareResult(nsMainThreadPtrHandle<nsICameraReleaseCallback> onSuccess, uint64_t aWindowId)
|
|
: mOnSuccessCb(onSuccess)
|
|
, mWindowId(aWindowId)
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~ReleaseHardwareResult()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (mOnSuccessCb.get() && nsDOMCameraManager::IsWindowStillActive(mWindowId)) {
|
|
mOnSuccessCb->HandleEvent();
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
protected:
|
|
nsMainThreadPtrHandle<nsICameraReleaseCallback> mOnSuccessCb;
|
|
uint64_t mWindowId;
|
|
};
|
|
|
|
// Release the camera hardware.
|
|
class ReleaseHardwareTask : public nsRunnable
|
|
{
|
|
public:
|
|
ReleaseHardwareTask(CameraControlImpl* aCameraControl, nsICameraReleaseCallback* onSuccess, nsICameraErrorCallback* onError)
|
|
: mCameraControl(aCameraControl)
|
|
, mOnSuccessCb(new nsMainThreadPtrHolder<nsICameraReleaseCallback>(onSuccess))
|
|
, mOnErrorCb(new nsMainThreadPtrHolder<nsICameraErrorCallback>(onError))
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
virtual ~ReleaseHardwareTask()
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d : this=%p\n", __func__, __LINE__, this);
|
|
}
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
nsresult rv = mCameraControl->ReleaseHardwareImpl(this);
|
|
DOM_CAMERA_LOGT("%s:%d\n", __func__, __LINE__);
|
|
|
|
if (NS_FAILED(rv)) {
|
|
nsCOMPtr<nsIRunnable> cameraErrorResult = new CameraErrorResult(mOnErrorCb, NS_LITERAL_STRING("FAILURE"), mCameraControl->GetWindowId());
|
|
rv = NS_DispatchToMainThread(cameraErrorResult);
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
return rv;
|
|
}
|
|
|
|
nsRefPtr<CameraControlImpl> mCameraControl;
|
|
nsMainThreadPtrHandle<nsICameraReleaseCallback> mOnSuccessCb;
|
|
nsMainThreadPtrHandle<nsICameraErrorCallback> mOnErrorCb;
|
|
};
|
|
|
|
// Report that the video recorder state has changed.
|
|
class CameraRecorderStateChange : public nsRunnable
|
|
{
|
|
public:
|
|
CameraRecorderStateChange(nsMainThreadPtrHandle<nsICameraRecorderStateChange> onStateChange, const nsString& aStateMsg, int32_t aStatus, int32_t aTrackNumber, uint64_t aWindowId)
|
|
: mOnStateChangeCb(onStateChange)
|
|
, mStateMsg(aStateMsg)
|
|
, mStatus(aStatus)
|
|
, mTrackNumber(aTrackNumber)
|
|
, mWindowId(aWindowId)
|
|
{ }
|
|
|
|
NS_IMETHOD Run() MOZ_OVERRIDE
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (mOnStateChangeCb.get() && nsDOMCameraManager::IsWindowStillActive(mWindowId)) {
|
|
// For now, just pass the state message and swallow mStatus and mTrackNumber
|
|
mOnStateChangeCb->HandleStateChange(mStateMsg);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
protected:
|
|
nsMainThreadPtrHandle<nsICameraRecorderStateChange> mOnStateChangeCb;
|
|
const nsString mStateMsg;
|
|
int32_t mStatus;
|
|
int32_t mTrackNumber;
|
|
uint64_t mWindowId;
|
|
};
|
|
|
|
// Report that the preview stream state has changed.
|
|
class CameraPreviewStateChange : public nsRunnable
|
|
{
|
|
public:
|
|
CameraPreviewStateChange(nsMainThreadPtrHandle<nsICameraPreviewStateChange> onStateChange, const nsString& aStateMsg, uint64_t aWindowId)
|
|
: mOnStateChangeCb(onStateChange)
|
|
, mStateMsg(aStateMsg)
|
|
, mWindowId(aWindowId)
|
|
{ }
|
|
|
|
NS_IMETHOD Run()
|
|
{
|
|
MOZ_ASSERT(NS_IsMainThread());
|
|
|
|
if (mOnStateChangeCb.get() && nsDOMCameraManager::IsWindowStillActive(mWindowId)) {
|
|
mOnStateChangeCb->HandleStateChange(mStateMsg);
|
|
}
|
|
return NS_OK;
|
|
}
|
|
|
|
protected:
|
|
nsMainThreadPtrHandle<nsICameraPreviewStateChange> mOnStateChangeCb;
|
|
const nsString mStateMsg;
|
|
uint64_t mWindowId;
|
|
};
|
|
|
|
} // namespace mozilla
|
|
|
|
#endif // DOM_CAMERA_CAMERACONTROLIMPL_H
|