qrcode.html 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154
  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="UTF-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1.0">
  6. <title>QR Scanner and QR Code Generator</title>
  7. <script src="https://cdn.jsdelivr.net/npm/qrcodejs/qrcode.min.js"></script>
  8. <style>
  9. body { font-family: Arial, sans-serif; text-align: center; margin: 20px; }
  10. video { width: 400px; height: 400px; border: 1px solid black; display: block; margin: auto; object-fit: cover; max-width: 100%;}
  11. #qrcode { margin-top: 20px; }
  12. #qrcode img, #qrcode canvas { margin: 0 auto; width: 400px; height: 400px; max-width: 100%;}
  13. button { margin: 5px; padding: 10px; }
  14. </style>
  15. </head>
  16. <body>
  17. <h1>QR Scanner & QR Code</h1>
  18. <h2>QR Code Scanner</h2>
  19. <video id="video"></video>
  20. <button id="startCamera">Start Camera</button>
  21. <button id="stopCamera">Stop Camera</button>
  22. <button id="switchCamera">Switch Camera</button>
  23. <button id="toggleFlash">Toggle Flash</button>
  24. <p id="cam-status"></p>
  25. <h2>QR Code</h2>
  26. <div id="qrcode"></div>
  27. <p><a id="qrcode-status" target="_parent"></a></p>
  28. <button id="generateQRCode">Generate QR Code</button>
  29. <script type="module">
  30. const share_url = "{{share_url}}" || window.location.origin;
  31. const conversation_id = "{{conversation_id}}";
  32. if (!conversation_id) {
  33. document.getElementById('generateQRCode')
  34. .setAttribute('disabled', 'disabled');
  35. }
  36. import QrScanner from 'https://cdn.jsdelivr.net/npm/qr-scanner/qr-scanner.min.js';
  37. const videoElem = document.getElementById('video');
  38. const camStatus = document.getElementById('cam-status');
  39. let qrScanner;
  40. document.getElementById('stopCamera').addEventListener('click', () => {
  41. if (currentStream) {
  42. currentStream.getTracks().forEach(track => track.stop());
  43. }
  44. if (qrScanner) {
  45. qrScanner.stop();
  46. }
  47. });
  48. document.getElementById('toggleFlash').addEventListener('click', async () => {
  49. if (qrScanner) {
  50. const hasFlash = await qrScanner.hasFlash();
  51. if (hasFlash) {
  52. qrScanner.toggleFlash();
  53. } else {
  54. alert('Flash not supported on this camera.');
  55. }
  56. }
  57. });
  58. let share_id = null;
  59. document.getElementById('generateQRCode').addEventListener('click', async () => {
  60. if (share_id) {
  61. const delete_url = `${share_url}/backend-api/v2/files/${encodeURI(share_id)}`;
  62. await fetch(delete_url, {
  63. method: 'DELETE'
  64. });
  65. }
  66. share_id = crypto.randomUUID();
  67. const url = `${share_url}/backend-api/v2/chat/${encodeURI(share_id)}`;
  68. const response = await fetch(url, {
  69. method: 'POST',
  70. headers: {'content-type': 'application/json'},
  71. body: localStorage.getItem(`conversation:${conversation_id}`)
  72. });
  73. const share = `${share_url}/chat/${encodeURI(share_id)}/${encodeURI(conversation_id)}`;
  74. const qrcodeStatus = document.getElementById('qrcode-status');
  75. if (response.status !== 200) {
  76. qrcodeStatus.innerText = 'Error generating QR code: ' + response.statusText;
  77. return;
  78. }
  79. qrcodeStatus.innerText = share;
  80. qrcodeStatus.href = share;
  81. document.getElementById("qrcode").innerHTML = '';
  82. const qrcode = new QRCode(
  83. document.getElementById("qrcode"),
  84. share,
  85. {
  86. width: 400,
  87. height: 400,
  88. colorDark: "#000000",
  89. colorLight: "#ffffff",
  90. correctLevel: QRCode.CorrectLevel.H
  91. });
  92. });
  93. const switchButton = document.getElementById('switchCamera');
  94. let currentStream = null;
  95. let facingMode = 'environment';
  96. async function startCamera() {
  97. try {
  98. if (currentStream) {
  99. currentStream.getTracks().forEach(track => track.stop());
  100. }
  101. const constraints = {
  102. video: {
  103. width: { ideal: 800 },
  104. height: { ideal: 800 },
  105. facingMode: facingMode
  106. },
  107. audio: false
  108. };
  109. const stream = await navigator.mediaDevices.getUserMedia(constraints);
  110. currentStream = stream;
  111. video.srcObject = stream;
  112. video.play();
  113. qrScanner = new QrScanner(videoElem, result => {
  114. camStatus.innerText = 'Camera Success: ' + result.data;
  115. console.log('decoded QR code:', result);
  116. if (result.data.startsWith(share_url)) {
  117. window.parent.location = result.data;
  118. }
  119. }, {
  120. highlightScanRegion: true,
  121. highlightCodeOutline: true,
  122. });
  123. await qrScanner.start();
  124. } catch (error) {
  125. console.error('Error accessing the camera:', error);
  126. alert(`Could not access the camera: ${error.message}`);
  127. }
  128. }
  129. switchButton.addEventListener('click', () => {
  130. facingMode = facingMode === 'user' ? 'environment' : 'user';
  131. startCamera();
  132. });
  133. document.getElementById('startCamera').addEventListener('click', () => {
  134. startCamera();
  135. });
  136. </script>
  137. </body>
  138. </html>