index.html 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  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-current-file {
  33. color: #9ecfff; font-size: 12px;
  34. max-width: 300px; overflow: hidden;
  35. text-overflow: ellipsis; white-space: nowrap;
  36. }
  37. #bb-open-settings {
  38. margin-left: auto;
  39. background: none; color: #aaa;
  40. border: 1px solid #555; border-radius: 4px;
  41. padding: 3px 8px; cursor: pointer; font-size: 12px;
  42. }
  43. #bb-open-settings:hover { color: #fff; border-color: #888; }
  44. /* Playback controls */
  45. #bb-play-btn {
  46. background: #2a7a4a; color: #fff;
  47. border: none; border-radius: 4px;
  48. padding: 4px 10px; cursor: pointer; font-size: 14px;
  49. min-width: 34px;
  50. }
  51. #bb-play-btn:hover { background: #1d5c38; }
  52. #bb-play-btn:disabled { background: #444; cursor: default; }
  53. #bb-play-speed {
  54. background: #222; color: #ddd;
  55. border: 1px solid #444; border-radius: 4px;
  56. padding: 3px 5px; font-size: 12px;
  57. }
  58. /* Main layout */
  59. #bb-layout {
  60. display: flex;
  61. flex-direction: column;
  62. height: 100vh;
  63. }
  64. #bb-viewer-wrap {
  65. flex: 1;
  66. position: relative;
  67. overflow: hidden;
  68. }
  69. /* The prettygcode container — must fill its parent */
  70. .page-container {
  71. width: 100%;
  72. height: 100%;
  73. position: relative;
  74. }
  75. #tab_plugin_prettygcode {
  76. width: 100%;
  77. height: 100%;
  78. }
  79. .gwin {
  80. width: 100%;
  81. height: 100%;
  82. position: relative;
  83. }
  84. #mycanvas {
  85. width: 100%;
  86. height: 100%;
  87. display: block;
  88. }
  89. /* Webcam element cloned by prettygcode.js — keep hidden until fullscreen */
  90. #webcam_rotator {
  91. display: none;
  92. }
  93. #webcam_image { max-width: 100%; }
  94. /* Control buttons inside .gwin */
  95. .pgstatetoggle, .pgfilestoggle, .pgsettingstoggle, .fstoggle,
  96. .pgdashtoggle, .pgcameratoggle {
  97. background: rgba(0,0,0,0.5);
  98. border: 1px solid #666;
  99. border-radius: 4px;
  100. color: #ddd;
  101. cursor: pointer;
  102. font-size: 16px;
  103. padding: 4px 7px;
  104. z-index: 10;
  105. }
  106. .pgstatetoggle:hover, .pgfilestoggle:hover, .pgsettingstoggle:hover,
  107. .fstoggle:hover, .pgdashtoggle:hover, .pgcameratoggle:hover {
  108. background: rgba(60,60,60,0.8);
  109. }
  110. /* Slider shim styles */
  111. .slider {
  112. position: absolute;
  113. right: 20px;
  114. top: 5%;
  115. height: 90%;
  116. width: 28px;
  117. z-index: 10;
  118. cursor: pointer;
  119. user-select: none;
  120. }
  121. .slider-vertical { writing-mode: vertical-lr; }
  122. .slider-track {
  123. position: absolute;
  124. left: 50%;
  125. top: 0; bottom: 0;
  126. width: 4px;
  127. transform: translateX(-50%);
  128. background: #555;
  129. border-radius: 2px;
  130. }
  131. .slider-selection {
  132. position: absolute;
  133. left: 0; right: 0;
  134. bottom: 0;
  135. background: #4a9;
  136. border-radius: 2px;
  137. }
  138. .slider-handle {
  139. position: absolute;
  140. left: 50%;
  141. transform: translateX(-50%);
  142. width: 24px; height: 24px;
  143. background: #6cf;
  144. border-radius: 50%;
  145. line-height: 24px;
  146. text-align: center;
  147. font-size: 9px;
  148. color: #111;
  149. font-weight: bold;
  150. }
  151. </style>
  152. </head>
  153. <body>
  154. <div id="bb-layout">
  155. <!-- Toolbar -->
  156. <div id="bb-toolbar">
  157. <span id="bb-current-file">— no file loaded —</span>
  158. <button id="bb-play-btn" title="Play layer animation" disabled>&#9654;</button>
  159. <select id="bb-play-speed" title="Playback speed">
  160. <option value="1">1&#215; slow</option>
  161. <option value="3" selected>3&#215;</option>
  162. <option value="10">10&#215; fast</option>
  163. <option value="25">25&#215; turbo</option>
  164. </select>
  165. </div>
  166. <!-- Viewer area -->
  167. <div id="bb-viewer-wrap">
  168. <div class="page-container">
  169. <div id="tab_plugin_prettygcode">
  170. <div class="gwin">
  171. <canvas id="mycanvas"></canvas>
  172. <div id="pgstatus" class="pgstatus">T:0/0 B:0/0</div>
  173. <div id="mygui" class="pghidden">View Options</div>
  174. <button class="pgstatetoggle" title="Toggle state window">&#9432;</button>
  175. <button class="pgfilestoggle" title="Toggle file list">&#9776;</button>
  176. <button class="pgsettingstoggle" title="Toggle settings">&#9881;</button>
  177. <button class="pgcameratoggle" title="Toggle webcam">&#128247;</button>
  178. </div>
  179. </div>
  180. <!-- Webcam source element — prettygcode.js clones this into .gwin -->
  181. <div id="webcam_rotator" style="display:none;">
  182. <img id="webcam_image" src="" alt="webcam">
  183. </div>
  184. </div>
  185. </div><!-- /bb-viewer-wrap -->
  186. </div><!-- /bb-layout -->
  187. <!-- Scripts — load order matters -->
  188. <script src="js/jquery.min.js"></script>
  189. <script src="js/slider-shim.js"></script>
  190. <script src="js/three.min.js"></script>
  191. <script src="js/LineSegmentsGeometry.js"></script>
  192. <script src="js/LineGeometry.js"></script>
  193. <script src="js/LineMaterial.js"></script>
  194. <script src="js/LineSegments2.js"></script>
  195. <script src="js/Line2.js"></script>
  196. <script src="js/Lut.js"></script>
  197. <script src="js/OBJLoader.js"></script>
  198. <script src="js/camera-controls.js"></script>
  199. <script src="js/dat.gui.js"></script>
  200. <!-- Adapter MUST load before prettygcode.js -->
  201. <script src="js/bambuddy_adapter.js"></script>
  202. <script src="js/prettygcode.js"></script>
  203. <!-- Button wiring is in bambuddy_adapter.js — inline scripts are blocked by the
  204. script-src CSP on this page (no 'unsafe-inline'). -->
  205. </body>
  206. </html>