viewer_old.html 58 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196
  1. <!DOCTYPE html>
  2. <html dir="ltr" mozdisallowselectionprint>
  3. <head>
  4. <meta charset="utf-8">
  5. <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
  6. <meta name="google" content="notranslate">
  7. <title>PDF.js viewer</title>
  8. <!-- This snippet is used in production (included from viewer.html) -->
  9. <link rel="resource" type="application/l10n" href="locale/locale.json">
  10. <script src="../build/pdf.mjs" type="module"></script>
  11. <link rel="stylesheet" href="viewer.css">
  12. <script src="viewer.mjs" type="module"></script>
  13. <style>
  14. /* 模态窗口样式 */
  15. .modal {
  16. display: none;
  17. /* 默认隐藏 */
  18. position: fixed;
  19. z-index: 10002;
  20. left: 50%;
  21. top: 50%;
  22. transform: translate(-50%, -50%);
  23. width: 500px;
  24. background-color: white;
  25. border: 1px solid #ccc;
  26. border-radius: 5px;
  27. padding: 20px;
  28. box-shadow: 0 5px 15px rgba(0, 0, 0, 0.3);
  29. }
  30. /* 遮罩层样式 */
  31. .modal-overlay {
  32. display: none;
  33. /* 默认隐藏 */
  34. position: fixed;
  35. top: 0;
  36. left: 0;
  37. width: 100%;
  38. height: 100%;
  39. background-color: rgba(0, 0, 0, 0.5);
  40. z-index: 10001;
  41. }
  42. /* 按钮样式 */
  43. .button {
  44. padding: 10px 20px;
  45. background-color: #007bff;
  46. color: white;
  47. border: none;
  48. border-radius: 5px;
  49. cursor: pointer;
  50. }
  51. .button-secondary {
  52. background-color: #6c757d;
  53. }
  54. .button-success {
  55. background-color: #28a745;
  56. }
  57. .button-light {
  58. background-color: #ded520;
  59. }
  60. .button-danger {
  61. background-color: #dc3545;
  62. }
  63. </style>
  64. </head>
  65. <body tabindex="0">
  66. <!-- 等待指示器 -->
  67. <div id="loading-indicator"
  68. style="display: none; position: fixed; top: 20px; left: 20px; background: rgba(0,0,0,0.7); color: #fff; padding: 10px; border-radius: 5px;">
  69. 正在生成语音,请稍候...
  70. </div>
  71. <audio id="audio-player" controls
  72. style="position: fixed; width: 190px;height: 50px; bottom: 10px; left: 10px; z-index: 1000;"></audio>
  73. <!-- 上传按钮定位在页面底部 -->
  74. <button id="uploadButton" style="
  75. position: fixed;
  76. bottom: 20px;
  77. right: 20px;
  78. z-index: 1000;
  79. padding: 10px 20px;
  80. background-color: #007bff;
  81. color: white;
  82. border: none;
  83. border-radius: 5px;
  84. cursor: pointer;">
  85. 上传 PDF
  86. </button>
  87. <!-- 隐藏的文件输入 -->
  88. <input type="file" id="pdfInput" accept="application/pdf" style="display: none;" />
  89. <!-- 模态窗口 -->
  90. <div id="filenameModal" class="modal">
  91. <h3>输入保存的文件名</h3>
  92. <input type="text" id="customFileName" placeholder="不含扩展名" style="width: 100%; padding: 5px; margin-top: 10px;" />
  93. <div style="margin-top: 20px; text-align: right;">
  94. <button id="cancelUpload" class="button button-secondary">取消</button>
  95. <button id="confirmUpload" class="button button-success" style="margin-left: 10px;">确认</button>
  96. </div>
  97. </div>
  98. <!-- 页面遮罩 -->
  99. <div id="modalOverlay" class="modal-overlay"></div>
  100. <!-- 打开目录按钮 -->
  101. <button id="openDirectoryButton" class="button" style="
  102. position: fixed;
  103. bottom: 70px;
  104. right: 20px;
  105. z-index: 1000;
  106. ">打开目录
  107. </button>
  108. <button id="select-all-text" class="button button-light" style="
  109. position: fixed;
  110. bottom: 170px;
  111. right: 20px;
  112. z-index: 1000;
  113. color:black;
  114. ">阅读整页</button>
  115. <button id="toggle-text-select" class="button button-success" style="
  116. position: fixed;
  117. bottom: 120px;
  118. right: 20px;
  119. z-index: 1000;
  120. ">文本选择</button>
  121. <!-- 目录弹出层 -->
  122. <div id="directoryModal" class="modal">
  123. <h2>已上传的 PDF 文档</h2>
  124. <ul id="pdfList" style="list-style-type: none; padding: 0;"></ul>
  125. <button id="closeDirectoryButton" class="button button-danger" style="margin-top: 20px; width: 100%;">关闭</button>
  126. </div>
  127. <div id="outerContainer">
  128. <div id="sidebarContainer">
  129. <div id="toolbarSidebar" class="toolbarHorizontalGroup">
  130. <div id="toolbarSidebarLeft">
  131. <div id="sidebarViewButtons" class="toolbarHorizontalGroup toggled" role="radiogroup">
  132. <button id="viewThumbnail" class="toolbarButton toggled" type="button" title="Show Thumbnails" tabindex="0"
  133. data-l10n-id="pdfjs-thumbs-button" role="radio" aria-checked="true" aria-controls="thumbnailView">
  134. <span data-l10n-id="pdfjs-thumbs-button-label">Thumbnails</span>
  135. </button>
  136. <button id="viewOutline" class="toolbarButton" type="button"
  137. title="Show Document Outline (double-click to expand/collapse all items)" tabindex="0"
  138. data-l10n-id="pdfjs-document-outline-button" role="radio" aria-checked="false"
  139. aria-controls="outlineView">
  140. <span data-l10n-id="pdfjs-document-outline-button-label">Document Outline</span>
  141. </button>
  142. <button id="viewAttachments" class="toolbarButton" type="button" title="Show Attachments" tabindex="0"
  143. data-l10n-id="pdfjs-attachments-button" role="radio" aria-checked="false" aria-controls="attachmentsView">
  144. <span data-l10n-id="pdfjs-attachments-button-label">Attachments</span>
  145. </button>
  146. <button id="viewLayers" class="toolbarButton" type="button"
  147. title="Show Layers (double-click to reset all layers to the default state)" tabindex="0"
  148. data-l10n-id="pdfjs-layers-button" role="radio" aria-checked="false" aria-controls="layersView">
  149. <span data-l10n-id="pdfjs-layers-button-label">Layers</span>
  150. </button>
  151. </div>
  152. </div>
  153. <div id="toolbarSidebarRight">
  154. <div id="outlineOptionsContainer" class="toolbarHorizontalGroup">
  155. <div class="verticalToolbarSeparator"></div>
  156. <button id="currentOutlineItem" class="toolbarButton" type="button" disabled="disabled"
  157. title="Find Current Outline Item" tabindex="0" data-l10n-id="pdfjs-current-outline-item-button">
  158. <span data-l10n-id="pdfjs-current-outline-item-button-label">Current Outline Item</span>
  159. </button>
  160. </div>
  161. </div>
  162. </div>
  163. <div id="sidebarContent">
  164. <div id="thumbnailView">
  165. </div>
  166. <div id="outlineView" class="hidden">
  167. </div>
  168. <div id="attachmentsView" class="hidden">
  169. </div>
  170. <div id="layersView" class="hidden">
  171. </div>
  172. </div>
  173. <div id="sidebarResizer"></div>
  174. </div> <!-- sidebarContainer -->
  175. <div id="mainContainer">
  176. <div class="toolbar">
  177. <div id="toolbarContainer">
  178. <div id="toolbarViewer" class="toolbarHorizontalGroup">
  179. <div id="toolbarViewerLeft" class="toolbarHorizontalGroup">
  180. <button id="sidebarToggleButton" class="toolbarButton" type="button" title="Toggle Sidebar" tabindex="0"
  181. data-l10n-id="pdfjs-toggle-sidebar-button" aria-expanded="false" aria-haspopup="true"
  182. aria-controls="sidebarContainer">
  183. <span data-l10n-id="pdfjs-toggle-sidebar-button-label">Toggle Sidebar</span>
  184. </button>
  185. <div class="toolbarButtonSpacer"></div>
  186. <div class="toolbarButtonWithContainer">
  187. <button id="viewFindButton" class="toolbarButton" type="button" title="Find in Document" tabindex="0"
  188. data-l10n-id="pdfjs-findbar-button" aria-expanded="false" aria-controls="findbar">
  189. <span data-l10n-id="pdfjs-findbar-button-label">Find</span>
  190. </button>
  191. <div class="hidden doorHanger toolbarHorizontalGroup" id="findbar">
  192. <div id="findInputContainer" class="toolbarHorizontalGroup">
  193. <span class="loadingInput end toolbarHorizontalGroup">
  194. <input id="findInput" class="toolbarField" title="Find" placeholder="Find in document…"
  195. tabindex="0" data-l10n-id="pdfjs-find-input" aria-invalid="false">
  196. </span>
  197. <div class="toolbarHorizontalGroup">
  198. <button id="findPreviousButton" class="toolbarButton" type="button"
  199. title="Find the previous occurrence of the phrase" tabindex="0"
  200. data-l10n-id="pdfjs-find-previous-button">
  201. <span data-l10n-id="pdfjs-find-previous-button-label">Previous</span>
  202. </button>
  203. <div class="splitToolbarButtonSeparator"></div>
  204. <button id="findNextButton" class="toolbarButton" type="button"
  205. title="Find the next occurrence of the phrase" tabindex="0"
  206. data-l10n-id="pdfjs-find-next-button">
  207. <span data-l10n-id="pdfjs-find-next-button-label">Next</span>
  208. </button>
  209. </div>
  210. </div>
  211. <div id="findbarOptionsOneContainer" class="toolbarHorizontalGroup">
  212. <div class="toggleButton toolbarLabel">
  213. <input type="checkbox" id="findHighlightAll" tabindex="0" />
  214. <label for="findHighlightAll" data-l10n-id="pdfjs-find-highlight-checkbox">Highlight All</label>
  215. </div>
  216. <div class="toggleButton toolbarLabel">
  217. <input type="checkbox" id="findMatchCase" tabindex="0" />
  218. <label for="findMatchCase" data-l10n-id="pdfjs-find-match-case-checkbox-label">Match Case</label>
  219. </div>
  220. </div>
  221. <div id="findbarOptionsTwoContainer" class="toolbarHorizontalGroup">
  222. <div class="toggleButton toolbarLabel">
  223. <input type="checkbox" id="findMatchDiacritics" tabindex="0" />
  224. <label for="findMatchDiacritics" data-l10n-id="pdfjs-find-match-diacritics-checkbox-label">Match
  225. Diacritics</label>
  226. </div>
  227. <div class="toggleButton toolbarLabel">
  228. <input type="checkbox" id="findEntireWord" tabindex="0" />
  229. <label for="findEntireWord" data-l10n-id="pdfjs-find-entire-word-checkbox-label">Whole
  230. Words</label>
  231. </div>
  232. </div>
  233. <div id="findbarMessageContainer" class="toolbarHorizontalGroup" aria-live="polite">
  234. <span id="findResultsCount" class="toolbarLabel"></span>
  235. <span id="findMsg" class="toolbarLabel"></span>
  236. </div>
  237. </div> <!-- findbar -->
  238. </div>
  239. <div class="toolbarHorizontalGroup hiddenSmallView">
  240. <button class="toolbarButton" title="Previous Page" type="button" id="previous" tabindex="0"
  241. data-l10n-id="pdfjs-previous-button">
  242. <span data-l10n-id="pdfjs-previous-button-label">Previous</span>
  243. </button>
  244. <div class="splitToolbarButtonSeparator"></div>
  245. <button class="toolbarButton" type="button" title="Next Page" id="next" tabindex="0"
  246. data-l10n-id="pdfjs-next-button">
  247. <span data-l10n-id="pdfjs-next-button-label">Next</span>
  248. </button>
  249. </div>
  250. <div class="toolbarHorizontalGroup">
  251. <span class="loadingInput start toolbarHorizontalGroup">
  252. <input type="number" id="pageNumber" class="toolbarField" title="Page" value="1" min="1" tabindex="0"
  253. data-l10n-id="pdfjs-page-input" autocomplete="off">
  254. </span>
  255. <span id="numPages" class="toolbarLabel"></span>
  256. </div>
  257. </div>
  258. <div id="toolbarViewerMiddle" class="toolbarHorizontalGroup">
  259. <div class="toolbarHorizontalGroup">
  260. <button id="zoomOutButton" class="toolbarButton" type="button" title="Zoom Out" tabindex="0"
  261. data-l10n-id="pdfjs-zoom-out-button">
  262. <span data-l10n-id="pdfjs-zoom-out-button-label">Zoom Out</span>
  263. </button>
  264. <div class="splitToolbarButtonSeparator"></div>
  265. <button id="zoomInButton" class="toolbarButton" type="button" title="Zoom In" tabindex="0"
  266. data-l10n-id="pdfjs-zoom-in-button">
  267. <span data-l10n-id="pdfjs-zoom-in-button-label">Zoom In</span>
  268. </button>
  269. </div>
  270. <span id="scaleSelectContainer" class="dropdownToolbarButton">
  271. <select id="scaleSelect" title="Zoom" tabindex="0" data-l10n-id="pdfjs-zoom-select">
  272. <option id="pageAutoOption" title="" value="auto" selected="selected"
  273. data-l10n-id="pdfjs-page-scale-auto">Automatic Zoom</option>
  274. <option id="pageActualOption" title="" value="page-actual" data-l10n-id="pdfjs-page-scale-actual">
  275. Actual Size</option>
  276. <option id="pageFitOption" title="" value="page-fit" data-l10n-id="pdfjs-page-scale-fit">Page Fit
  277. </option>
  278. <option id="pageWidthOption" title="" value="page-width" data-l10n-id="pdfjs-page-scale-width">Page
  279. Width</option>
  280. <option id="customScaleOption" title="" value="custom" disabled="disabled" hidden="true"
  281. data-l10n-id="pdfjs-page-scale-percent" data-l10n-args='{ "scale": 0 }'>0%</option>
  282. <option title="" value="0.5" data-l10n-id="pdfjs-page-scale-percent" data-l10n-args='{ "scale": 50 }'>
  283. 50%</option>
  284. <option title="" value="0.75" data-l10n-id="pdfjs-page-scale-percent"
  285. data-l10n-args='{ "scale": 75 }'>
  286. 75%</option>
  287. <option title="" value="1" data-l10n-id="pdfjs-page-scale-percent" data-l10n-args='{ "scale": 100 }'>
  288. 100%</option>
  289. <option title="" value="1.25" data-l10n-id="pdfjs-page-scale-percent"
  290. data-l10n-args='{ "scale": 125 }'>
  291. 125%</option>
  292. <option title="" value="1.5" data-l10n-id="pdfjs-page-scale-percent"
  293. data-l10n-args='{ "scale": 150 }'>
  294. 150%</option>
  295. <option title="" value="2" data-l10n-id="pdfjs-page-scale-percent" data-l10n-args='{ "scale": 200 }'>
  296. 200%</option>
  297. <option title="" value="3" data-l10n-id="pdfjs-page-scale-percent" data-l10n-args='{ "scale": 300 }'>
  298. 300%</option>
  299. <option title="" value="4" data-l10n-id="pdfjs-page-scale-percent" data-l10n-args='{ "scale": 400 }'>
  300. 400%</option>
  301. </select>
  302. </span>
  303. </div>
  304. <div id="toolbarViewerRight" class="toolbarHorizontalGroup">
  305. <div id="editorModeButtons" class="toolbarHorizontalGroup" role="radiogroup">
  306. <div id="editorHighlight" class="toolbarButtonWithContainer">
  307. <button id="editorHighlightButton" class="toolbarButton" type="button" disabled="disabled"
  308. title="Highlight" role="radio" aria-expanded="false" aria-haspopup="true"
  309. aria-controls="editorHighlightParamsToolbar" tabindex="0"
  310. data-l10n-id="pdfjs-editor-highlight-button">
  311. <span data-l10n-id="pdfjs-editor-highlight-button-label">Highlight</span>
  312. </button>
  313. <div class="editorParamsToolbar hidden doorHangerRight" id="editorHighlightParamsToolbar">
  314. <div id="highlightParamsToolbarContainer" class="editorParamsToolbarContainer">
  315. <div id="editorHighlightColorPicker" class="colorPicker">
  316. <span id="highlightColorPickerLabel" class="editorParamsLabel"
  317. data-l10n-id="pdfjs-editor-highlight-colorpicker-label">Highlight color</span>
  318. </div>
  319. <div id="editorHighlightThickness">
  320. <label for="editorFreeHighlightThickness" class="editorParamsLabel"
  321. data-l10n-id="pdfjs-editor-free-highlight-thickness-input">Thickness</label>
  322. <div class="thicknessPicker">
  323. <input type="range" id="editorFreeHighlightThickness" class="editorParamsSlider"
  324. data-l10n-id="pdfjs-editor-free-highlight-thickness-title" value="12" min="8" max="24"
  325. step="1" tabindex="0">
  326. </div>
  327. </div>
  328. <div id="editorHighlightVisibility">
  329. <div class="divider"></div>
  330. <div class="toggler">
  331. <label for="editorHighlightShowAll" class="editorParamsLabel"
  332. data-l10n-id="pdfjs-editor-highlight-show-all-button-label">Show all</label>
  333. <button id="editorHighlightShowAll" class="toggle-button" type="button"
  334. data-l10n-id="pdfjs-editor-highlight-show-all-button" aria-pressed="true"
  335. tabindex="0"></button>
  336. </div>
  337. </div>
  338. </div>
  339. </div>
  340. </div>
  341. <div id="editorFreeText" class="toolbarButtonWithContainer">
  342. <button id="editorFreeTextButton" class="toolbarButton" type="button" disabled="disabled" title="Text"
  343. role="radio" aria-expanded="false" aria-haspopup="true" aria-controls="editorFreeTextParamsToolbar"
  344. tabindex="0" data-l10n-id="pdfjs-editor-free-text-button">
  345. <span data-l10n-id="pdfjs-editor-free-text-button-label">Text</span>
  346. </button>
  347. <div class="editorParamsToolbar hidden doorHangerRight" id="editorFreeTextParamsToolbar">
  348. <div class="editorParamsToolbarContainer">
  349. <div class="editorParamsSetter">
  350. <label for="editorFreeTextColor" class="editorParamsLabel"
  351. data-l10n-id="pdfjs-editor-free-text-color-input">Color</label>
  352. <input type="color" id="editorFreeTextColor" class="editorParamsColor" tabindex="0">
  353. </div>
  354. <div class="editorParamsSetter">
  355. <label for="editorFreeTextFontSize" class="editorParamsLabel"
  356. data-l10n-id="pdfjs-editor-free-text-size-input">Size</label>
  357. <input type="range" id="editorFreeTextFontSize" class="editorParamsSlider" value="10" min="5"
  358. max="100" step="1" tabindex="0">
  359. </div>
  360. </div>
  361. </div>
  362. </div>
  363. <div id="editorInk" class="toolbarButtonWithContainer">
  364. <button id="editorInkButton" class="toolbarButton" type="button" disabled="disabled" title="Draw"
  365. role="radio" aria-expanded="false" aria-haspopup="true" aria-controls="editorInkParamsToolbar"
  366. tabindex="0" data-l10n-id="pdfjs-editor-ink-button">
  367. <span data-l10n-id="pdfjs-editor-ink-button-label">Draw</span>
  368. </button>
  369. <div class="editorParamsToolbar hidden doorHangerRight" id="editorInkParamsToolbar">
  370. <div class="editorParamsToolbarContainer">
  371. <div class="editorParamsSetter">
  372. <label for="editorInkColor" class="editorParamsLabel"
  373. data-l10n-id="pdfjs-editor-ink-color-input">Color</label>
  374. <input type="color" id="editorInkColor" class="editorParamsColor" tabindex="0">
  375. </div>
  376. <div class="editorParamsSetter">
  377. <label for="editorInkThickness" class="editorParamsLabel"
  378. data-l10n-id="pdfjs-editor-ink-thickness-input">Thickness</label>
  379. <input type="range" id="editorInkThickness" class="editorParamsSlider" value="1" min="1"
  380. max="20" step="1" tabindex="0">
  381. </div>
  382. <div class="editorParamsSetter">
  383. <label for="editorInkOpacity" class="editorParamsLabel"
  384. data-l10n-id="pdfjs-editor-ink-opacity-input">Opacity</label>
  385. <input type="range" id="editorInkOpacity" class="editorParamsSlider" value="100" min="1"
  386. max="100" step="1" tabindex="0">
  387. </div>
  388. </div>
  389. </div>
  390. </div>
  391. <div id="editorStamp" class="toolbarButtonWithContainer">
  392. <button id="editorStampButton" class="toolbarButton" type="button" disabled="disabled"
  393. title="Add or edit images" role="radio" aria-expanded="false" aria-haspopup="true"
  394. aria-controls="editorStampParamsToolbar" tabindex="0" data-l10n-id="pdfjs-editor-stamp-button">
  395. <span data-l10n-id="pdfjs-editor-stamp-button-label">Add or edit images</span>
  396. </button>
  397. <div class="editorParamsToolbar hidden doorHangerRight menu" id="editorStampParamsToolbar">
  398. <div class="menuContainer">
  399. <button id="editorStampAddImage" class="toolbarButton labeled" type="button" title="Add image"
  400. tabindex="0" data-l10n-id="pdfjs-editor-stamp-add-image-button">
  401. <span class="editorParamsLabel" data-l10n-id="pdfjs-editor-stamp-add-image-button-label">Add
  402. image</span>
  403. </button>
  404. </div>
  405. </div>
  406. </div>
  407. </div>
  408. <div id="editorModeSeparator" class="verticalToolbarSeparator"></div>
  409. <div class="toolbarHorizontalGroup hiddenMediumView">
  410. <button id="printButton" class="toolbarButton" type="button" title="Print" tabindex="0"
  411. data-l10n-id="pdfjs-print-button">
  412. <span data-l10n-id="pdfjs-print-button-label">Print</span>
  413. </button>
  414. <button id="downloadButton" class="toolbarButton" type="button" title="Save" tabindex="0"
  415. data-l10n-id="pdfjs-save-button">
  416. <span data-l10n-id="pdfjs-save-button-label">Save</span>
  417. </button>
  418. </div>
  419. <div class="verticalToolbarSeparator hiddenMediumView"></div>
  420. <div id="secondaryToolbarToggle" class="toolbarButtonWithContainer">
  421. <button id="secondaryToolbarToggleButton" class="toolbarButton" type="button" title="Tools" tabindex="0"
  422. data-l10n-id="pdfjs-tools-button" aria-expanded="false" aria-haspopup="true"
  423. aria-controls="secondaryToolbar">
  424. <span data-l10n-id="pdfjs-tools-button-label">Tools</span>
  425. </button>
  426. <div id="secondaryToolbar" class="hidden doorHangerRight menu">
  427. <div id="secondaryToolbarButtonContainer" class="menuContainer">
  428. <button id="secondaryOpenFile" class="toolbarButton labeled" type="button" title="Open File"
  429. tabindex="0" data-l10n-id="pdfjs-open-file-button">
  430. <span data-l10n-id="pdfjs-open-file-button-label">Open</span>
  431. </button>
  432. <div class="visibleMediumView">
  433. <button id="secondaryPrint" class="toolbarButton labeled" type="button" title="Print" tabindex="0"
  434. data-l10n-id="pdfjs-print-button">
  435. <span data-l10n-id="pdfjs-print-button-label">Print</span>
  436. </button>
  437. <button id="secondaryDownload" class="toolbarButton labeled" type="button" title="Save"
  438. tabindex="0" data-l10n-id="pdfjs-save-button">
  439. <span data-l10n-id="pdfjs-save-button-label">Save</span>
  440. </button>
  441. </div>
  442. <div class="horizontalToolbarSeparator"></div>
  443. <button id="presentationMode" class="toolbarButton labeled" type="button"
  444. title="Switch to Presentation Mode" tabindex="0" data-l10n-id="pdfjs-presentation-mode-button">
  445. <span data-l10n-id="pdfjs-presentation-mode-button-label">Presentation Mode</span>
  446. </button>
  447. <a href="#" id="viewBookmark" class="toolbarButton labeled"
  448. title="Current Page (View URL from Current Page)" tabindex="0"
  449. data-l10n-id="pdfjs-bookmark-button">
  450. <span data-l10n-id="pdfjs-bookmark-button-label">Current Page</span>
  451. </a>
  452. <div id="viewBookmarkSeparator" class="horizontalToolbarSeparator"></div>
  453. <button id="firstPage" class="toolbarButton labeled" type="button" title="Go to First Page"
  454. tabindex="0" data-l10n-id="pdfjs-first-page-button">
  455. <span data-l10n-id="pdfjs-first-page-button-label">Go to First Page</span>
  456. </button>
  457. <button id="lastPage" class="toolbarButton labeled" type="button" title="Go to Last Page"
  458. tabindex="0" data-l10n-id="pdfjs-last-page-button">
  459. <span data-l10n-id="pdfjs-last-page-button-label">Go to Last Page</span>
  460. </button>
  461. <div class="horizontalToolbarSeparator"></div>
  462. <button id="pageRotateCw" class="toolbarButton labeled" type="button" title="Rotate Clockwise"
  463. tabindex="0" data-l10n-id="pdfjs-page-rotate-cw-button">
  464. <span data-l10n-id="pdfjs-page-rotate-cw-button-label">Rotate Clockwise</span>
  465. </button>
  466. <button id="pageRotateCcw" class="toolbarButton labeled" type="button"
  467. title="Rotate Counterclockwise" tabindex="0" data-l10n-id="pdfjs-page-rotate-ccw-button">
  468. <span data-l10n-id="pdfjs-page-rotate-ccw-button-label">Rotate Counterclockwise</span>
  469. </button>
  470. <div class="horizontalToolbarSeparator"></div>
  471. <div id="cursorToolButtons" role="radiogroup">
  472. <button id="cursorSelectTool" class="toolbarButton labeled toggled" type="button"
  473. title="Enable Text Selection Tool" tabindex="0"
  474. data-l10n-id="pdfjs-cursor-text-select-tool-button" role="radio" aria-checked="true">
  475. <span data-l10n-id="pdfjs-cursor-text-select-tool-button-label">Text Selection Tool</span>
  476. </button>
  477. <button id="cursorHandTool" class="toolbarButton labeled" type="button" title="Enable Hand Tool"
  478. tabindex="0" data-l10n-id="pdfjs-cursor-hand-tool-button" role="radio" aria-checked="false">
  479. <span data-l10n-id="pdfjs-cursor-hand-tool-button-label">Hand Tool</span>
  480. </button>
  481. </div>
  482. <div class="horizontalToolbarSeparator"></div>
  483. <div id="scrollModeButtons" role="radiogroup">
  484. <button id="scrollPage" class="toolbarButton labeled" type="button" title="Use Page Scrolling"
  485. tabindex="0" data-l10n-id="pdfjs-scroll-page-button" role="radio" aria-checked="false">
  486. <span data-l10n-id="pdfjs-scroll-page-button-label">Page Scrolling</span>
  487. </button>
  488. <button id="scrollVertical" class="toolbarButton labeled toggled" type="button"
  489. title="Use Vertical Scrolling" tabindex="0" data-l10n-id="pdfjs-scroll-vertical-button"
  490. role="radio" aria-checked="true">
  491. <span data-l10n-id="pdfjs-scroll-vertical-button-label">Vertical Scrolling</span>
  492. </button>
  493. <button id="scrollHorizontal" class="toolbarButton labeled" type="button"
  494. title="Use Horizontal Scrolling" tabindex="0" data-l10n-id="pdfjs-scroll-horizontal-button"
  495. role="radio" aria-checked="false">
  496. <span data-l10n-id="pdfjs-scroll-horizontal-button-label">Horizontal Scrolling</span>
  497. </button>
  498. <button id="scrollWrapped" class="toolbarButton labeled" type="button"
  499. title="Use Wrapped Scrolling" tabindex="0" data-l10n-id="pdfjs-scroll-wrapped-button"
  500. role="radio" aria-checked="false">
  501. <span data-l10n-id="pdfjs-scroll-wrapped-button-label">Wrapped Scrolling</span>
  502. </button>
  503. </div>
  504. <div class="horizontalToolbarSeparator"></div>
  505. <div id="spreadModeButtons" role="radiogroup">
  506. <button id="spreadNone" class="toolbarButton labeled toggled" type="button"
  507. title="Do not join page spreads" tabindex="0" data-l10n-id="pdfjs-spread-none-button"
  508. role="radio" aria-checked="true">
  509. <span data-l10n-id="pdfjs-spread-none-button-label">No Spreads</span>
  510. </button>
  511. <button id="spreadOdd" class="toolbarButton labeled" type="button"
  512. title="Join page spreads starting with odd-numbered pages" tabindex="0"
  513. data-l10n-id="pdfjs-spread-odd-button" role="radio" aria-checked="false">
  514. <span data-l10n-id="pdfjs-spread-odd-button-label">Odd Spreads</span>
  515. </button>
  516. <button id="spreadEven" class="toolbarButton labeled" type="button"
  517. title="Join page spreads starting with even-numbered pages" tabindex="0"
  518. data-l10n-id="pdfjs-spread-even-button" role="radio" aria-checked="false">
  519. <span data-l10n-id="pdfjs-spread-even-button-label">Even Spreads</span>
  520. </button>
  521. </div>
  522. <div id="imageAltTextSettingsSeparator" class="horizontalToolbarSeparator hidden"></div>
  523. <button id="imageAltTextSettings" type="button" class="toolbarButton labeled hidden"
  524. title="Image alt text settings" tabindex="0" data-l10n-id="pdfjs-image-alt-text-settings-button"
  525. aria-controls="altTextSettingsDialog">
  526. <span data-l10n-id="pdfjs-image-alt-text-settings-button-label">Image alt text settings</span>
  527. </button>
  528. <div class="horizontalToolbarSeparator"></div>
  529. <button id="documentProperties" class="toolbarButton labeled" type="button"
  530. title="Document Properties…" tabindex="0" data-l10n-id="pdfjs-document-properties-button"
  531. aria-controls="documentPropertiesDialog">
  532. <span data-l10n-id="pdfjs-document-properties-button-label">Document Properties…</span>
  533. </button>
  534. </div>
  535. </div> <!-- secondaryToolbar -->
  536. </div>
  537. </div>
  538. </div>
  539. <div id="loadingBar">
  540. <div class="progress">
  541. <div class="glimmer">
  542. </div>
  543. </div>
  544. </div>
  545. </div>
  546. </div>
  547. <div id="viewerContainer" tabindex="0">
  548. <div id="viewer" class="pdfViewer"></div>
  549. </div>
  550. </div> <!-- mainContainer -->
  551. <div id="dialogContainer">
  552. <dialog id="passwordDialog">
  553. <div class="row">
  554. <label for="password" id="passwordText" data-l10n-id="pdfjs-password-label">Enter the password to open this
  555. PDF file:</label>
  556. </div>
  557. <div class="row">
  558. <input type="password" id="password" class="toolbarField">
  559. </div>
  560. <div class="buttonRow">
  561. <button id="passwordCancel" class="dialogButton" type="button"><span
  562. data-l10n-id="pdfjs-password-cancel-button">Cancel</span></button>
  563. <button id="passwordSubmit" class="dialogButton" type="button"><span
  564. data-l10n-id="pdfjs-password-ok-button">OK</span></button>
  565. </div>
  566. </dialog>
  567. <dialog id="documentPropertiesDialog">
  568. <div class="row">
  569. <span id="fileNameLabel" data-l10n-id="pdfjs-document-properties-file-name">File name:</span>
  570. <p id="fileNameField" aria-labelledby="fileNameLabel">-</p>
  571. </div>
  572. <div class="row">
  573. <span id="fileSizeLabel" data-l10n-id="pdfjs-document-properties-file-size">File size:</span>
  574. <p id="fileSizeField" aria-labelledby="fileSizeLabel">-</p>
  575. </div>
  576. <div class="separator"></div>
  577. <div class="row">
  578. <span id="titleLabel" data-l10n-id="pdfjs-document-properties-title">Title:</span>
  579. <p id="titleField" aria-labelledby="titleLabel">-</p>
  580. </div>
  581. <div class="row">
  582. <span id="authorLabel" data-l10n-id="pdfjs-document-properties-author">Author:</span>
  583. <p id="authorField" aria-labelledby="authorLabel">-</p>
  584. </div>
  585. <div class="row">
  586. <span id="subjectLabel" data-l10n-id="pdfjs-document-properties-subject">Subject:</span>
  587. <p id="subjectField" aria-labelledby="subjectLabel">-</p>
  588. </div>
  589. <div class="row">
  590. <span id="keywordsLabel" data-l10n-id="pdfjs-document-properties-keywords">Keywords:</span>
  591. <p id="keywordsField" aria-labelledby="keywordsLabel">-</p>
  592. </div>
  593. <div class="row">
  594. <span id="creationDateLabel" data-l10n-id="pdfjs-document-properties-creation-date">Creation Date:</span>
  595. <p id="creationDateField" aria-labelledby="creationDateLabel">-</p>
  596. </div>
  597. <div class="row">
  598. <span id="modificationDateLabel" data-l10n-id="pdfjs-document-properties-modification-date">Modification
  599. Date:</span>
  600. <p id="modificationDateField" aria-labelledby="modificationDateLabel">-</p>
  601. </div>
  602. <div class="row">
  603. <span id="creatorLabel" data-l10n-id="pdfjs-document-properties-creator">Creator:</span>
  604. <p id="creatorField" aria-labelledby="creatorLabel">-</p>
  605. </div>
  606. <div class="separator"></div>
  607. <div class="row">
  608. <span id="producerLabel" data-l10n-id="pdfjs-document-properties-producer">PDF Producer:</span>
  609. <p id="producerField" aria-labelledby="producerLabel">-</p>
  610. </div>
  611. <div class="row">
  612. <span id="versionLabel" data-l10n-id="pdfjs-document-properties-version">PDF Version:</span>
  613. <p id="versionField" aria-labelledby="versionLabel">-</p>
  614. </div>
  615. <div class="row">
  616. <span id="pageCountLabel" data-l10n-id="pdfjs-document-properties-page-count">Page Count:</span>
  617. <p id="pageCountField" aria-labelledby="pageCountLabel">-</p>
  618. </div>
  619. <div class="row">
  620. <span id="pageSizeLabel" data-l10n-id="pdfjs-document-properties-page-size">Page Size:</span>
  621. <p id="pageSizeField" aria-labelledby="pageSizeLabel">-</p>
  622. </div>
  623. <div class="separator"></div>
  624. <div class="row">
  625. <span id="linearizedLabel" data-l10n-id="pdfjs-document-properties-linearized">Fast Web View:</span>
  626. <p id="linearizedField" aria-labelledby="linearizedLabel">-</p>
  627. </div>
  628. <div class="buttonRow">
  629. <button id="documentPropertiesClose" class="dialogButton" type="button"><span
  630. data-l10n-id="pdfjs-document-properties-close-button">Close</span></button>
  631. </div>
  632. </dialog>
  633. <dialog class="dialog altText" id="altTextDialog" aria-labelledby="dialogLabel"
  634. aria-describedby="dialogDescription">
  635. <div id="altTextContainer" class="mainContainer">
  636. <div id="overallDescription">
  637. <span id="dialogLabel" data-l10n-id="pdfjs-editor-alt-text-dialog-label" class="title">Choose an
  638. option</span>
  639. <span id="dialogDescription" data-l10n-id="pdfjs-editor-alt-text-dialog-description">
  640. Alt text (alternative text) helps when people can’t see the image or when it doesn’t load.
  641. </span>
  642. </div>
  643. <div id="addDescription">
  644. <div class="radio">
  645. <div class="radioButton">
  646. <input type="radio" id="descriptionButton" name="altTextOption" tabindex="0"
  647. aria-describedby="descriptionAreaLabel" checked>
  648. <label for="descriptionButton" data-l10n-id="pdfjs-editor-alt-text-add-description-label">Add a
  649. description</label>
  650. </div>
  651. <div class="radioLabel">
  652. <span id="descriptionAreaLabel" data-l10n-id="pdfjs-editor-alt-text-add-description-description">
  653. Aim for 1-2 sentences that describe the subject, setting, or actions.
  654. </span>
  655. </div>
  656. </div>
  657. <div class="descriptionArea">
  658. <textarea id="descriptionTextarea"
  659. placeholder="For example, “A young man sits down at a table to eat a meal”"
  660. aria-labelledby="descriptionAreaLabel" data-l10n-id="pdfjs-editor-alt-text-textarea"
  661. tabindex="0"></textarea>
  662. </div>
  663. </div>
  664. <div id="markAsDecorative">
  665. <div class="radio">
  666. <div class="radioButton">
  667. <input type="radio" id="decorativeButton" name="altTextOption" aria-describedby="decorativeLabel">
  668. <label for="decorativeButton" data-l10n-id="pdfjs-editor-alt-text-mark-decorative-label">Mark as
  669. decorative</label>
  670. </div>
  671. <div class="radioLabel">
  672. <span id="decorativeLabel" data-l10n-id="pdfjs-editor-alt-text-mark-decorative-description">
  673. This is used for ornamental images, like borders or watermarks.
  674. </span>
  675. </div>
  676. </div>
  677. </div>
  678. <div id="buttons">
  679. <button id="altTextCancel" class="secondaryButton" type="button" tabindex="0"><span
  680. data-l10n-id="pdfjs-editor-alt-text-cancel-button">Cancel</span></button>
  681. <button id="altTextSave" class="primaryButton" type="button" tabindex="0"><span
  682. data-l10n-id="pdfjs-editor-alt-text-save-button">Save</span></button>
  683. </div>
  684. </div>
  685. </dialog>
  686. <dialog class="dialog newAltText" id="newAltTextDialog" aria-labelledby="newAltTextTitle"
  687. aria-describedby="newAltTextDescription" tabindex="0">
  688. <div id="newAltTextContainer" class="mainContainer">
  689. <div class="title">
  690. <span id="newAltTextTitle" data-l10n-id="pdfjs-editor-new-alt-text-dialog-edit-label" role="sectionhead"
  691. tabindex="0">Edit alt text (image description)</span>
  692. </div>
  693. <div id="mainContent">
  694. <div id="descriptionAndSettings">
  695. <div id="descriptionInstruction">
  696. <div id="newAltTextDescriptionContainer">
  697. <div class="altTextSpinner" role="status" aria-live="polite"></div>
  698. <textarea id="newAltTextDescriptionTextarea" placeholder="Write your description here…"
  699. aria-labelledby="descriptionAreaLabel" data-l10n-id="pdfjs-editor-new-alt-text-textarea"
  700. tabindex="0"></textarea>
  701. </div>
  702. <span id="newAltTextDescription" role="note" data-l10n-id="pdfjs-editor-new-alt-text-description">Short
  703. description for people who can’t see the image or when the image doesn’t load.</span>
  704. <div id="newAltTextDisclaimer" role="note">
  705. <div><span data-l10n-id="pdfjs-editor-new-alt-text-disclaimer1">This alt text was created
  706. automatically and may be inaccurate.</span> <a
  707. href="https://support.mozilla.org/en-US/kb/pdf-alt-text" target="_blank" rel="noopener noreferrer"
  708. id="newAltTextLearnMore" data-l10n-id="pdfjs-editor-new-alt-text-disclaimer-learn-more-url"
  709. tabindex="0">Learn more</a></div>
  710. </div>
  711. </div>
  712. <div id="newAltTextCreateAutomatically" class="toggler">
  713. <button id="newAltTextCreateAutomaticallyButton" class="toggle-button" type="button" aria-pressed="true"
  714. tabindex="0"></button>
  715. <label for="newAltTextCreateAutomaticallyButton" class="togglerLabel"
  716. data-l10n-id="pdfjs-editor-new-alt-text-create-automatically-button-label">Create alt text
  717. automatically</label>
  718. </div>
  719. <div id="newAltTextDownloadModel" class="hidden">
  720. <span id="newAltTextDownloadModelDescription"
  721. data-l10n-id="pdfjs-editor-new-alt-text-ai-model-downloading-progress" aria-valuemin="0"
  722. data-l10n-args='{ "totalSize": 0, "downloadedSize": 0 }'>Downloading alt text AI model (0 of 0
  723. MB)</span>
  724. </div>
  725. </div>
  726. <div id="newAltTextImagePreview"></div>
  727. </div>
  728. <div id="newAltTextError" class="messageBar">
  729. <div>
  730. <div>
  731. <span class="title" data-l10n-id="pdfjs-editor-new-alt-text-error-title">Couldn’t create alt text
  732. automatically</span>
  733. <span class="description" data-l10n-id="pdfjs-editor-new-alt-text-error-description">Please write your
  734. own alt text or try again later.</span>
  735. </div>
  736. <button id="newAltTextCloseButton" class="closeButton" type="button" tabindex="0" title="Close"><span
  737. data-l10n-id="pdfjs-editor-new-alt-text-error-close-button">Close</span></button>
  738. </div>
  739. </div>
  740. <div id="newAltTextButtons" class="dialogButtonsGroup">
  741. <button id="newAltTextCancel" type="button" class="secondaryButton hidden" tabindex="0"><span
  742. data-l10n-id="pdfjs-editor-alt-text-cancel-button">Cancel</span></button>
  743. <button id="newAltTextNotNow" type="button" class="secondaryButton" tabindex="0"><span
  744. data-l10n-id="pdfjs-editor-new-alt-text-not-now-button">Not now</span></button>
  745. <button id="newAltTextSave" type="button" class="primaryButton" tabindex="0"><span
  746. data-l10n-id="pdfjs-editor-alt-text-save-button">Save</span></button>
  747. </div>
  748. </div>
  749. </dialog>
  750. <dialog class="dialog" id="altTextSettingsDialog" aria-labelledby="altTextSettingsTitle">
  751. <div id="altTextSettingsContainer" class="mainContainer">
  752. <div class="title">
  753. <span id="altTextSettingsTitle" data-l10n-id="pdfjs-editor-alt-text-settings-dialog-label"
  754. role="sectionhead" tabindex="0" class="title">Image alt text settings</span>
  755. </div>
  756. <div id="automaticAltText">
  757. <span data-l10n-id="pdfjs-editor-alt-text-settings-automatic-title">Automatic alt text</span>
  758. <div id="automaticSettings">
  759. <div id="createModelSetting">
  760. <div class="toggler">
  761. <button id="createModelButton" type="button" class="toggle-button" aria-pressed="true"
  762. tabindex="0"></button>
  763. <label for="createModelButton" class="togglerLabel"
  764. data-l10n-id="pdfjs-editor-alt-text-settings-create-model-button-label">Create alt text
  765. automatically</label>
  766. </div>
  767. <div id="createModelDescription" class="description">
  768. <span data-l10n-id="pdfjs-editor-alt-text-settings-create-model-description">Suggests descriptions to
  769. help people who can’t see the image or when the image doesn’t load.</span> <a
  770. href="https://support.mozilla.org/en-US/kb/pdf-alt-text" target="_blank" rel="noopener noreferrer"
  771. id="altTextSettingsLearnMore" data-l10n-id="pdfjs-editor-new-alt-text-disclaimer-learn-more-url"
  772. tabindex="0">Learn more</a>
  773. </div>
  774. </div>
  775. <div id="aiModelSettings">
  776. <div>
  777. <span data-l10n-id="pdfjs-editor-alt-text-settings-download-model-label"
  778. data-l10n-args='{ "totalSize": 180 }'>Alt text AI model (180MB)</span>
  779. <div id="aiModelDescription" class="description">
  780. <span data-l10n-id="pdfjs-editor-alt-text-settings-ai-model-description">Runs locally on your device
  781. so your data stays private. Required for automatic alt text.</span>
  782. </div>
  783. </div>
  784. <button id="deleteModelButton" type="button" class="secondaryButton" tabindex="0"><span
  785. data-l10n-id="pdfjs-editor-alt-text-settings-delete-model-button">Delete</span></button>
  786. <button id="downloadModelButton" type="button" class="secondaryButton" tabindex="0"><span
  787. data-l10n-id="pdfjs-editor-alt-text-settings-download-model-button">Download</span></button>
  788. </div>
  789. </div>
  790. </div>
  791. <div class="dialogSeparator"></div>
  792. <div id="altTextEditor">
  793. <span data-l10n-id="pdfjs-editor-alt-text-settings-editor-title">Alt text editor</span>
  794. <div id="showAltTextEditor">
  795. <div class="toggler">
  796. <button id="showAltTextDialogButton" type="button" class="toggle-button" aria-pressed="true"
  797. tabindex="0"></button>
  798. <label for="showAltTextDialogButton" class="togglerLabel"
  799. data-l10n-id="pdfjs-editor-alt-text-settings-show-dialog-button-label">Show alt text editor right away
  800. when adding an image</label>
  801. </div>
  802. <div id="showAltTextDialogDescription" class="description">
  803. <span data-l10n-id="pdfjs-editor-alt-text-settings-show-dialog-description">Helps you make sure all your
  804. images have alt text.</span>
  805. </div>
  806. </div>
  807. </div>
  808. <div id="buttons" class="dialogButtonsGroup">
  809. <button id="altTextSettingsCloseButton" type="button" class="primaryButton" tabindex="0"><span
  810. data-l10n-id="pdfjs-editor-alt-text-settings-close-button">Close</span></button>
  811. </div>
  812. </div>
  813. </dialog>
  814. <dialog id="printServiceDialog" style="min-width: 200px;">
  815. <div class="row">
  816. <span data-l10n-id="pdfjs-print-progress-message">Preparing document for printing…</span>
  817. </div>
  818. <div class="row">
  819. <progress value="0" max="100"></progress>
  820. <span data-l10n-id="pdfjs-print-progress-percent" data-l10n-args='{ "progress": 0 }'
  821. class="relative-progress">0%</span>
  822. </div>
  823. <div class="buttonRow">
  824. <button id="printCancel" class="dialogButton" type="button"><span
  825. data-l10n-id="pdfjs-print-progress-close-button">Cancel</span></button>
  826. </div>
  827. </dialog>
  828. </div> <!-- dialogContainer -->
  829. </div> <!-- outerContainer -->
  830. <div id="printContainer"></div>
  831. <script>
  832. const uploadButton = document.getElementById('uploadButton');
  833. const pdfInput = document.getElementById('pdfInput');
  834. const filenameModal = document.getElementById('filenameModal');
  835. const modalOverlay = document.getElementById('modalOverlay');
  836. const confirmUploadButton = document.getElementById('confirmUpload');
  837. const cancelUploadButton = document.getElementById('cancelUpload');
  838. const customFileNameInput = document.getElementById('customFileName');
  839. const openDirectoryButton = document.getElementById('openDirectoryButton');
  840. const directoryModal = document.getElementById('directoryModal');
  841. const closeDirectoryButton = document.getElementById('closeDirectoryButton');
  842. const pdfList = document.getElementById('pdfList');
  843. let selectedFile = null;
  844. // 上传按钮点击事件
  845. uploadButton.addEventListener('click', function () {
  846. // 触发隐藏的文件输入
  847. pdfInput.click();
  848. });
  849. // 文件选择事件
  850. pdfInput.addEventListener('change', function (event) {
  851. const file = event.target.files[0];
  852. if (file && file.type === 'application/pdf') {
  853. selectedFile = file;
  854. // 显示模态窗口
  855. modalOverlay.style.display = 'block';
  856. filenameModal.style.display = 'block';
  857. customFileNameInput.value = ""; // 清空之前的输入
  858. customFileNameInput.focus();
  859. } else {
  860. alert('请选择一个 PDF 文件');
  861. }
  862. });
  863. // 取消上传
  864. cancelUploadButton.addEventListener('click', function () {
  865. // 隐藏模态窗口和遮罩
  866. filenameModal.style.display = 'none';
  867. modalOverlay.style.display = 'none';
  868. // 清空文件输入
  869. pdfInput.value = "";
  870. selectedFile = null;
  871. });
  872. // 确认上传
  873. confirmUploadButton.addEventListener('click', function () {
  874. const customFileName = customFileNameInput.value.trim();
  875. if (customFileName === "") {
  876. alert('请输入保存的文件名');
  877. customFileNameInput.focus();
  878. return;
  879. }
  880. // 检查文件名是否包含非法字符
  881. const invalidChars = /[\\/:"*?<>|]+/;
  882. if (invalidChars.test(customFileName)) {
  883. alert('文件名包含非法字符:\\ / : " * ? < > |');
  884. customFileNameInput.focus();
  885. return;
  886. }
  887. if (!selectedFile) {
  888. alert('没有选择文件');
  889. return;
  890. }
  891. // 进行上传
  892. const formData = new FormData();
  893. formData.append('file', selectedFile);
  894. formData.append('custom_name', customFileName); // 添加自定义文件名
  895. // 可以根据需要显示上传进度
  896. fetch('http://117.50.195.224:8005/upload-pdf', { // 根据实际部署情况调整URL
  897. method: 'POST',
  898. body: formData
  899. })
  900. .then(response => response.json())
  901. .then(data => {
  902. if (data.success) {
  903. const uploadedFilePath = data.file_path; // 如 "/static/files/XXXX.pdf"
  904. const redirectUrl = `viewer.html?file=${encodeURIComponent(uploadedFilePath)}`;
  905. window.location.href = redirectUrl;
  906. } else {
  907. alert('上传失败: ' + data.error);
  908. }
  909. })
  910. .catch(error => {
  911. console.error('Error:', error);
  912. alert('上传过程中发生错误');
  913. })
  914. .finally(() => {
  915. // 隐藏模态窗口和遮罩
  916. filenameModal.style.display = 'none';
  917. modalOverlay.style.display = 'none';
  918. // 清空文件输入
  919. pdfInput.value = "";
  920. selectedFile = null;
  921. });
  922. });
  923. // 处理“打开目录”按钮点击事件
  924. openDirectoryButton.addEventListener('click', function () {
  925. // 显示遮罩和弹出层
  926. modalOverlay.style.display = 'block';
  927. directoryModal.style.display = 'block';
  928. // 清空现有的列表
  929. pdfList.innerHTML = '';
  930. // 调用 FastAPI 获取文件列表
  931. fetch('http://117.50.195.224:8005/list-pdfs') // 根据实际部署情况调整URL
  932. .then(response => response.json())
  933. .then(data => {
  934. if (data.success) {
  935. const files = data.files;
  936. if (files.length === 0) {
  937. const listItem = document.createElement('li');
  938. listItem.textContent = '没有已上传的 PDF 文档。';
  939. pdfList.appendChild(listItem);
  940. } else {
  941. files.forEach(file => {
  942. const listItem = document.createElement('li');
  943. listItem.style.marginBottom = '10px';
  944. const link = document.createElement('a');
  945. link.href = `viewer.html?file=${file.url}`; // 确保 URL 正确
  946. link.textContent = file.name;
  947. link.style.textDecoration = 'none';
  948. link.style.color = '#007bff';
  949. // 添加点击事件,在当前页面打开
  950. link.addEventListener('click', function (event) {
  951. event.preventDefault(); // 防止默认的跳转行为
  952. const fileUrl = link.getAttribute('href');
  953. const redirectUrl = `${fileUrl}`;
  954. window.location.href = redirectUrl; // 在当前页面跳转
  955. });
  956. listItem.appendChild(link);
  957. pdfList.appendChild(listItem);
  958. });
  959. }
  960. } else {
  961. alert('无法获取文件列表: ' + data.error);
  962. }
  963. })
  964. .catch(error => {
  965. console.error('Error:', error);
  966. alert('获取文件列表时发生错误');
  967. });
  968. });
  969. // 处理“关闭”按钮点击事件
  970. closeDirectoryButton.addEventListener('click', function () {
  971. directoryModal.style.display = 'none';
  972. modalOverlay.style.display = 'none';
  973. });
  974. // 点击遮罩关闭弹出层
  975. modalOverlay.addEventListener('click', function () {
  976. if (filenameModal.style.display === 'block') {
  977. filenameModal.style.display = 'none';
  978. uploadButton.focus();
  979. }
  980. if (directoryModal.style.display === 'block') {
  981. directoryModal.style.display = 'none';
  982. }
  983. modalOverlay.style.display = 'none';
  984. });
  985. // 语音控制
  986. let isListening = false;
  987. // 点击按钮开关文本选择监听
  988. document.getElementById('toggle-text-select').addEventListener('click', () => {
  989. isListening = !isListening; // 切换状态
  990. const button = document.getElementById('toggle-text-select');
  991. button.textContent = isListening ? "关闭监听" : "开启监听";
  992. // 切换按钮的样式
  993. if (isListening) {
  994. button.classList.add('button-danger');
  995. } else {
  996. button.classList.remove('button-danger');
  997. }
  998. });
  999. document.addEventListener('DOMContentLoaded', () => {
  1000. const audioPlayer = document.getElementById('audio-player');
  1001. const loadingIndicator = document.getElementById('loading-indicator');
  1002. // 函数:发送选中的文本到后端进行 TTS 处理
  1003. function sendTextToTTS(selectedText) {
  1004. loadingIndicator.style.display = 'block';
  1005. fetch('http://117.50.195.224:8005/text-to-speech/', {
  1006. method: 'POST',
  1007. headers: {
  1008. 'Content-Type': 'application/json'
  1009. },
  1010. body: JSON.stringify({ user_input: selectedText })
  1011. })
  1012. .then(response => {
  1013. if (!response.ok) {
  1014. throw new Error('语音生成失败');
  1015. }
  1016. return response.blob();
  1017. })
  1018. .then(blob => {
  1019. const audioURL = URL.createObjectURL(blob);
  1020. audioPlayer.src = audioURL;
  1021. audioPlayer.play();
  1022. })
  1023. .catch(err => {
  1024. alert('生成语音失败');
  1025. console.error(err);
  1026. })
  1027. .finally(() => {
  1028. // 隐藏等待指示器
  1029. loadingIndicator.style.display = 'none';
  1030. });
  1031. }
  1032. // 监听文本选择事件
  1033. document.addEventListener('mouseup', () => {
  1034. if (!isListening) return; // 若未开启监听,直接返回
  1035. const selection = window.getSelection();
  1036. const selectedText = selection.toString().trim();
  1037. if (selectedText.length === 0) return;
  1038. // 清除选择
  1039. selection.removeAllRanges();
  1040. // 发送选中的文本到后端
  1041. sendTextToTTS(selectedText);
  1042. });
  1043. });
  1044. // 在其他 JavaScript 代码里保持不变
  1045. document.getElementById('select-all-text').addEventListener('click', function() {
  1046. const { pdfViewer } = PDFViewerApplication;
  1047. const currentPage = pdfViewer.currentPageNumber;
  1048. const audioPlayer = document.getElementById('audio-player');
  1049. const loadingIndicator = document.getElementById('loading-indicator');
  1050. // 函数:发送选中的文本到后端进行 TTS 处理
  1051. function sendPageToTTS(selectedText) {
  1052. loadingIndicator.style.display = 'block';
  1053. fetch('http://117.50.195.224:8005/text-to-speech/', {
  1054. method: 'POST',
  1055. headers: {
  1056. 'Content-Type': 'application/json'
  1057. },
  1058. body: JSON.stringify({ user_input: selectedText })
  1059. })
  1060. .then(response => {
  1061. if (!response.ok) {
  1062. throw new Error('语音生成失败');
  1063. }
  1064. return response.blob();
  1065. })
  1066. .then(blob => {
  1067. const audioURL = URL.createObjectURL(blob);
  1068. audioPlayer.src = audioURL;
  1069. audioPlayer.play();
  1070. })
  1071. .catch(err => {
  1072. alert('生成语音失败');
  1073. console.error(err);
  1074. })
  1075. .finally(() => {
  1076. // 隐藏等待指示器
  1077. loadingIndicator.style.display = 'none';
  1078. });
  1079. }
  1080. PDFViewerApplication.pdfDocument.getPage(currentPage).then(function(page) {
  1081. page.getTextContent().then(function(textContent) {
  1082. // 创建一个临时的、不可见的 div 来容纳文本
  1083. const tempDiv = document.createElement('div');
  1084. tempDiv.style.position = 'fixed';
  1085. tempDiv.style.top = '0';
  1086. tempDiv.style.left = '0';
  1087. tempDiv.style.width = '100%';
  1088. tempDiv.style.height = '100%';
  1089. tempDiv.style.opacity = '0';
  1090. tempDiv.style.pointerEvents = 'none';
  1091. tempDiv.contentEditable = true; // 使其可选中
  1092. // 将所有文本合并为一个字符串
  1093. const fullText = textContent.items.map(item => item.str).join(' ');
  1094. // alert(fullText);
  1095. sendPageToTTS(fullText);
  1096. });
  1097. });
  1098. });
  1099. </script>
  1100. </body>
  1101. </html>