index.html 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301
  1. <!DOCTYPE html>
  2. <html>
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="apple-mobile-web-app-capable" content="yes" />
  6. <meta name="viewport" content="user-scalable=no, width=device-width, initial-scale=1, maximum-scale=1"/>
  7. <meta name="theme-color" content="#4b75ff">
  8. <link rel="stylesheet" href="../script/semantic/semantic.min.css">
  9. <script src="../script/jquery.min.js"></script>
  10. <script src="../script/semantic/semantic.min.js"></script>
  11. <script src="../script/ao_module.js"></script>
  12. <link rel="manifest" crossorigin="use-credentials" href="manifest.json">
  13. <title>Camera</title>
  14. <style>
  15. body{
  16. background-color: #000000;
  17. }
  18. .previewer{
  19. width: 100%;
  20. max-height: 100%;
  21. }
  22. #controls{
  23. position: fixed;
  24. left: 0px;
  25. bottom: 0px;
  26. width: 100%;
  27. padding: 12px;
  28. padding-bottom: 20px;
  29. }
  30. #verticalUI .shutterContainer{
  31. display: flex;
  32. justify-content: space-evenly;
  33. }
  34. #verticalUI .shutter.image{
  35. width: 5em;
  36. }
  37. #sidewayUI .shutterContainer{
  38. position: fixed;
  39. right: 0px;
  40. height: 100%;
  41. top:0px;
  42. width: 5em;
  43. }
  44. #sidewayUI .shutter.image{
  45. position: absolute;
  46. right: 20px;
  47. width: 5em;
  48. top: 50%;
  49. -ms-transform: translateY(-50%);
  50. transform: translateY(-50%);
  51. }
  52. #shutterCover{
  53. position: fixed;
  54. top:0px;
  55. left: 0px;
  56. width: 100%;
  57. height: 100%;
  58. background-color: black;
  59. display:none;
  60. }
  61. #albumn{
  62. position: fixed;
  63. bottom: 20px;
  64. right: 12px;
  65. width: 6em;
  66. }
  67. .ablumnpreview{
  68. width: 5em;
  69. }
  70. .latestPreview{
  71. width: 5em !important;
  72. height: 5em !important;
  73. }
  74. .zoombarcontainer{
  75. width: 100%;
  76. padding-bottom: 8px;
  77. padding-left: 20px;
  78. padding-right: 20px;
  79. }
  80. .sideway.zoombarcontainer{
  81. display: flex;
  82. justify-content: space-evenly;
  83. }
  84. #zoombar{
  85. width: 100%;
  86. }
  87. #zoombar.sideway{
  88. width: 50% !important;
  89. }
  90. </style>
  91. </head>
  92. <body>
  93. <video id="camera" class="previewer" autoplay></video>
  94. <div id="shutterCover"></div>
  95. <canvas id="canvas" style="display:none;"></canvas>
  96. <div id="albumn">
  97. <div class="ablumnpreview">
  98. <img class="ui fluid image latestPreview" src="img/module_icon.png">
  99. </div>
  100. </div>
  101. <div id="controls">
  102. <div class="zoombarcontainer">
  103. <input id="zoombar" type="range" min="1" max="100" value="50">
  104. </div>
  105. <div id="verticalUI">
  106. <div class="shutterContainer">
  107. <div onclick="takePicture();" ontouchdown="takePicture();">
  108. <img class="ui shutter image" src="img/shutter.png">
  109. </div>
  110. </div>
  111. </div>
  112. <div id="sidewayUI">
  113. <div class="shutterContainer">
  114. <div onclick="takePicture();" ontouchdown="takePicture();">
  115. <img class="ui shutter image" src="img/shutter.png">
  116. </div>
  117. </div>
  118. </div>
  119. </div>
  120. <script>
  121. let isMobile = false;
  122. let saveFolder = "user:/Photo/DCIM";
  123. if( /Android|webOS|iPhone|iPad|Mac|Macintosh|iPod|BlackBerry|IEMobile|Opera Mini/i.test(navigator.userAgent) ) {
  124. isMobile = true;
  125. }
  126. $(document).ready(function(){
  127. ao_module_agirun("Camera/backend/readydir.js", {savetarget: saveFolder}, function(){
  128. //Folder structure ready
  129. initcamera();
  130. });
  131. loadAblumnPreview();
  132. });
  133. $(window).on("resize", function(){
  134. handleWindowResize();
  135. });
  136. handleWindowResize();
  137. function handleWindowResize(){
  138. var width = window.innerWidth;
  139. var height = window.innerHeight;
  140. if (width > height){
  141. //Using the phone sideway
  142. $("#verticalUI").hide();
  143. $("#sidewayUI").show();
  144. $(".zoombarcontainer").addClass("sideway");
  145. $("#zoombar").addClass("sideway");
  146. }else{
  147. //Use the phone vertically
  148. $("#verticalUI").show();
  149. $("#sidewayUI").hide();
  150. $(".zoombarcontainer").removeClass("sideway");
  151. $("#zoombar").removeClass("sideway");
  152. }
  153. }
  154. function loadAblumnPreview(){
  155. ao_module_agirun("Camera/backend/loadLatestPhoto.js", {savetarget: saveFolder}, function(data){
  156. if (data.error !== undefined){
  157. console.log(data.error);
  158. }else{
  159. $(".latestPreview").attr("src", `../media/?file=${data}`);
  160. console.log(data);
  161. }
  162. });
  163. }
  164. function takePicture(){
  165. $("#shutterCover").show(0);
  166. setTimeout(function(){
  167. $("#shutterCover").hide();
  168. }, 500);
  169. var canvas = document.getElementById('canvas');
  170. var video = document.getElementById('camera');
  171. canvas.width = video.videoWidth;
  172. canvas.height = video.videoHeight;
  173. canvas.getContext('2d').drawImage(video, 0, 0, video.videoWidth, video.videoHeight);
  174. canvas.toBlob(function(blob){
  175. //const img = new Image();
  176. //img.src = URL.createObjectURL(blob);
  177. //window.open(img.src);
  178. //Generate a filename for this photo
  179. var d = new Date();
  180. var filename = "DSC_" + d.getFullYear() + d.getMonth() + d.getDate() + "_" + d.getHours() + d.getMinutes() + d.getSeconds() + ".jpg";
  181. //Upload it to server
  182. uploadImageToServer(filename, blob, function(){
  183. //Update the image preview
  184. loadAblumnPreview();
  185. });
  186. }, "image/jpeg");
  187. }
  188. //Upload the file to server
  189. function uploadFile(file, uploadPath, callback=undefined){
  190. let url = '../../system/file_system/upload'
  191. let formData = new FormData()
  192. let xhr = new XMLHttpRequest()
  193. formData.append('file', file);
  194. formData.append('path', uploadPath);
  195. xhr.open('POST', url, true)
  196. xhr.upload.addEventListener("progress", function(e) {
  197. var progress = (e.loaded * 100.0 / e.total) || 100;
  198. console.log(progress);
  199. });
  200. xhr.addEventListener('readystatechange', function(e) {
  201. if (xhr.readyState == 4 && xhr.status == 200) {
  202. //Uplaod process ok
  203. var resp = JSON.parse(e.target.response);
  204. if (callback != undefined){
  205. callback(resp);
  206. }
  207. }else{
  208. //Upload failed
  209. if (callback != undefined){
  210. callback({
  211. error: "File upload failed"
  212. });
  213. }
  214. }
  215. });
  216. xhr.send(formData);
  217. }
  218. function uploadImageToServer(filename, blob, callback){
  219. var imageFile = ao_module_utils.blobToFile(blob, filename, blob.type);
  220. uploadFile(imageFile, "user:/Photo/DCIM", callback);
  221. }
  222. function initcamera(){
  223. //Define the camera constraints
  224. const constraints = {
  225. video: {
  226. width: { ideal: 2048 },
  227. height: { ideal: 1080 },
  228. zoom: true,
  229. facingMode: 'environment'
  230. },
  231. audio: false
  232. };
  233. const video = document.querySelector("video");
  234. navigator.mediaDevices.getUserMedia(constraints).then((stream) => {
  235. const [track] = stream.getVideoTracks();
  236. try{
  237. const capabilities = track.getCapabilities();
  238. const settings = track.getSettings();
  239. console.log(track, settings);
  240. video.srcObject = stream;
  241. //Check if zoom exists. If yes, allow zoom
  242. if (!('zoom' in settings)) {
  243. $("#zoombar").hide();
  244. }else{
  245. var input = $("#zoombar")[0];
  246. input.min = capabilities.zoom.min;
  247. input.max = capabilities.zoom.max;
  248. input.step = capabilities.zoom.step;
  249. input.value = settings.zoom;
  250. input.oninput = function(event) {
  251. track.applyConstraints({
  252. advanced: [ {zoom: event.target.value} ]
  253. });
  254. }
  255. }
  256. }catch(ex){
  257. //This video input support nothing
  258. $("#zoombar").hide();
  259. video.srcObject = stream;
  260. }
  261. });
  262. }
  263. </script>
  264. </body>
  265. </html>