662 lines
30 KiB
JavaScript
662 lines
30 KiB
JavaScript
/*
|
|
*
|
|
* Licensed to the Apache Software Foundation (ASF) under one
|
|
* or more contributor license agreements. See the NOTICE file
|
|
* distributed with this work for additional information
|
|
* regarding copyright ownership. The ASF licenses this file
|
|
* to you under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance
|
|
* with the License. You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing,
|
|
* software distributed under the License is distributed on an
|
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
* KIND, either express or implied. See the License for the
|
|
* specific language governing permissions and limitations
|
|
* under the License.
|
|
*
|
|
*/
|
|
|
|
/*jshint unused:true, undef:true, browser:true */
|
|
/*global Windows:true, URL:true, module:true, require:true */
|
|
|
|
|
|
var Camera = require('./Camera');
|
|
|
|
module.exports = {
|
|
|
|
// args will contain :
|
|
// ... it is an array, so be careful
|
|
// 0 quality:50,
|
|
// 1 destinationType:Camera.DestinationType.FILE_URI,
|
|
// 2 sourceType:Camera.PictureSourceType.CAMERA,
|
|
// 3 targetWidth:-1,
|
|
// 4 targetHeight:-1,
|
|
// 5 encodingType:Camera.EncodingType.JPEG,
|
|
// 6 mediaType:Camera.MediaType.PICTURE,
|
|
// 7 allowEdit:false,
|
|
// 8 correctOrientation:false,
|
|
// 9 saveToPhotoAlbum:false,
|
|
// 10 popoverOptions:null
|
|
// 11 cameraDirection:0
|
|
|
|
takePicture: function (successCallback, errorCallback, args) {
|
|
var sourceType = args[2];
|
|
|
|
if (sourceType != Camera.PictureSourceType.CAMERA) {
|
|
takePictureFromFile(successCallback, errorCallback, args);
|
|
} else {
|
|
takePictureFromCamera(successCallback, errorCallback, args);
|
|
}
|
|
}
|
|
};
|
|
|
|
// https://msdn.microsoft.com/en-us/library/windows/apps/ff462087(v=vs.105).aspx
|
|
var windowsVideoContainers = [".avi", ".flv", ".asx", ".asf", ".mov", ".mp4", ".mpg", ".rm", ".srt", ".swf", ".wmv", ".vob"];
|
|
var windowsPhoneVideoContainers = [".avi", ".3gp", ".3g2", ".wmv", ".3gp", ".3g2", ".mp4", ".m4v"];
|
|
|
|
// Default aspect ratio 1.78 (16:9 hd video standard)
|
|
var DEFAULT_ASPECT_RATIO = '1.8';
|
|
|
|
// Resize method
|
|
function resizeImage(successCallback, errorCallback, file, targetWidth, targetHeight, encodingType) {
|
|
var tempPhotoFileName = "";
|
|
if (encodingType == Camera.EncodingType.PNG) {
|
|
tempPhotoFileName = "camera_cordova_temp_return.png";
|
|
} else {
|
|
tempPhotoFileName = "camera_cordova_temp_return.jpg";
|
|
}
|
|
|
|
var storageFolder = Windows.Storage.ApplicationData.current.localFolder;
|
|
file.copyAsync(storageFolder, file.name, Windows.Storage.NameCollisionOption.replaceExisting)
|
|
.then(function (storageFile) { return Windows.Storage.FileIO.readBufferAsync(storageFile); })
|
|
.then(function(buffer) {
|
|
var strBase64 = Windows.Security.Cryptography.CryptographicBuffer.encodeToBase64String(buffer);
|
|
var imageData = "data:" + file.contentType + ";base64," + strBase64;
|
|
var image = new Image();
|
|
image.src = imageData;
|
|
image.onload = function() {
|
|
var imageWidth = targetWidth,
|
|
imageHeight = targetHeight;
|
|
var canvas = document.createElement('canvas');
|
|
var storageFileName;
|
|
|
|
canvas.width = imageWidth;
|
|
canvas.height = imageHeight;
|
|
|
|
canvas.getContext("2d").drawImage(this, 0, 0, imageWidth, imageHeight);
|
|
|
|
var fileContent = canvas.toDataURL(file.contentType).split(',')[1];
|
|
|
|
var storageFolder = Windows.Storage.ApplicationData.current.localFolder;
|
|
|
|
storageFolder.createFileAsync(tempPhotoFileName, Windows.Storage.CreationCollisionOption.generateUniqueName)
|
|
.then(function (storagefile) {
|
|
var content = Windows.Security.Cryptography.CryptographicBuffer.decodeFromBase64String(fileContent);
|
|
storageFileName = storagefile.name;
|
|
return Windows.Storage.FileIO.writeBufferAsync(storagefile, content);
|
|
})
|
|
.done(function () {
|
|
successCallback("ms-appdata:///local/" + storageFileName);
|
|
}, errorCallback);
|
|
};
|
|
})
|
|
.done(null, function(err) {
|
|
errorCallback(err);
|
|
}
|
|
);
|
|
}
|
|
|
|
// Because of asynchronous method, so let the successCallback be called in it.
|
|
function resizeImageBase64(successCallback, errorCallback, file, targetWidth, targetHeight) {
|
|
Windows.Storage.FileIO.readBufferAsync(file).done( function(buffer) {
|
|
var strBase64 = Windows.Security.Cryptography.CryptographicBuffer.encodeToBase64String(buffer);
|
|
var imageData = "data:" + file.contentType + ";base64," + strBase64;
|
|
|
|
var image = new Image();
|
|
image.src = imageData;
|
|
|
|
image.onload = function() {
|
|
var imageWidth = targetWidth,
|
|
imageHeight = targetHeight;
|
|
var canvas = document.createElement('canvas');
|
|
|
|
canvas.width = imageWidth;
|
|
canvas.height = imageHeight;
|
|
|
|
var ctx = canvas.getContext("2d");
|
|
ctx.drawImage(this, 0, 0, imageWidth, imageHeight);
|
|
|
|
// The resized file ready for upload
|
|
var finalFile = canvas.toDataURL(file.contentType);
|
|
|
|
// Remove the prefix such as "data:" + contentType + ";base64," , in order to meet the Cordova API.
|
|
var arr = finalFile.split(",");
|
|
var newStr = finalFile.substr(arr[0].length + 1);
|
|
successCallback(newStr);
|
|
};
|
|
}, function(err) { errorCallback(err); });
|
|
}
|
|
|
|
function takePictureFromFile(successCallback, errorCallback, args) {
|
|
// Detect Windows Phone
|
|
if (navigator.appVersion.indexOf('Windows Phone 8.1') >= 0) {
|
|
takePictureFromFileWP(successCallback, errorCallback, args);
|
|
} else {
|
|
takePictureFromFileWindows(successCallback, errorCallback, args);
|
|
}
|
|
}
|
|
|
|
function takePictureFromFileWP(successCallback, errorCallback, args) {
|
|
var mediaType = args[6],
|
|
destinationType = args[1],
|
|
targetWidth = args[3],
|
|
targetHeight = args[4],
|
|
encodingType = args[5];
|
|
|
|
/*
|
|
Need to add and remove an event listener to catch activation state
|
|
Using FileOpenPicker will suspend the app and it's required to catch the PickSingleFileAndContinue
|
|
https://msdn.microsoft.com/en-us/library/windows/apps/xaml/dn631755.aspx
|
|
*/
|
|
var filePickerActivationHandler = function(eventArgs) {
|
|
if (eventArgs.kind === Windows.ApplicationModel.Activation.ActivationKind.pickFileContinuation) {
|
|
var file = eventArgs.files[0];
|
|
if (!file) {
|
|
errorCallback("User didn't choose a file.");
|
|
Windows.UI.WebUI.WebUIApplication.removeEventListener("activated", filePickerActivationHandler);
|
|
return;
|
|
}
|
|
if (destinationType == Camera.DestinationType.FILE_URI || destinationType == Camera.DestinationType.NATIVE_URI) {
|
|
if (targetHeight > 0 && targetWidth > 0) {
|
|
resizeImage(successCallback, errorCallback, file, targetWidth, targetHeight, encodingType);
|
|
}
|
|
else {
|
|
var storageFolder = Windows.Storage.ApplicationData.current.localFolder;
|
|
file.copyAsync(storageFolder, file.name, Windows.Storage.NameCollisionOption.replaceExisting).done(function (storageFile) {
|
|
successCallback(URL.createObjectURL(storageFile));
|
|
}, function () {
|
|
errorCallback("Can't access localStorage folder.");
|
|
});
|
|
}
|
|
}
|
|
else {
|
|
if (targetHeight > 0 && targetWidth > 0) {
|
|
resizeImageBase64(successCallback, errorCallback, file, targetWidth, targetHeight);
|
|
} else {
|
|
Windows.Storage.FileIO.readBufferAsync(file).done(function (buffer) {
|
|
var strBase64 = Windows.Security.Cryptography.CryptographicBuffer.encodeToBase64String(buffer);
|
|
successCallback(strBase64);
|
|
}, errorCallback);
|
|
}
|
|
}
|
|
Windows.UI.WebUI.WebUIApplication.removeEventListener("activated", filePickerActivationHandler);
|
|
}
|
|
};
|
|
|
|
var fileOpenPicker = new Windows.Storage.Pickers.FileOpenPicker();
|
|
if (mediaType == Camera.MediaType.PICTURE) {
|
|
fileOpenPicker.fileTypeFilter.replaceAll([".png", ".jpg", ".jpeg"]);
|
|
fileOpenPicker.suggestedStartLocation = Windows.Storage.Pickers.PickerLocationId.picturesLibrary;
|
|
}
|
|
else if (mediaType == Camera.MediaType.VIDEO) {
|
|
fileOpenPicker.fileTypeFilter.replaceAll(windowsPhoneVideoContainers);
|
|
fileOpenPicker.suggestedStartLocation = Windows.Storage.Pickers.PickerLocationId.videosLibrary;
|
|
}
|
|
else {
|
|
fileOpenPicker.fileTypeFilter.replaceAll(["*"]);
|
|
fileOpenPicker.suggestedStartLocation = Windows.Storage.Pickers.PickerLocationId.documentsLibrary;
|
|
}
|
|
|
|
Windows.UI.WebUI.WebUIApplication.addEventListener("activated", filePickerActivationHandler);
|
|
fileOpenPicker.pickSingleFileAndContinue();
|
|
}
|
|
|
|
function takePictureFromFileWindows(successCallback, errorCallback, args) {
|
|
var mediaType = args[6],
|
|
destinationType = args[1],
|
|
targetWidth = args[3],
|
|
targetHeight = args[4],
|
|
encodingType = args[5];
|
|
|
|
var fileOpenPicker = new Windows.Storage.Pickers.FileOpenPicker();
|
|
if (mediaType == Camera.MediaType.PICTURE) {
|
|
fileOpenPicker.fileTypeFilter.replaceAll([".png", ".jpg", ".jpeg"]);
|
|
fileOpenPicker.suggestedStartLocation = Windows.Storage.Pickers.PickerLocationId.picturesLibrary;
|
|
}
|
|
else if (mediaType == Camera.MediaType.VIDEO) {
|
|
fileOpenPicker.fileTypeFilter.replaceAll(windowsVideoContainers);
|
|
fileOpenPicker.suggestedStartLocation = Windows.Storage.Pickers.PickerLocationId.videosLibrary;
|
|
}
|
|
else {
|
|
fileOpenPicker.fileTypeFilter.replaceAll(["*"]);
|
|
fileOpenPicker.suggestedStartLocation = Windows.Storage.Pickers.PickerLocationId.documentsLibrary;
|
|
}
|
|
|
|
fileOpenPicker.pickSingleFileAsync().done(function (file) {
|
|
if (!file) {
|
|
errorCallback("User didn't choose a file.");
|
|
return;
|
|
}
|
|
if (destinationType == Camera.DestinationType.FILE_URI || destinationType == Camera.DestinationType.NATIVE_URI) {
|
|
if (targetHeight > 0 && targetWidth > 0) {
|
|
resizeImage(successCallback, errorCallback, file, targetWidth, targetHeight, encodingType);
|
|
}
|
|
else {
|
|
var storageFolder = Windows.Storage.ApplicationData.current.localFolder;
|
|
file.copyAsync(storageFolder, file.name, Windows.Storage.NameCollisionOption.replaceExisting).done(function (storageFile) {
|
|
successCallback(URL.createObjectURL(storageFile));
|
|
}, function () {
|
|
errorCallback("Can't access localStorage folder.");
|
|
});
|
|
}
|
|
}
|
|
else {
|
|
if (targetHeight > 0 && targetWidth > 0) {
|
|
resizeImageBase64(successCallback, errorCallback, file, targetWidth, targetHeight);
|
|
} else {
|
|
Windows.Storage.FileIO.readBufferAsync(file).done(function (buffer) {
|
|
var strBase64 = Windows.Security.Cryptography.CryptographicBuffer.encodeToBase64String(buffer);
|
|
successCallback(strBase64);
|
|
}, errorCallback);
|
|
}
|
|
}
|
|
}, function () {
|
|
errorCallback("User didn't choose a file.");
|
|
});
|
|
}
|
|
|
|
function takePictureFromCamera(successCallback, errorCallback, args) {
|
|
// Check if necessary API available
|
|
if (!Windows.Media.Capture.CameraCaptureUI) {
|
|
takePictureFromCameraWP(successCallback, errorCallback, args);
|
|
} else {
|
|
takePictureFromCameraWindows(successCallback, errorCallback, args);
|
|
}
|
|
}
|
|
|
|
function takePictureFromCameraWP(successCallback, errorCallback, args) {
|
|
// We are running on WP8.1 which lacks CameraCaptureUI class
|
|
// so we need to use MediaCapture class instead and implement custom UI for camera
|
|
var destinationType = args[1],
|
|
targetWidth = args[3],
|
|
targetHeight = args[4],
|
|
encodingType = args[5],
|
|
saveToPhotoAlbum = args[9],
|
|
cameraDirection = args[11],
|
|
capturePreview = null,
|
|
captureCancelButton = null,
|
|
capture = null,
|
|
captureSettings = null,
|
|
CaptureNS = Windows.Media.Capture;
|
|
|
|
function cameraPreviewOrientation() {
|
|
// Rotate the cam since WP8.1 MediaCapture outputs video stream rotated 90° CCW
|
|
if (screen.msOrientation === "portrait-primary" || screen.msOrientation === "portrait-secondary") {
|
|
capture.setPreviewRotation(Windows.Media.Capture.VideoRotation.clockwise90Degrees);
|
|
} else if (screen.msOrientation === "landscape-secondary") {
|
|
capture.setPreviewRotation(Windows.Media.Capture.VideoRotation.clockwise180Degrees);
|
|
} else {
|
|
capture.setPreviewRotation(Windows.Media.Capture.VideoRotation.none);
|
|
}
|
|
}
|
|
|
|
var createCameraUI = function () {
|
|
// Create fullscreen preview
|
|
capturePreview = document.createElement("video");
|
|
|
|
// z-order style element for capturePreview and captureCancelButton elts
|
|
// is necessary to avoid overriding by another page elements, -1 sometimes is not enough
|
|
capturePreview.style.cssText = "position: fixed; left: 0; top: 0; width: 100%; height: 100%; z-index: 999";
|
|
|
|
// Create cancel button
|
|
captureCancelButton = document.createElement("button");
|
|
captureCancelButton.innerText = "Cancel";
|
|
captureCancelButton.style.cssText = "position: fixed; right: 0; bottom: 0; display: block; margin: 20px; z-index: 1000";
|
|
|
|
capture = new CaptureNS.MediaCapture();
|
|
|
|
captureSettings = new CaptureNS.MediaCaptureInitializationSettings();
|
|
captureSettings.streamingCaptureMode = CaptureNS.StreamingCaptureMode.video;
|
|
};
|
|
|
|
var startCameraPreview = function () {
|
|
// Search for available camera devices
|
|
// This is necessary to detect which camera (front or back) we should use
|
|
var expectedPanel = cameraDirection === 1 ? Windows.Devices.Enumeration.Panel.front : Windows.Devices.Enumeration.Panel.back;
|
|
Windows.Devices.Enumeration.DeviceInformation.findAllAsync(Windows.Devices.Enumeration.DeviceClass.videoCapture).then(function (devices) {
|
|
if (devices.length <= 0) {
|
|
destroyCameraPreview();
|
|
errorCallback('Camera not found');
|
|
return;
|
|
}
|
|
|
|
devices.forEach(function(currDev) {
|
|
if (currDev.enclosureLocation.panel && currDev.enclosureLocation.panel == expectedPanel) {
|
|
captureSettings.videoDeviceId = currDev.id;
|
|
}
|
|
});
|
|
|
|
captureSettings.photoCaptureSource = Windows.Media.Capture.PhotoCaptureSource.photo;
|
|
|
|
return capture.initializeAsync(captureSettings);
|
|
}).then(function () {
|
|
// msdn.microsoft.com/en-us/library/windows/apps/hh452807.aspx
|
|
capturePreview.msZoom = true;
|
|
capturePreview.src = URL.createObjectURL(capture);
|
|
capturePreview.play();
|
|
|
|
// Insert preview frame and controls into page
|
|
document.body.appendChild(capturePreview);
|
|
document.body.appendChild(captureCancelButton);
|
|
|
|
// Bind events to controls
|
|
window.addEventListener('deviceorientation', cameraPreviewOrientation, false);
|
|
capturePreview.addEventListener('click', captureAction);
|
|
captureCancelButton.addEventListener('click', function () {
|
|
destroyCameraPreview();
|
|
errorCallback('Cancelled');
|
|
}, false);
|
|
|
|
// Get available aspect ratios
|
|
var aspectRatios = getAspectRatios(capture);
|
|
|
|
// Couldn't find a good ratio
|
|
if (aspectRatios.length === 0) {
|
|
destroyCameraPreview();
|
|
errorCallback('There\'s not a good aspect ratio available');
|
|
return;
|
|
}
|
|
|
|
if (aspectRatios.indexOf(DEFAULT_ASPECT_RATIO) > -1) {
|
|
return setAspectRatio(capture, DEFAULT_ASPECT_RATIO);
|
|
} else {
|
|
// Doesn't support 16:9 - pick next best
|
|
return setAspectRatio(capture, aspectRatios[0]);
|
|
}
|
|
}).done(null, function (err) {
|
|
destroyCameraPreview();
|
|
errorCallback('Camera intitialization error ' + err);
|
|
});
|
|
};
|
|
|
|
var destroyCameraPreview = function () {
|
|
window.removeEventListener('deviceorientation', cameraPreviewOrientation, false);
|
|
capturePreview.pause();
|
|
capturePreview.src = null;
|
|
[capturePreview, captureCancelButton].forEach(function(elem) {
|
|
if (elem /* && elem in document.body.childNodes */) {
|
|
document.body.removeChild(elem);
|
|
}
|
|
});
|
|
if (capture) {
|
|
capture.stopRecordAsync();
|
|
capture = null;
|
|
}
|
|
};
|
|
|
|
var captureAction = function () {
|
|
|
|
var encodingProperties,
|
|
fileName,
|
|
generateUniqueCollisionOption = Windows.Storage.CreationCollisionOption.generateUniqueName,
|
|
tempFolder = Windows.Storage.ApplicationData.current.temporaryFolder;
|
|
|
|
if (encodingType == Camera.EncodingType.PNG) {
|
|
fileName = 'photo.png';
|
|
encodingProperties = Windows.Media.MediaProperties.ImageEncodingProperties.createPng();
|
|
} else {
|
|
fileName = 'photo.jpg';
|
|
encodingProperties = Windows.Media.MediaProperties.ImageEncodingProperties.createJpeg();
|
|
}
|
|
|
|
tempFolder.createFileAsync(fileName, generateUniqueCollisionOption)
|
|
.then(function(tempCapturedFile) {
|
|
return new WinJS.Promise(function (complete) {
|
|
var imgStream = new Windows.Storage.Streams.InMemoryRandomAccessStream();
|
|
capture.capturePhotoToStreamAsync(encodingProperties, imgStream)
|
|
.then(function() {
|
|
return Windows.Graphics.Imaging.BitmapDecoder.createAsync(imgStream);
|
|
})
|
|
.then(function(dec) {
|
|
return Windows.Graphics.Imaging.BitmapEncoder.createForTranscodingAsync(imgStream, dec);
|
|
})
|
|
.then(function(enc) {
|
|
// We need to rotate the photo 90° CW because by default wp8.1 takes 90° CCW rotated photos.
|
|
enc.bitmapTransform.rotation = Windows.Graphics.Imaging.BitmapRotation.clockwise90Degrees;
|
|
return enc.flushAsync();
|
|
})
|
|
.then(function() {
|
|
return tempCapturedFile.openAsync(Windows.Storage.FileAccessMode.readWrite);
|
|
})
|
|
.then(function(fileStream) {
|
|
imgStream.seek(0); // required for win8.1 emulator
|
|
return Windows.Storage.Streams.RandomAccessStream.copyAsync(imgStream, fileStream);
|
|
})
|
|
.done(function() {
|
|
imgStream.close();
|
|
complete(tempCapturedFile);
|
|
}, function() {
|
|
imgStream.close();
|
|
throw new Error("An error has occured while capturing the photo.");
|
|
});
|
|
});
|
|
})
|
|
.done(function(capturedFile) {
|
|
destroyCameraPreview();
|
|
savePhoto(capturedFile, {
|
|
destinationType: destinationType,
|
|
targetHeight: targetHeight,
|
|
targetWidth: targetWidth,
|
|
encodingType: encodingType,
|
|
saveToPhotoAlbum: saveToPhotoAlbum
|
|
}, successCallback, errorCallback);
|
|
}, function(err) {
|
|
destroyCameraPreview();
|
|
errorCallback(err);
|
|
});
|
|
};
|
|
|
|
var getAspectRatios = function (capture) {
|
|
var photoAspectRatios = capture.videoDeviceController.getAvailableMediaStreamProperties(Windows.Media.Capture.MediaStreamType.photo).map(function (element) {
|
|
return (element.width / element.height).toFixed(1);
|
|
}).filter(function (element, index, array) { return (index === array.indexOf(element)); });
|
|
|
|
var videoAspectRatios = capture.videoDeviceController.getAvailableMediaStreamProperties(Windows.Media.Capture.MediaStreamType.videoRecord).map(function (element) {
|
|
return (element.width / element.height).toFixed(1);
|
|
}).filter(function (element, index, array) { return (index === array.indexOf(element)); });
|
|
|
|
var videoPreviewAspectRatios = capture.videoDeviceController.getAvailableMediaStreamProperties(Windows.Media.Capture.MediaStreamType.videoPreview).map(function (element) {
|
|
return (element.width / element.height).toFixed(1);
|
|
}).filter(function (element, index, array) { return (index === array.indexOf(element)); });
|
|
|
|
var allAspectRatios = [].concat(photoAspectRatios, videoAspectRatios, videoPreviewAspectRatios);
|
|
|
|
var aspectObj = allAspectRatios.reduce(function (map, item) {
|
|
if (!map[item]) {
|
|
map[item] = 0;
|
|
}
|
|
map[item]++;
|
|
return map;
|
|
}, {});
|
|
|
|
return Object.keys(aspectObj).filter(function (k) {
|
|
return aspectObj[k] === 3;
|
|
});
|
|
};
|
|
|
|
var setAspectRatio = function (capture, aspect) {
|
|
// Max photo resolution with desired aspect ratio
|
|
var photoResolution = capture.videoDeviceController.getAvailableMediaStreamProperties(Windows.Media.Capture.MediaStreamType.photo).filter(function (elem) {
|
|
return ((elem.width / elem.height).toFixed(1) === aspect);
|
|
}).reduce(function (prop1, prop2) {
|
|
return (prop1.width * prop1.height) > (prop2.width * prop2.height) ? prop1 : prop2;
|
|
});
|
|
|
|
// Max video resolution with desired aspect ratio
|
|
var videoRecordResolution = capture.videoDeviceController.getAvailableMediaStreamProperties(Windows.Media.Capture.MediaStreamType.videoRecord).filter(function (elem) {
|
|
return ((elem.width / elem.height).toFixed(1) === aspect);
|
|
}).reduce(function (prop1, prop2) {
|
|
return (prop1.width * prop1.height) > (prop2.width * prop2.height) ? prop1 : prop2;
|
|
});
|
|
|
|
// Max video preview resolution with desired aspect ratio
|
|
var videoPreviewResolution = capture.videoDeviceController.getAvailableMediaStreamProperties(Windows.Media.Capture.MediaStreamType.videoPreview).filter(function (elem) {
|
|
return ((elem.width / elem.height).toFixed(1) === aspect);
|
|
}).reduce(function (prop1, prop2) {
|
|
return (prop1.width * prop1.height) > (prop2.width * prop2.height) ? prop1 : prop2;
|
|
});
|
|
|
|
return capture.videoDeviceController.setMediaStreamPropertiesAsync(Windows.Media.Capture.MediaStreamType.photo, photoResolution)
|
|
.then(function () { return capture.videoDeviceController.setMediaStreamPropertiesAsync(Windows.Media.Capture.MediaStreamType.videoPreview, videoPreviewResolution); })
|
|
.then(function () { return capture.videoDeviceController.setMediaStreamPropertiesAsync(Windows.Media.Capture.MediaStreamType.videoRecord, videoRecordResolution); });
|
|
};
|
|
|
|
try {
|
|
createCameraUI();
|
|
startCameraPreview();
|
|
} catch (ex) {
|
|
errorCallback(ex);
|
|
}
|
|
}
|
|
|
|
function takePictureFromCameraWindows(successCallback, errorCallback, args) {
|
|
var destinationType = args[1],
|
|
targetWidth = args[3],
|
|
targetHeight = args[4],
|
|
encodingType = args[5],
|
|
allowCrop = !!args[7],
|
|
saveToPhotoAlbum = args[9],
|
|
cameraCaptureUI = new Windows.Media.Capture.CameraCaptureUI();
|
|
cameraCaptureUI.photoSettings.allowCropping = allowCrop;
|
|
|
|
if (encodingType == Camera.EncodingType.PNG) {
|
|
cameraCaptureUI.photoSettings.format = Windows.Media.Capture.CameraCaptureUIPhotoFormat.png;
|
|
} else {
|
|
cameraCaptureUI.photoSettings.format = Windows.Media.Capture.CameraCaptureUIPhotoFormat.jpeg;
|
|
}
|
|
|
|
// decide which max pixels should be supported by targetWidth or targetHeight.
|
|
if (targetWidth >= 1280 || targetHeight >= 960) {
|
|
cameraCaptureUI.photoSettings.maxResolution = Windows.Media.Capture.CameraCaptureUIMaxPhotoResolution.large3M;
|
|
} else if (targetWidth >= 1024 || targetHeight >= 768) {
|
|
cameraCaptureUI.photoSettings.maxResolution = Windows.Media.Capture.CameraCaptureUIMaxPhotoResolution.mediumXga;
|
|
} else if (targetWidth >= 800 || targetHeight >= 600) {
|
|
cameraCaptureUI.photoSettings.maxResolution = Windows.Media.Capture.CameraCaptureUIMaxPhotoResolution.mediumXga;
|
|
} else if (targetWidth >= 640 || targetHeight >= 480) {
|
|
cameraCaptureUI.photoSettings.maxResolution = Windows.Media.Capture.CameraCaptureUIMaxPhotoResolution.smallVga;
|
|
} else if (targetWidth >= 320 || targetHeight >= 240) {
|
|
cameraCaptureUI.photoSettings.maxResolution = Windows.Media.Capture.CameraCaptureUIMaxPhotoResolution.verySmallQvga;
|
|
} else {
|
|
cameraCaptureUI.photoSettings.maxResolution = Windows.Media.Capture.CameraCaptureUIMaxPhotoResolution.highestAvailable;
|
|
}
|
|
|
|
cameraCaptureUI.captureFileAsync(Windows.Media.Capture.CameraCaptureUIMode.photo).done(function(picture) {
|
|
if (!picture) {
|
|
errorCallback("User didn't capture a photo.");
|
|
return;
|
|
}
|
|
|
|
savePhoto(picture, {
|
|
destinationType: destinationType,
|
|
targetHeight: targetHeight,
|
|
targetWidth: targetWidth,
|
|
encodingType: encodingType,
|
|
saveToPhotoAlbum: saveToPhotoAlbum
|
|
}, successCallback, errorCallback);
|
|
}, function() {
|
|
errorCallback("Fail to capture a photo.");
|
|
});
|
|
}
|
|
|
|
function savePhoto(picture, options, successCallback, errorCallback) {
|
|
// success callback for capture operation
|
|
var success = function(picture) {
|
|
var generateUniqueCollisionOption = Windows.Storage.CreationCollisionOption.generateUniqueName;
|
|
if (options.destinationType == Camera.DestinationType.FILE_URI || options.destinationType == Camera.DestinationType.NATIVE_URI) {
|
|
if (options.targetHeight > 0 && options.targetWidth > 0) {
|
|
resizeImage(successCallback, errorCallback, picture, options.targetWidth, options.targetHeight, options.encodingType);
|
|
} else {
|
|
picture.copyAsync(Windows.Storage.ApplicationData.current.localFolder, picture.name, generateUniqueCollisionOption).done(function(copiedFile) {
|
|
successCallback("ms-appdata:///local/" + copiedFile.name);
|
|
},errorCallback);
|
|
}
|
|
} else {
|
|
if (options.targetHeight > 0 && options.targetWidth > 0) {
|
|
resizeImageBase64(successCallback, errorCallback, picture, options.targetWidth, options.targetHeight);
|
|
} else {
|
|
Windows.Storage.FileIO.readBufferAsync(picture).done(function(buffer) {
|
|
var strBase64 = Windows.Security.Cryptography.CryptographicBuffer.encodeToBase64String(buffer);
|
|
picture.deleteAsync().done(function() {
|
|
successCallback(strBase64);
|
|
}, function(err) {
|
|
errorCallback(err);
|
|
});
|
|
}, errorCallback);
|
|
}
|
|
}
|
|
};
|
|
|
|
if (!options.saveToPhotoAlbum) {
|
|
success(picture);
|
|
return;
|
|
} else {
|
|
var savePicker = new Windows.Storage.Pickers.FileSavePicker();
|
|
var saveFile = function(file) {
|
|
if (file) {
|
|
// Prevent updates to the remote version of the file until we're done
|
|
Windows.Storage.CachedFileManager.deferUpdates(file);
|
|
picture.moveAndReplaceAsync(file)
|
|
.then(function() {
|
|
// Let Windows know that we're finished changing the file so
|
|
// the other app can update the remote version of the file.
|
|
return Windows.Storage.CachedFileManager.completeUpdatesAsync(file);
|
|
})
|
|
.done(function(updateStatus) {
|
|
if (updateStatus === Windows.Storage.Provider.FileUpdateStatus.complete) {
|
|
success(picture);
|
|
} else {
|
|
errorCallback("File update status is not complete.");
|
|
}
|
|
}, errorCallback);
|
|
} else {
|
|
errorCallback("Failed to select a file.");
|
|
}
|
|
};
|
|
savePicker.suggestedStartLocation = Windows.Storage.Pickers.PickerLocationId.picturesLibrary;
|
|
|
|
if (options.encodingType === Camera.EncodingType.PNG) {
|
|
savePicker.fileTypeChoices.insert("PNG", [".png"]);
|
|
savePicker.suggestedFileName = "photo.png";
|
|
} else {
|
|
savePicker.fileTypeChoices.insert("JPEG", [".jpg"]);
|
|
savePicker.suggestedFileName = "photo.jpg";
|
|
}
|
|
|
|
// If Windows Phone 8.1 use pickSaveFileAndContinue()
|
|
if (navigator.appVersion.indexOf('Windows Phone 8.1') >= 0) {
|
|
/*
|
|
Need to add and remove an event listener to catch activation state
|
|
Using FileSavePicker will suspend the app and it's required to catch the pickSaveFileContinuation
|
|
https://msdn.microsoft.com/en-us/library/windows/apps/xaml/dn631755.aspx
|
|
*/
|
|
var fileSaveHandler = function(eventArgs) {
|
|
if (eventArgs.kind === Windows.ApplicationModel.Activation.ActivationKind.pickSaveFileContinuation) {
|
|
var file = eventArgs.file;
|
|
saveFile(file);
|
|
Windows.UI.WebUI.WebUIApplication.removeEventListener("activated", fileSaveHandler);
|
|
}
|
|
};
|
|
Windows.UI.WebUI.WebUIApplication.addEventListener("activated", fileSaveHandler);
|
|
savePicker.pickSaveFileAndContinue();
|
|
} else {
|
|
savePicker.pickSaveFileAsync()
|
|
.done(saveFile, errorCallback);
|
|
}
|
|
}
|
|
}
|
|
|
|
require("cordova/exec/proxy").add("Camera",module.exports);
|