index.html 7.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  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. <!-- CSP: all scripts are local; unsafe-eval needed by Three.js shader compiler;
  7. unsafe-inline needed by dat.GUI and jQuery for inline styles/event handlers -->
  8. <meta http-equiv="Content-Security-Policy" content="default-src 'self'; script-src 'self' 'unsafe-eval'; style-src 'self' 'unsafe-inline'; img-src 'self' data: blob:; connect-src 'self' ws: wss:; font-src 'self' data:; worker-src blob:;">
  9. <title>PrettyGCode — Bambuddy</title>
  10. <link rel="stylesheet" href="css/prettygcode.css">
  11. <style>
  12. *, *::before, *::after { box-sizing: border-box; }
  13. html, body {
  14. margin: 0; padding: 0;
  15. height: 100%;
  16. background: #1a1a1a;
  17. color: #e0e0e0;
  18. font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', sans-serif;
  19. font-size: 14px;
  20. }
  21. /* Top toolbar */
  22. #bb-toolbar {
  23. display: flex;
  24. align-items: center;
  25. gap: 10px;
  26. padding: 6px 10px;
  27. background: #111;
  28. border-bottom: 1px solid #333;
  29. flex-shrink: 0;
  30. flex-wrap: wrap;
  31. }
  32. #bb-toolbar label { color: #aaa; font-size: 12px; white-space: nowrap; }
  33. #bb-printer-select {
  34. background: #222; color: #ddd;
  35. border: 1px solid #444; border-radius: 4px;
  36. padding: 3px 6px; font-size: 12px;
  37. }
  38. #bb-file-btn {
  39. background: #2a6496; color: #fff;
  40. border: none; border-radius: 4px;
  41. padding: 4px 10px; cursor: pointer; font-size: 12px;
  42. }
  43. #bb-file-btn:hover { background: #1c4d72; }
  44. #bb-current-file {
  45. color: #9ecfff; font-size: 12px;
  46. max-width: 300px; overflow: hidden;
  47. text-overflow: ellipsis; white-space: nowrap;
  48. }
  49. #bb-open-settings {
  50. margin-left: auto;
  51. background: none; color: #aaa;
  52. border: 1px solid #555; border-radius: 4px;
  53. padding: 3px 8px; cursor: pointer; font-size: 12px;
  54. }
  55. #bb-open-settings:hover { color: #fff; border-color: #888; }
  56. /* Playback controls */
  57. #bb-play-btn {
  58. background: #2a7a4a; color: #fff;
  59. border: none; border-radius: 4px;
  60. padding: 4px 10px; cursor: pointer; font-size: 14px;
  61. min-width: 34px;
  62. }
  63. #bb-play-btn:hover { background: #1d5c38; }
  64. #bb-play-btn:disabled { background: #444; cursor: default; }
  65. #bb-play-speed {
  66. background: #222; color: #ddd;
  67. border: 1px solid #444; border-radius: 4px;
  68. padding: 3px 5px; font-size: 12px;
  69. }
  70. /* File picker dropdown */
  71. #bb-file-picker {
  72. display: none;
  73. position: fixed; /* fixed so it's never clipped by viewer overflow */
  74. top: 38px; left: 10px;
  75. width: 320px;
  76. background: #222; border: 1px solid #444;
  77. border-radius: 6px; padding: 6px;
  78. z-index: 9999;
  79. box-shadow: 0 4px 16px rgba(0,0,0,0.6);
  80. }
  81. #bb-file-picker.bb-open { display: block; }
  82. /* Main layout */
  83. #bb-layout {
  84. display: flex;
  85. flex-direction: column;
  86. height: 100vh;
  87. }
  88. #bb-viewer-wrap {
  89. flex: 1;
  90. position: relative;
  91. overflow: hidden;
  92. }
  93. /* The prettygcode container — must fill its parent */
  94. .page-container {
  95. width: 100%;
  96. height: 100%;
  97. position: relative;
  98. }
  99. #tab_plugin_prettygcode {
  100. width: 100%;
  101. height: 100%;
  102. }
  103. .gwin {
  104. width: 100%;
  105. height: 100%;
  106. position: relative;
  107. }
  108. #mycanvas {
  109. width: 100%;
  110. height: 100%;
  111. display: block;
  112. }
  113. /* Webcam element cloned by prettygcode.js — keep hidden until fullscreen */
  114. #webcam_rotator {
  115. display: none;
  116. }
  117. #webcam_image { max-width: 100%; }
  118. /* Control buttons inside .gwin */
  119. .pgstatetoggle, .pgfilestoggle, .pgsettingstoggle, .fstoggle,
  120. .pgdashtoggle, .pgcameratoggle {
  121. background: rgba(0,0,0,0.5);
  122. border: 1px solid #666;
  123. border-radius: 4px;
  124. color: #ddd;
  125. cursor: pointer;
  126. font-size: 16px;
  127. padding: 4px 7px;
  128. z-index: 10;
  129. }
  130. .pgstatetoggle:hover, .pgfilestoggle:hover, .pgsettingstoggle:hover,
  131. .fstoggle:hover, .pgdashtoggle:hover, .pgcameratoggle:hover {
  132. background: rgba(60,60,60,0.8);
  133. }
  134. /* Slider shim styles */
  135. .slider {
  136. position: absolute;
  137. right: 20px;
  138. top: 5%;
  139. height: 90%;
  140. width: 28px;
  141. z-index: 10;
  142. cursor: pointer;
  143. user-select: none;
  144. }
  145. .slider-vertical { writing-mode: vertical-lr; }
  146. .slider-track {
  147. position: absolute;
  148. left: 50%;
  149. top: 0; bottom: 0;
  150. width: 4px;
  151. transform: translateX(-50%);
  152. background: #555;
  153. border-radius: 2px;
  154. }
  155. .slider-selection {
  156. position: absolute;
  157. left: 0; right: 0;
  158. bottom: 0;
  159. background: #4a9;
  160. border-radius: 2px;
  161. }
  162. .slider-handle {
  163. position: absolute;
  164. left: 50%;
  165. transform: translateX(-50%);
  166. width: 24px; height: 24px;
  167. background: #6cf;
  168. border-radius: 50%;
  169. line-height: 24px;
  170. text-align: center;
  171. font-size: 9px;
  172. color: #111;
  173. font-weight: bold;
  174. }
  175. </style>
  176. </head>
  177. <body>
  178. <div id="bb-layout">
  179. <!-- Toolbar -->
  180. <div id="bb-toolbar">
  181. <label for="bb-printer-select">Printer:</label>
  182. <select id="bb-printer-select"><option value="">Loading…</option></select>
  183. <button id="bb-file-btn">&#128196; Load file</button>
  184. <span id="bb-current-file">— no file loaded —</span>
  185. <button id="bb-play-btn" title="Play layer animation" disabled>&#9654;</button>
  186. <select id="bb-play-speed" title="Playback speed">
  187. <option value="1">1&#215; slow</option>
  188. <option value="3" selected>3&#215;</option>
  189. <option value="10">10&#215; fast</option>
  190. <option value="25">25&#215; turbo</option>
  191. </select>
  192. </div>
  193. <!-- File picker (positioned relative to toolbar) -->
  194. <div id="bb-file-picker"></div>
  195. <!-- Viewer area -->
  196. <div id="bb-viewer-wrap">
  197. <div class="page-container">
  198. <div id="tab_plugin_prettygcode">
  199. <div class="gwin">
  200. <canvas id="mycanvas"></canvas>
  201. <div id="pgstatus" class="pgstatus">T:0/0 B:0/0</div>
  202. <div id="mygui" class="pghidden">View Options</div>
  203. <button class="pgstatetoggle" title="Toggle state window">&#9432;</button>
  204. <button class="pgfilestoggle" title="Toggle file list">&#9776;</button>
  205. <button class="pgsettingstoggle" title="Toggle settings">&#9881;</button>
  206. <button class="pgcameratoggle" title="Toggle webcam">&#128247;</button>
  207. </div>
  208. </div>
  209. <!-- Webcam source element — prettygcode.js clones this into .gwin -->
  210. <div id="webcam_rotator" style="display:none;">
  211. <img id="webcam_image" src="" alt="webcam">
  212. </div>
  213. </div>
  214. </div><!-- /bb-viewer-wrap -->
  215. </div><!-- /bb-layout -->
  216. <!-- Scripts — load order matters -->
  217. <script src="js/jquery.min.js"></script>
  218. <script src="js/slider-shim.js"></script>
  219. <script src="js/three.min.js"></script>
  220. <script src="js/LineSegmentsGeometry.js"></script>
  221. <script src="js/LineGeometry.js"></script>
  222. <script src="js/LineMaterial.js"></script>
  223. <script src="js/LineSegments2.js"></script>
  224. <script src="js/Line2.js"></script>
  225. <script src="js/Lut.js"></script>
  226. <script src="js/OBJLoader.js"></script>
  227. <script src="js/camera-controls.js"></script>
  228. <script src="js/dat.gui.js"></script>
  229. <!-- Adapter MUST load before prettygcode.js -->
  230. <script src="js/bambuddy_adapter.js"></script>
  231. <script src="js/prettygcode.js"></script>
  232. <!-- Button wiring is in bambuddy_adapter.js — inline scripts are blocked by the
  233. script-src CSP on this page (no 'unsafe-inline'). -->
  234. </body>
  235. </html>