views.go 59 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385
  1. // Code generated by go generate; DO NOT EDIT.
  2. package template // import "miniflux.app/template"
  3. var templateViewsMap = map[string]string{
  4. "about": `{{ define "title"}}{{ t "page.about.title" }}{{ end }}
  5. {{ define "content"}}
  6. <section class="page-header">
  7. <h1>{{ t "page.about.title" }}</h1>
  8. {{ template "settings_menu" dict "user" .user }}
  9. </section>
  10. <div class="panel">
  11. <h3>Miniflux</h3>
  12. <ul>
  13. <li><strong>{{ t "page.about.version" }}</strong> {{ .version }}</li>
  14. <li><strong>{{ t "page.about.build_date" }}</strong> {{ .build_date }}</li>
  15. </ul>
  16. </div>
  17. <div class="panel">
  18. <h3>{{ t "page.about.credits" }}</h3>
  19. <ul>
  20. <li><strong>{{ t "page.about.author" }}</strong> Frédéric Guillot</li>
  21. <li><strong>{{ t "page.about.license" }}</strong> Apache 2.0</li>
  22. </ul>
  23. </div>
  24. {{ end }}
  25. `,
  26. "add_subscription": `{{ define "title"}}{{ t "page.add_feed.title" }}{{ end }}
  27. {{ define "content"}}
  28. <section class="page-header">
  29. <h1>{{ t "page.add_feed.title" }}</h1>
  30. <ul>
  31. <li>
  32. <a href="{{ route "feeds" }}">{{ t "menu.feeds" }}</a>
  33. </li>
  34. <li>
  35. <a href="{{ route "export" }}">{{ t "menu.export" }}</a>
  36. </li>
  37. <li>
  38. <a href="{{ route "import" }}">{{ t "menu.import" }}</a>
  39. </li>
  40. </ul>
  41. </section>
  42. {{ if not .categories }}
  43. <p class="alert alert-error">{{ t "page.add_feed.no_category" }}</p>
  44. {{ else }}
  45. <form action="{{ route "submitSubscription" }}" method="post" autocomplete="off">
  46. <input type="hidden" name="csrf" value="{{ .csrf }}">
  47. {{ if .errorMessage }}
  48. <div class="alert alert-error">{{ t .errorMessage }}</div>
  49. {{ end }}
  50. <label for="form-url">{{ t "page.add_feed.label.url" }}</label>
  51. <input type="url" name="url" id="form-url" placeholder="https://domain.tld/" value="{{ .form.URL }}" required autofocus>
  52. <label for="form-category">{{ t "form.feed.label.category" }}</label>
  53. <select id="form-category" name="category_id">
  54. {{ range .categories }}
  55. <option value="{{ .ID }}" {{ if eq $.form.CategoryID .ID }}selected="selected"{{ end }}>{{ .Title }}</option>
  56. {{ end }}
  57. </select>
  58. <details>
  59. <summary>{{ t "page.add_feed.legend.advanced_options" }}</summary>
  60. <div class="details-content">
  61. <label><input type="checkbox" name="crawler" value="1" {{ if .form.Crawler }}checked{{ end }}> {{ t "form.feed.label.crawler" }}</label>
  62. <label for="form-user-agent">{{ t "form.feed.label.user_agent" }}</label>
  63. <input type="text" name="user_agent" id="form-user-agent" placeholder="{{ .defaultUserAgent }}" value="{{ .form.UserAgent }}" autocomplete="off">
  64. <label for="form-feed-username">{{ t "form.feed.label.feed_username" }}</label>
  65. <input type="text" name="feed_username" id="form-feed-username" value="{{ .form.Username }}">
  66. <label for="form-feed-password">{{ t "form.feed.label.feed_password" }}</label>
  67. <!--
  68. We are using the type "text" otherwise Firefox always autocomplete this password:
  69. - autocomplete="off" or autocomplete="new-password" doesn't change anything
  70. - Changing the input ID doesn't change anything
  71. - Using a different input name doesn't change anything
  72. -->
  73. <input type="text" name="feed_password" id="form-feed-password" value="{{ .form.Password }}">
  74. </div>
  75. </details>
  76. <div class="buttons">
  77. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.loading" }}">{{ t "page.add_feed.submit" }}</button>
  78. </div>
  79. </form>
  80. {{ end }}
  81. {{ end }}
  82. `,
  83. "bookmark_entries": `{{ define "title"}}{{ t "page.starred.title" }} ({{ .total }}){{ end }}
  84. {{ define "content"}}
  85. <section class="page-header">
  86. <h1>{{ t "page.starred.title" }} ({{ .total }})</h1>
  87. </section>
  88. {{ if not .entries }}
  89. <p class="alert alert-info">{{ t "alert.no_bookmark" }}</p>
  90. {{ else }}
  91. <div class="items">
  92. {{ range .entries }}
  93. <article class="item touch-item item-status-{{ .Status }}" data-id="{{ .ID }}">
  94. <div class="item-header">
  95. <span class="item-title">
  96. {{ if ne .Feed.Icon.IconID 0 }}
  97. <img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="{{ .Feed.Title }}">
  98. {{ end }}
  99. <a href="{{ route "starredEntry" "entryID" .ID }}">{{ .Title }}</a>
  100. </span>
  101. <span class="category"><a href="{{ route "categoryEntries" "categoryID" .Feed.Category.ID }}">{{ .Feed.Category.Title }}</a></span>
  102. </div>
  103. {{ template "item_meta" dict "user" $.user "entry" . "hasSaveEntry" $.hasSaveEntry }}
  104. </article>
  105. {{ end }}
  106. </div>
  107. {{ template "pagination" .pagination }}
  108. {{ end }}
  109. {{ end }}
  110. `,
  111. "categories": `{{ define "title"}}{{ t "page.categories.title" }} ({{ .total }}){{ end }}
  112. {{ define "content"}}
  113. <section class="page-header">
  114. <h1>{{ t "page.categories.title" }} ({{ .total }})</h1>
  115. <ul>
  116. <li>
  117. <a href="{{ route "createCategory" }}">{{ t "menu.create_category" }}</a>
  118. </li>
  119. </ul>
  120. </section>
  121. {{ if not .categories }}
  122. <p class="alert alert-error">{{ t "alert.no_category" }}</p>
  123. {{ else }}
  124. <div class="items">
  125. {{ range .categories }}
  126. <article class="item">
  127. <div class="item-header">
  128. <span class="item-title">
  129. <a href="{{ route "categoryEntries" "categoryID" .ID }}">{{ .Title }}</a>
  130. </span>
  131. </div>
  132. <div class="item-meta">
  133. <ul>
  134. <li>
  135. {{ if eq .FeedCount 0 }}
  136. {{ t "page.categories.no_feed" }}
  137. {{ else }}
  138. {{ plural "page.categories.feed_count" .FeedCount .FeedCount }}
  139. {{ end }}
  140. </li>
  141. </ul>
  142. <ul>
  143. <li>
  144. <a href="{{ route "editCategory" "categoryID" .ID }}">{{ t "menu.edit_category" }}</a>
  145. </li>
  146. {{ if eq .FeedCount 0 }}
  147. <li>
  148. <a href="#"
  149. data-confirm="true"
  150. data-label-question="{{ t "confirm.question" }}"
  151. data-label-yes="{{ t "confirm.yes" }}"
  152. data-label-no="{{ t "confirm.no" }}"
  153. data-label-loading="{{ t "confirm.loading" }}"
  154. data-url="{{ route "removeCategory" "categoryID" .ID }}">{{ t "action.remove" }}</a>
  155. </li>
  156. {{ end }}
  157. </ul>
  158. </div>
  159. </article>
  160. {{ end }}
  161. </div>
  162. {{ end }}
  163. {{ end }}
  164. `,
  165. "category_entries": `{{ define "title"}}{{ .category.Title }} ({{ .total }}){{ end }}
  166. {{ define "content"}}
  167. <section class="page-header">
  168. <h1>{{ .category.Title }} ({{ .total }})</h1>
  169. <ul>
  170. {{ if .entries }}
  171. <li>
  172. <a href="#"
  173. data-action="markPageAsRead"
  174. data-label-question="{{ t "confirm.question" }}"
  175. data-label-yes="{{ t "confirm.yes" }}"
  176. data-label-no="{{ t "confirm.no" }}"
  177. data-label-loading="{{ t "confirm.loading" }}"
  178. data-show-only-unread="{{ if .showOnlyUnreadEntries }}1{{ end }}">{{ t "menu.mark_page_as_read" }}</a>
  179. </li>
  180. {{ end }}
  181. {{ if .showOnlyUnreadEntries }}
  182. <li>
  183. <a href="{{ route "categoryEntriesAll" "categoryID" .category.ID }}">{{ t "menu.show_all_entries" }}</a>
  184. </li>
  185. {{ else }}
  186. <li>
  187. <a href="{{ route "categoryEntries" "categoryID" .category.ID }}">{{ t "menu.show_only_unread_entries" }}</a>
  188. </li>
  189. {{ end }}
  190. </ul>
  191. </section>
  192. {{ if not .entries }}
  193. <p class="alert">{{ t "alert.no_category_entry" }}</p>
  194. {{ else }}
  195. <div class="items">
  196. {{ range .entries }}
  197. <article class="item touch-item item-status-{{ .Status }}" data-id="{{ .ID }}">
  198. <div class="item-header">
  199. <span class="item-title">
  200. {{ if ne .Feed.Icon.IconID 0 }}
  201. <img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="{{ .Feed.Title }}">
  202. {{ end }}
  203. <a href="{{ route "categoryEntry" "categoryID" .Feed.Category.ID "entryID" .ID }}">{{ .Title }}</a>
  204. </span>
  205. <span class="category"><a href="{{ route "categoryEntries" "categoryID" .Feed.Category.ID }}">{{ .Feed.Category.Title }}</a></span>
  206. </div>
  207. {{ template "item_meta" dict "user" $.user "entry" . "hasSaveEntry" $.hasSaveEntry }}
  208. </article>
  209. {{ end }}
  210. </div>
  211. <section class="page-footer">
  212. {{ if .entries }}
  213. <ul>
  214. <li>
  215. <a href="#"
  216. data-action="markPageAsRead"
  217. data-label-question="{{ t "confirm.question" }}"
  218. data-label-yes="{{ t "confirm.yes" }}"
  219. data-label-no="{{ t "confirm.no" }}"
  220. data-label-loading="{{ t "confirm.loading" }}"
  221. data-show-only-unread="{{ if .showOnlyUnreadEntries }}1{{ end }}">{{ t "menu.mark_page_as_read" }}</a>
  222. </li>
  223. </ul>
  224. {{ end }}
  225. </section>
  226. {{ template "pagination" .pagination }}
  227. {{ end }}
  228. {{ end }}
  229. `,
  230. "choose_subscription": `{{ define "title"}}{{ t "page.add_feed.title" }}{{ end }}
  231. {{ define "content"}}
  232. <section class="page-header">
  233. <h1>{{ t "page.add_feed.title" }}</h1>
  234. <ul>
  235. <li>
  236. <a href="{{ route "feeds" }}">{{ t "menu.feeds" }}</a>
  237. </li>
  238. <li>
  239. <a href="{{ route "export" }}">{{ t "menu.export" }}</a>
  240. </li>
  241. <li>
  242. <a href="{{ route "import" }}">{{ t "menu.import" }}</a>
  243. </li>
  244. </ul>
  245. </section>
  246. <form action="{{ route "chooseSubscription" }}" method="POST">
  247. <input type="hidden" name="csrf" value="{{ .csrf }}">
  248. <input type="hidden" name="category_id" value="{{ .form.CategoryID }}">
  249. <input type="hidden" name="user_agent" value="{{ .form.UserAgent }}">
  250. <input type="hidden" name="feed_username" value="{{ .form.Username }}">
  251. <input type="hidden" name="feed_password" value="{{ .form.Password }}">
  252. {{ if .form.Crawler }}
  253. <input type="hidden" name="crawler" value="1">
  254. {{ end }}
  255. <h3>{{ t "page.add_feed.choose_feed" }}</h3>
  256. {{ range .subscriptions }}
  257. <div class="radio-group">
  258. <label title="{{ .URL }}"><input type="radio" name="url" value="{{ .URL }}"> {{ .Title }}</label> ({{ .Type }})
  259. <small title="Type = {{ .Type }}"><a href="{{ .URL }}" target="_blank" rel="noopener noreferrer" referrerpolicy="no-referrer">{{ .URL }}</a></small>
  260. </div>
  261. {{ end }}
  262. <div class="buttons">
  263. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.loading" }}">{{ t "action.subscribe" }}</button>
  264. </div>
  265. </form>
  266. {{ end }}
  267. `,
  268. "create_category": `{{ define "title"}}{{ t "page.new_category.title" }}{{ end }}
  269. {{ define "content"}}
  270. <section class="page-header">
  271. <h1>{{ t "page.new_category.title" }}</h1>
  272. <ul>
  273. <li>
  274. <a href="{{ route "categories" }}">{{ t "menu.categories" }}</a>
  275. </li>
  276. </ul>
  277. </section>
  278. <form action="{{ route "saveCategory" }}" method="post" autocomplete="off">
  279. <input type="hidden" name="csrf" value="{{ .csrf }}">
  280. {{ if .errorMessage }}
  281. <div class="alert alert-error">{{ t .errorMessage }}</div>
  282. {{ end }}
  283. <label for="form-title">{{ t "form.category.label.title" }}</label>
  284. <input type="text" name="title" id="form-title" value="{{ .form.Title }}" required autofocus>
  285. <div class="buttons">
  286. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.saving" }}">{{ t "action.save" }}</button> {{ t "action.or" }} <a href="{{ route "categories" }}">{{ t "action.cancel" }}</a>
  287. </div>
  288. </form>
  289. {{ end }}
  290. `,
  291. "create_user": `{{ define "title"}}{{ t "page.new_user.title" }}{{ end }}
  292. {{ define "content"}}
  293. <section class="page-header">
  294. <h1>{{ t "page.new_user.title" }}</h1>
  295. {{ template "settings_menu" dict "user" .user }}
  296. </section>
  297. <form action="{{ route "saveUser" }}" method="post" autocomplete="off">
  298. <input type="hidden" name="csrf" value="{{ .csrf }}">
  299. {{ if .errorMessage }}
  300. <div class="alert alert-error">{{ t .errorMessage }}</div>
  301. {{ end }}
  302. <label for="form-username">{{ t "form.user.label.username" }}</label>
  303. <input type="text" name="username" id="form-username" value="{{ .form.Username }}" autocomplete="new-password" required autofocus>
  304. <label for="form-password">{{ t "form.user.label.password" }}</label>
  305. <input type="password" name="password" id="form-password" value="{{ .form.Password }}" autocomplete="new-password" required>
  306. <label for="form-confirmation">{{ t "form.user.label.confirmation" }}</label>
  307. <input type="password" name="confirmation" id="form-confirmation" value="{{ .form.Confirmation }}" required>
  308. <label><input type="checkbox" name="is_admin" value="1" {{ if .form.IsAdmin }}checked{{ end }}> {{ t "form.user.label.admin" }}</label>
  309. <div class="buttons">
  310. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.saving" }}">{{ t "action.save" }}</button> {{ t "action.or" }} <a href="{{ route "users" }}">{{ t "action.cancel" }}</a>
  311. </div>
  312. </form>
  313. {{ end }}
  314. `,
  315. "edit_category": `{{ define "title"}}{{ t "page.edit_category.title" .category.Title }}{{ end }}
  316. {{ define "content"}}
  317. <section class="page-header">
  318. <h1>{{ t "page.edit_category.title" .category.Title }}</h1>
  319. <ul>
  320. <li>
  321. <a href="{{ route "categories" }}">{{ t "menu.categories" }}</a>
  322. </li>
  323. <li>
  324. <a href="{{ route "createCategory" }}">{{ t "menu.create_category" }}</a>
  325. </li>
  326. </ul>
  327. </section>
  328. <form action="{{ route "updateCategory" "categoryID" .category.ID }}" method="post" autocomplete="off">
  329. <input type="hidden" name="csrf" value="{{ .csrf }}">
  330. {{ if .errorMessage }}
  331. <div class="alert alert-error">{{ t .errorMessage }}</div>
  332. {{ end }}
  333. <label for="form-title">{{ t "form.category.label.title" }}</label>
  334. <input type="text" name="title" id="form-title" value="{{ .form.Title }}" required autofocus>
  335. <div class="buttons">
  336. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.saving" }}">{{ t "action.update" }}</button> {{ t "action.or" }} <a href="{{ route "categories" }}">{{ t "action.cancel" }}</a>
  337. </div>
  338. </form>
  339. {{ end }}
  340. `,
  341. "edit_feed": `{{ define "title"}}{{ t "page.edit_feed.title" .feed.Title }}{{ end }}
  342. {{ define "content"}}
  343. <section class="page-header">
  344. <h1>{{ .feed.Title }}</h1>
  345. <ul>
  346. <li>
  347. <a href="{{ route "feeds" }}">{{ t "menu.feeds" }}</a>
  348. </li>
  349. <li>
  350. <a href="{{ route "addSubscription" }}">{{ t "menu.add_feed" }}</a>
  351. </li>
  352. <li>
  353. <a href="{{ route "export" }}">{{ t "menu.export" }}</a>
  354. </li>
  355. <li>
  356. <a href="{{ route "import" }}">{{ t "menu.import" }}</a>
  357. </li>
  358. </ul>
  359. </section>
  360. {{ if not .categories }}
  361. <p class="alert alert-error">{{ t "page.add_feed.no_category" }}</p>
  362. {{ else }}
  363. {{ if ne .feed.ParsingErrorCount 0 }}
  364. <div class="alert alert-error">
  365. <h3>{{ t "page.edit_feed.last_parsing_error" }}</h3>
  366. <p>{{ t .feed.ParsingErrorMsg }}</p>
  367. </div>
  368. {{ end }}
  369. <form action="{{ route "updateFeed" "feedID" .feed.ID }}" method="post" autocomplete="off">
  370. <input type="hidden" name="csrf" value="{{ .csrf }}">
  371. {{ if .errorMessage }}
  372. <div class="alert alert-error">{{ t .errorMessage }}</div>
  373. {{ end }}
  374. <label for="form-title">{{ t "form.feed.label.title" }}</label>
  375. <input type="text" name="title" id="form-title" value="{{ .form.Title }}" required autofocus>
  376. <label for="form-site-url">{{ t "form.feed.label.site_url" }}</label>
  377. <input type="url" name="site_url" id="form-site-url" placeholder="https://domain.tld/" value="{{ .form.SiteURL }}" required>
  378. <label for="form-feed-url">{{ t "form.feed.label.feed_url" }}</label>
  379. <input type="url" name="feed_url" id="form-feed-url" placeholder="https://domain.tld/" value="{{ .form.FeedURL }}" required>
  380. <label for="form-feed-username">{{ t "form.feed.label.feed_username" }}</label>
  381. <input type="text" name="feed_username" id="form-feed-username" value="{{ .form.Username }}">
  382. <label for="form-feed-password">{{ t "form.feed.label.feed_password" }}</label>
  383. <!--
  384. We are using the type "text" otherwise Firefox always autocomplete this password:
  385. - autocomplete="off" or autocomplete="new-password" doesn't change anything
  386. - Changing the input ID doesn't change anything
  387. - Using a different input name doesn't change anything
  388. -->
  389. <input type="text" name="feed_password" id="form-feed-password" value="{{ .form.Password }}">
  390. <label for="form-user-agent">{{ t "form.feed.label.user_agent" }}</label>
  391. <input type="text" name="user_agent" id="form-user-agent" placeholder="{{ .defaultUserAgent }}" value="{{ .form.UserAgent }}">
  392. <label for="form-scraper-rules">{{ t "form.feed.label.scraper_rules" }}</label>
  393. <input type="text" name="scraper_rules" id="form-scraper-rules" value="{{ .form.ScraperRules }}">
  394. <label for="form-rewrite-rules">{{ t "form.feed.label.rewrite_rules" }}</label>
  395. <input type="text" name="rewrite_rules" id="form-rewrite-rules" value="{{ .form.RewriteRules }}">
  396. <label for="form-category">{{ t "form.feed.label.category" }}</label>
  397. <select id="form-category" name="category_id">
  398. {{ range .categories }}
  399. <option value="{{ .ID }}" {{ if eq .ID $.form.CategoryID }}selected="selected"{{ end }}>{{ .Title }}</option>
  400. {{ end }}
  401. </select>
  402. <label><input type="checkbox" name="crawler" value="1" {{ if .form.Crawler }}checked{{ end }}> {{ t "form.feed.label.crawler" }}</label>
  403. <label><input type="checkbox" name="disabled" value="1" {{ if .form.Disabled }}checked{{ end }}> {{ t "form.feed.label.disabled" }}</label>
  404. <div class="buttons">
  405. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.saving" }}">{{ t "action.update" }}</button> {{ t "action.or" }} <a href="{{ route "feeds" }}">{{ t "action.cancel" }}</a>
  406. </div>
  407. </form>
  408. <div class="panel">
  409. <ul>
  410. <li><strong>{{ t "page.edit_feed.last_check" }} </strong><time datetime="{{ isodate .feed.CheckedAt }}" title="{{ isodate .feed.CheckedAt }}">{{ elapsed $.user.Timezone .feed.CheckedAt }}</time></li>
  411. <li><strong>{{ t "page.edit_feed.etag_header" }} </strong>{{ if .feed.EtagHeader }}{{ .feed.EtagHeader }}{{ else }}{{ t "page.edit_feed.no_header" }}{{ end }}</li>
  412. <li><strong>{{ t "page.edit_feed.last_modified_header" }} </strong>{{ if .feed.LastModifiedHeader }}{{ .feed.LastModifiedHeader }}{{ else }}{{ t "page.edit_feed.no_header" }}{{ end }}</li>
  413. </ul>
  414. </div>
  415. <div class="alert alert-error">
  416. <a href="#"
  417. data-confirm="true"
  418. data-action="remove-feed"
  419. data-label-question="{{ t "confirm.question" }}"
  420. data-label-yes="{{ t "confirm.yes" }}"
  421. data-label-no="{{ t "confirm.no" }}"
  422. data-label-loading="{{ t "confirm.loading" }}"
  423. data-url="{{ route "removeFeed" "feedID" .feed.ID }}"
  424. data-redirect-url="{{ route "feeds" }}">{{ t "action.remove_feed" }}</a>
  425. </div>
  426. {{ end }}
  427. {{ end }}
  428. `,
  429. "edit_user": `{{ define "title"}}{{ t "page.edit_user.title" .selected_user.Username }}{{ end }}
  430. {{ define "content"}}
  431. <section class="page-header">
  432. <h1>{{ t "page.edit_user.title" .selected_user.Username }}</h1>
  433. {{ template "settings_menu" dict "user" .user }}
  434. </section>
  435. <form action="{{ route "updateUser" "userID" .selected_user.ID }}" method="post" autocomplete="off">
  436. <input type="hidden" name="csrf" value="{{ .csrf }}">
  437. {{ if .errorMessage }}
  438. <div class="alert alert-error">{{ t .errorMessage }}</div>
  439. {{ end }}
  440. <label for="form-username">{{ t "form.user.label.username" }}</label>
  441. <input type="text" name="username" id="form-username" value="{{ .form.Username }}" autocomplete="new-password" required autofocus>
  442. <label for="form-password">{{ t "form.user.label.password" }}</label>
  443. <input type="password" name="password" id="form-password" value="{{ .form.Password }}" autocomplete="new-password">
  444. <label for="form-confirmation">{{ t "form.user.label.confirmation" }}</label>
  445. <input type="password" name="confirmation" id="form-confirmation" value="{{ .form.Confirmation }}" autocomplete="new-password">
  446. <label><input type="checkbox" name="is_admin" value="1" {{ if .form.IsAdmin }}checked{{ end }}> {{ t "form.user.label.admin" }}</label>
  447. <div class="buttons">
  448. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.saving" }}">{{ t "action.update" }}</button> {{ t "action.or" }} <a href="{{ route "users" }}">{{ t "action.cancel" }}</a>
  449. </div>
  450. </form>
  451. {{ end }}
  452. `,
  453. "entry": `{{ define "title"}}{{ .entry.Title }}{{ end }}
  454. {{ define "content"}}
  455. <section class="entry" data-id="{{ .entry.ID }}">
  456. <header class="entry-header">
  457. <h1>
  458. <a href="{{ .entry.URL }}" target="_blank" rel="noopener noreferrer" referrerpolicy="no-referrer">{{ .entry.Title }}</a>
  459. </h1>
  460. <div class="entry-actions">
  461. <ul>
  462. <li>
  463. <a href="#"
  464. title="{{ t "entry.status.title" }}"
  465. data-toggle-status="true"
  466. data-label-read="✔&#xfe0e;&nbsp;{{ t "entry.status.read" }}"
  467. data-label-unread="✘&nbsp;{{ t "entry.status.unread" }}"
  468. data-toast-read="✔︎&nbsp;{{ t "entry.status.toast.read" }}"
  469. data-toast-unread="✘&nbsp;{{ t "entry.status.toast.unread" }}"
  470. data-value="{{ if eq .entry.Status "read" }}read{{ else }}unread{{ end }}"
  471. >{{ if eq .entry.Status "read" }}✘&nbsp;{{ t "entry.status.unread" }}{{ else }}✔&#xfe0e;&nbsp;{{ t "entry.status.read" }}{{ end }}</a>
  472. </li>
  473. <li>
  474. <a href="#"
  475. data-toggle-bookmark="true"
  476. data-bookmark-url="{{ route "toggleBookmark" "entryID" .entry.ID }}"
  477. data-label-loading="{{ t "entry.state.saving" }}"
  478. data-label-star="☆&nbsp;{{ t "entry.bookmark.toggle.on" }}"
  479. data-label-unstar="★&nbsp;{{ t "entry.bookmark.toggle.off" }}"
  480. data-toast-star="★&nbsp;{{ t "entry.bookmark.toast.on" }}"
  481. data-toast-unstar="☆&nbsp;{{ t "entry.bookmark.toast.off" }}"
  482. data-value="{{ if .entry.Starred }}star{{ else }}unstar{{ end }}"
  483. >{{ if .entry.Starred }}★&nbsp;{{ t "entry.bookmark.toggle.off" }}{{ else }}☆&nbsp;{{ t "entry.bookmark.toggle.on" }}{{ end }}</a>
  484. </li>
  485. {{ if .hasSaveEntry }}
  486. <li>
  487. <a href="#"
  488. title="{{ t "entry.save.title" }}"
  489. data-save-entry="true"
  490. data-save-url="{{ route "saveEntry" "entryID" .entry.ID }}"
  491. data-label-loading="{{ t "entry.state.saving" }}"
  492. data-label-done="{{ t "entry.save.completed" }}"
  493. data-toast-done="{{ t "entry.save.toast.completed" }}"
  494. >{{ t "entry.save.title" }}</a>
  495. </li>
  496. {{ end }}
  497. <li>
  498. <a href="#"
  499. title="{{ t "entry.scraper.title" }}"
  500. data-fetch-content-entry="true"
  501. data-fetch-content-url="{{ route "fetchContent" "entryID" .entry.ID }}"
  502. data-label-loading="{{ t "entry.state.loading" }}"
  503. data-label-done="{{ t "entry.scraper.completed" }}"
  504. >{{ t "entry.scraper.label" }}</a>
  505. </li>
  506. {{ if .entry.CommentsURL }}
  507. <li>
  508. <a href="{{ .entry.CommentsURL }}" title="{{ t "entry.comments.title" }}" target="_blank" rel="noopener noreferrer" referrerpolicy="no-referrer">{{ t "entry.comments.label" }}</a>
  509. </li>
  510. {{ end }}
  511. </ul>
  512. </div>
  513. <div class="entry-meta">
  514. <span class="entry-website">
  515. {{ if ne .entry.Feed.Icon.IconID 0 }}
  516. <img src="{{ route "icon" "iconID" .entry.Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="{{ .entry.Feed.Title }}">
  517. {{ end }}
  518. <a href="{{ route "feedEntries" "feedID" .entry.Feed.ID }}">{{ .entry.Feed.Title }}</a>
  519. </span>
  520. {{ if .entry.Author }}
  521. <span class="entry-author">
  522. {{ if isEmail .entry.Author }}
  523. - <a href="mailto:{{ .entry.Author }}">{{ .entry.Author }}</a>
  524. {{ else }}
  525. – <em>{{ .entry.Author }}</em>
  526. {{ end }}
  527. </span>
  528. {{ end }}
  529. <span class="category">
  530. <a href="{{ route "categoryEntries" "categoryID" .entry.Feed.Category.ID }}">{{ .entry.Feed.Category.Title }}</a>
  531. </span>
  532. </div>
  533. <div class="entry-date">
  534. <time datetime="{{ isodate .entry.Date }}" title="{{ isodate .entry.Date }}">{{ elapsed $.user.Timezone .entry.Date }}</time>
  535. </div>
  536. </header>
  537. {{ if gt (len .entry.Content) 120 }}
  538. <div class="pagination-top">
  539. {{ template "entry_pagination" . }}
  540. </div>
  541. {{ end }}
  542. <article class="entry-content">
  543. {{ noescape (proxyFilter .entry.Content) }}
  544. </article>
  545. {{ if .entry.Enclosures }}
  546. <aside class="entry-enclosures">
  547. <h3>{{ t "page.entry.attachments" }}</h3>
  548. {{ range .entry.Enclosures }}
  549. <div class="entry-enclosure">
  550. {{ if hasPrefix .MimeType "audio/" }}
  551. <div class="enclosure-audio">
  552. <audio controls preload="metadata">
  553. <source src="{{ .URL }}" type="{{ .MimeType }}">
  554. </audio>
  555. </div>
  556. {{ else if hasPrefix .MimeType "video/" }}
  557. <div class="enclosure-video">
  558. <video controls preload="metadata">
  559. <source src="{{ .URL }}" type="{{ .MimeType }}">
  560. </video>
  561. </div>
  562. {{ else if hasPrefix .MimeType "image/" }}
  563. <div class="enclosure-image">
  564. <img src="{{ proxyURL .URL }}" title="{{ .URL }} ({{ .MimeType }})" loading="lazy" alt="{{ .URL }} ({{ .MimeType }})">
  565. </div>
  566. {{ end }}
  567. <div class="entry-enclosure-download">
  568. <a href="{{ .URL }}" title="{{ .URL }} ({{ .MimeType }})" target="_blank" rel="noopener noreferrer" referrerpolicy="no-referrer">{{ t "action.download" }}</a>
  569. <small>({{ .URL }})</small>
  570. </div>
  571. </div>
  572. {{ end }}
  573. </aside>
  574. {{ end }}
  575. </section>
  576. <div class="pagination-bottom">
  577. {{ template "entry_pagination" . }}
  578. </div>
  579. {{ end }}
  580. `,
  581. "feed_entries": `{{ define "title"}}{{ .feed.Title }} ({{ .total }}){{ end }}
  582. {{ define "content"}}
  583. <section class="page-header">
  584. <h1>{{ .feed.Title }} ({{ .total }})</h1>
  585. <ul>
  586. {{ if .entries }}
  587. <li>
  588. <a href="#"
  589. data-action="markPageAsRead"
  590. data-label-question="{{ t "confirm.question" }}"
  591. data-label-yes="{{ t "confirm.yes" }}"
  592. data-label-no="{{ t "confirm.no" }}"
  593. data-label-loading="{{ t "confirm.loading" }}"
  594. data-show-only-unread="{{ if .showOnlyUnreadEntries }}1{{ end }}">{{ t "menu.mark_page_as_read" }}</a>
  595. </li>
  596. {{ end }}
  597. {{ if .showOnlyUnreadEntries }}
  598. <li>
  599. <a href="{{ route "feedEntriesAll" "feedID" .feed.ID }}">{{ t "menu.show_all_entries" }}</a>
  600. </li>
  601. {{ else }}
  602. <li>
  603. <a href="{{ route "feedEntries" "feedID" .feed.ID }}">{{ t "menu.show_only_unread_entries" }}</a>
  604. </li>
  605. {{ end }}
  606. <li>
  607. <a href="{{ route "refreshFeed" "feedID" .feed.ID }}">{{ t "menu.refresh_feed" }}</a>
  608. </li>
  609. <li>
  610. <a href="{{ route "editFeed" "feedID" .feed.ID }}">{{ t "menu.edit_feed" }}</a>
  611. </li>
  612. <li>
  613. <a href="#"
  614. data-confirm="true"
  615. data-action="remove-feed"
  616. data-label-question="{{ t "confirm.question" }}"
  617. data-label-yes="{{ t "confirm.yes" }}"
  618. data-label-no="{{ t "confirm.no" }}"
  619. data-label-loading="{{ t "confirm.loading" }}"
  620. data-url="{{ route "removeFeed" "feedID" .feed.ID }}"
  621. data-redirect-url="{{ route "feeds" }}">{{ t "action.remove_feed" }}</a>
  622. </li>
  623. </ul>
  624. </section>
  625. {{ if ne .feed.ParsingErrorCount 0 }}
  626. <div class="alert alert-error">
  627. <h3>{{ t "alert.feed_error" }}</h3>
  628. <p>{{ t .feed.ParsingErrorMsg }}</p>
  629. </div>
  630. {{ end }}
  631. {{ if not .entries }}
  632. {{ if .showOnlyUnreadEntries }}
  633. <p class="alert">{{ t "alert.no_unread_entry" }}</p>
  634. {{ else }}
  635. <p class="alert">{{ t "alert.no_feed_entry" }}</p>
  636. {{ end }}
  637. {{ else }}
  638. <div class="items">
  639. {{ range .entries }}
  640. <article class="item touch-item item-status-{{ .Status }}" data-id="{{ .ID }}">
  641. <div class="item-header">
  642. <span class="item-title">
  643. {{ if ne .Feed.Icon.IconID 0 }}
  644. <img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="{{ .Feed.Title }}">
  645. {{ end }}
  646. <a href="{{ route "feedEntry" "feedID" .Feed.ID "entryID" .ID }}">{{ .Title }}</a>
  647. </span>
  648. <span class="category"><a href="{{ route "categoryEntries" "categoryID" .Feed.Category.ID }}">{{ .Feed.Category.Title }}</a></span>
  649. </div>
  650. {{ template "item_meta" dict "user" $.user "entry" . "hasSaveEntry" $.hasSaveEntry }}
  651. </article>
  652. {{ end }}
  653. </div>
  654. <section class="page-footer">
  655. {{ if .entries }}
  656. <ul>
  657. <li>
  658. <a href="#"
  659. data-action="markPageAsRead"
  660. data-label-question="{{ t "confirm.question" }}"
  661. data-label-yes="{{ t "confirm.yes" }}"
  662. data-label-no="{{ t "confirm.no" }}"
  663. data-label-loading="{{ t "confirm.loading" }}"
  664. data-show-only-unread="{{ if .showOnlyUnreadEntries }}1{{ end }}">{{ t "menu.mark_page_as_read" }}</a>
  665. </li>
  666. </ul>
  667. {{ end }}
  668. </section>
  669. {{ template "pagination" .pagination }}
  670. {{ end }}
  671. {{ end }}
  672. `,
  673. "feeds": `{{ define "title"}}{{ t "page.feeds.title" }} ({{ .total }}){{ end }}
  674. {{ define "content"}}
  675. <section class="page-header">
  676. <h1>{{ t "page.feeds.title" }} ({{ .total }})</h1>
  677. <ul>
  678. <li>
  679. <a href="{{ route "addSubscription" }}">{{ t "menu.add_feed" }}</a>
  680. </li>
  681. <li>
  682. <a href="{{ route "export" }}">{{ t "menu.export" }}</a>
  683. </li>
  684. <li>
  685. <a href="{{ route "import" }}">{{ t "menu.import" }}</a>
  686. </li>
  687. <li>
  688. <a href="{{ route "refreshAllFeeds" }}">{{ t "menu.refresh_all_feeds" }}</a>
  689. </li>
  690. </ul>
  691. </section>
  692. {{ if not .feeds }}
  693. <p class="alert">{{ t "alert.no_feed" }}</p>
  694. {{ else }}
  695. <div class="items">
  696. {{ range .feeds }}
  697. <article class="item {{ if ne .ParsingErrorCount 0 }}feed-parsing-error{{ end }}">
  698. <div class="item-header">
  699. <span class="item-title">
  700. {{ if .Icon }}
  701. <img src="{{ route "icon" "iconID" .Icon.IconID }}" width="16" height="16" loading="lazy" alt="{{ .Title }}">
  702. {{ end }}
  703. {{ if .Disabled }} 🚫 {{ end }}
  704. <a href="{{ route "feedEntries" "feedID" .ID }}">{{ .Title }}</a>
  705. </span>
  706. <span class="feed-entries-counter">
  707. (<span title="{{ t "page.feeds.unread_counter" }}">{{ .UnreadCount }}</span>/<span title="{{ t "page.feeds.read_counter" }}">{{ .ReadCount }}</span>)
  708. </span>
  709. <span class="category">
  710. <a href="{{ route "categoryEntries" "categoryID" .Category.ID }}">{{ .Category.Title }}</a>
  711. </span>
  712. </div>
  713. <div class="item-meta">
  714. <ul>
  715. <li>
  716. <a href="{{ .SiteURL }}" title="{{ .SiteURL }}" target="_blank" rel="noopener noreferrer" referrerpolicy="no-referrer" data-original-link="true">{{ domain .SiteURL }}</a>
  717. </li>
  718. <li>
  719. {{ t "page.feeds.last_check" }} <time datetime="{{ isodate .CheckedAt }}" title="{{ isodate .CheckedAt }}">{{ elapsed $.user.Timezone .CheckedAt }}</time>
  720. </li>
  721. </ul>
  722. <ul>
  723. <li>
  724. <a href="{{ route "refreshFeed" "feedID" .ID }}">{{ t "menu.refresh_feed" }}</a>
  725. </li>
  726. <li>
  727. <a href="{{ route "editFeed" "feedID" .ID }}">{{ t "menu.edit_feed" }}</a>
  728. </li>
  729. <li>
  730. <a href="#"
  731. data-confirm="true"
  732. data-label-question="{{ t "confirm.question" }}"
  733. data-label-yes="{{ t "confirm.yes" }}"
  734. data-label-no="{{ t "confirm.no" }}"
  735. data-label-loading="{{ t "confirm.loading" }}"
  736. data-url="{{ route "removeFeed" "feedID" .ID }}">{{ t "action.remove" }}</a>
  737. </li>
  738. </ul>
  739. </div>
  740. {{ if ne .ParsingErrorCount 0 }}
  741. <div class="parsing-error">
  742. <strong title="{{ .ParsingErrorMsg }}" class="parsing-error-count">{{ plural "page.feeds.error_count" .ParsingErrorCount .ParsingErrorCount }}</strong>
  743. - <small class="parsing-error-message">{{ .ParsingErrorMsg }}</small>
  744. </div>
  745. {{ end }}
  746. </article>
  747. {{ end }}
  748. </div>
  749. {{ end }}
  750. {{ end }}
  751. `,
  752. "history_entries": `{{ define "title"}}{{ t "page.history.title" }} ({{ .total }}){{ end }}
  753. {{ define "content"}}
  754. <section class="page-header">
  755. <h1>{{ t "page.history.title" }} ({{ .total }})</h1>
  756. {{ if .entries }}
  757. <ul>
  758. <li>
  759. <a href="#"
  760. data-confirm="true"
  761. data-url="{{ route "flushHistory" }}"
  762. data-label-question="{{ t "confirm.question" }}"
  763. data-label-yes="{{ t "confirm.yes" }}"
  764. data-label-no="{{ t "confirm.no" }}"
  765. data-label-loading="{{ t "confirm.loading" }}">{{ t "menu.flush_history" }}</a>
  766. </li>
  767. </ul>
  768. {{ end }}
  769. </section>
  770. {{ if not .entries }}
  771. <p class="alert alert-info">{{ t "alert.no_history" }}</p>
  772. {{ else }}
  773. <div class="items">
  774. {{ range .entries }}
  775. <article class="item touch-item item-status-{{ .Status }}" data-id="{{ .ID }}">
  776. <div class="item-header">
  777. <span class="item-title">
  778. {{ if ne .Feed.Icon.IconID 0 }}
  779. <img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="{{ .Feed.Title }}">
  780. {{ end }}
  781. <a href="{{ route "readEntry" "entryID" .ID }}">{{ .Title }}</a>
  782. </span>
  783. <span class="category"><a href="{{ route "categoryEntries" "categoryID" .Feed.Category.ID }}">{{ .Feed.Category.Title }}</a></span>
  784. </div>
  785. {{ template "item_meta" dict "user" $.user "entry" . "hasSaveEntry" $.hasSaveEntry }}
  786. </article>
  787. {{ end }}
  788. </div>
  789. {{ template "pagination" .pagination }}
  790. {{ end }}
  791. {{ end }}
  792. `,
  793. "import": `{{ define "title"}}{{ t "page.import.title" }}{{ end }}
  794. {{ define "content"}}
  795. <section class="page-header">
  796. <h1>{{ t "page.import.title" }}</h1>
  797. <ul>
  798. <li>
  799. <a href="{{ route "feeds" }}">{{ t "menu.feeds" }}</a>
  800. </li>
  801. <li>
  802. <a href="{{ route "addSubscription" }}">{{ t "menu.add_feed" }}</a>
  803. </li>
  804. <li>
  805. <a href="{{ route "export" }}">{{ t "menu.export" }}</a>
  806. </li>
  807. </ul>
  808. </section>
  809. {{ if .errorMessage }}
  810. <div class="alert alert-error">{{ t .errorMessage }}</div>
  811. {{ end }}
  812. <form action="{{ route "uploadOPML" }}" method="post" enctype="multipart/form-data">
  813. <input type="hidden" name="csrf" value="{{ .csrf }}">
  814. <label for="form-file">{{ t "form.import.label.file" }}</label>
  815. <input type="file" name="file" id="form-file">
  816. <div class="buttons">
  817. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.saving" }}">{{ t "action.import" }}</button>
  818. </div>
  819. </form>
  820. <hr>
  821. <form action="{{ route "fetchOPML" }}" method="post" enctype="multipart/form-data">
  822. <input type="hidden" name="csrf" value="{{ .csrf }}">
  823. <label for="form-url">{{ t "form.import.label.url" }}</label>
  824. <input type="url" name="url" id="form-url" required>
  825. <div class="buttons">
  826. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.saving" }}">{{ t "action.import" }}</button>
  827. </div>
  828. </form>
  829. {{ end }}
  830. `,
  831. "integrations": `{{ define "title"}}{{ t "page.integrations.title" }}{{ end }}
  832. {{ define "content"}}
  833. <section class="page-header">
  834. <h1>{{ t "page.integrations.title" }}</h1>
  835. {{ template "settings_menu" dict "user" .user }}
  836. </section>
  837. <form method="post" autocomplete="off" action="{{ route "updateIntegration" }}">
  838. <input type="hidden" name="csrf" value="{{ .csrf }}">
  839. {{ if .errorMessage }}
  840. <div class="alert alert-error">{{ t .errorMessage }}</div>
  841. {{ end }}
  842. <h3>Fever</h3>
  843. <div class="form-section">
  844. <label>
  845. <input type="checkbox" name="fever_enabled" value="1" {{ if .form.FeverEnabled }}checked{{ end }}> {{ t "form.integration.fever_activate" }}
  846. </label>
  847. <label for="form-fever-username">{{ t "form.integration.fever_username" }}</label>
  848. <input type="text" name="fever_username" id="form-fever-username" value="{{ .form.FeverUsername }}">
  849. <label for="form-fever-password">{{ t "form.integration.fever_password" }}</label>
  850. <input type="password" name="fever_password" id="form-fever-password" value="{{ .form.FeverPassword }}" autocomplete="new-password">
  851. <p>{{ t "form.integration.fever_endpoint" }} <strong>{{ rootURL }}{{ route "feverEndpoint" }}</strong></p>
  852. </div>
  853. <h3>Pinboard</h3>
  854. <div class="form-section">
  855. <label>
  856. <input type="checkbox" name="pinboard_enabled" value="1" {{ if .form.PinboardEnabled }}checked{{ end }}> {{ t "form.integration.pinboard_activate" }}
  857. </label>
  858. <label for="form-pinboard-token">{{ t "form.integration.pinboard_token" }}</label>
  859. <input type="password" name="pinboard_token" id="form-pinboard-token" value="{{ .form.PinboardToken }}" autocomplete="new-password">
  860. <label for="form-pinboard-tags">{{ t "form.integration.pinboard_tags" }}</label>
  861. <input type="text" name="pinboard_tags" id="form-pinboard-tags" value="{{ .form.PinboardTags }}">
  862. <label>
  863. <input type="checkbox" name="pinboard_mark_as_unread" value="1" {{ if .form.PinboardMarkAsUnread }}checked{{ end }}> {{ t "form.integration.pinboard_bookmark" }}
  864. </label>
  865. </div>
  866. <h3>Instapaper</h3>
  867. <div class="form-section">
  868. <label>
  869. <input type="checkbox" name="instapaper_enabled" value="1" {{ if .form.InstapaperEnabled }}checked{{ end }}> {{ t "form.integration.instapaper_activate" }}
  870. </label>
  871. <label for="form-instapaper-username">{{ t "form.integration.instapaper_username" }}</label>
  872. <input type="text" name="instapaper_username" id="form-instapaper-username" value="{{ .form.InstapaperUsername }}">
  873. <label for="form-instapaper-password">{{ t "form.integration.instapaper_password" }}</label>
  874. <input type="password" name="instapaper_password" id="form-instapaper-password" value="{{ .form.InstapaperPassword }}" autocomplete="new-password">
  875. </div>
  876. <h3>Pocket</h3>
  877. <div class="form-section">
  878. <label>
  879. <input type="checkbox" name="pocket_enabled" value="1" {{ if .form.PocketEnabled }}checked{{ end }}> {{ t "form.integration.pocket_activate" }}
  880. </label>
  881. {{ if not .hasPocketConsumerKeyConfigured }}
  882. <label for="form-pocket-consumer-key">{{ t "form.integration.pocket_consumer_key" }}</label>
  883. <input type="text" name="pocket_consumer_key" id="form-pocket-consumer-key" value="{{ .form.PocketConsumerKey }}">
  884. {{ end }}
  885. <label for="form-pocket-access-token">{{ t "form.integration.pocket_access_token" }}</label>
  886. <input type="password" name="pocket_access_token" id="form-pocket-access-token" value="{{ .form.PocketAccessToken }}" autocomplete="new-password">
  887. {{ if not .form.PocketAccessToken }}
  888. <p><a href="{{ route "pocketAuthorize" }}">{{ t "form.integration.pocket_connect_link" }}</a></p>
  889. {{ end }}
  890. </div>
  891. <h3>Wallabag</h3>
  892. <div class="form-section">
  893. <label>
  894. <input type="checkbox" name="wallabag_enabled" value="1" {{ if .form.WallabagEnabled }}checked{{ end }}> {{ t "form.integration.wallabag_activate" }}
  895. </label>
  896. <label for="form-wallabag-url">{{ t "form.integration.wallabag_endpoint" }}</label>
  897. <input type="url" name="wallabag_url" id="form-wallabag-url" value="{{ .form.WallabagURL }}" placeholder="http://v2.wallabag.org/">
  898. <label for="form-wallabag-client-id">{{ t "form.integration.wallabag_client_id" }}</label>
  899. <input type="text" name="wallabag_client_id" id="form-wallabag-client-id" value="{{ .form.WallabagClientID }}">
  900. <label for="form-wallabag-client-secret">{{ t "form.integration.wallabag_client_secret" }}</label>
  901. <input type="password" name="wallabag_client_secret" id="form-wallabag-client-secret" value="{{ .form.WallabagClientSecret }}" autocomplete="new-password">
  902. <label for="form-wallabag-username">{{ t "form.integration.wallabag_username" }}</label>
  903. <input type="text" name="wallabag_username" id="form-wallabag-username" value="{{ .form.WallabagUsername }}">
  904. <label for="form-wallabag-password">{{ t "form.integration.wallabag_password" }}</label>
  905. <input type="password" name="wallabag_password" id="form-wallabag-password" value="{{ .form.WallabagPassword }}" autocomplete="new-password">
  906. </div>
  907. <h3>Nunux Keeper</h3>
  908. <div class="form-section">
  909. <label>
  910. <input type="checkbox" name="nunux_keeper_enabled" value="1" {{ if .form.NunuxKeeperEnabled }}checked{{ end }}> {{ t "form.integration.nunux_keeper_activate" }}
  911. </label>
  912. <label for="form-nunux-keeper-url">{{ t "form.integration.nunux_keeper_endpoint" }}</label>
  913. <input type="url" name="nunux_keeper_url" id="form-nunux-keeper-url" value="{{ .form.NunuxKeeperURL }}" placeholder="https://api.nunux.org/keeper">
  914. <label for="form-nunux-keeper-api-key">{{ t "form.integration.nunux_keeper_api_key" }}</label>
  915. <input type="text" name="nunux_keeper_api_key" id="form-nunux-keeper-api-key" value="{{ .form.NunuxKeeperAPIKey }}">
  916. </div>
  917. <div class="buttons">
  918. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.saving" }}">{{ t "action.update" }}</button>
  919. </div>
  920. </form>
  921. <h3>{{ t "page.integration.miniflux_api" }}</h3>
  922. <div class="panel">
  923. <ul>
  924. <li>
  925. {{ t "page.integration.miniflux_api_endpoint" }} = <strong>{{ baseURL }}/v1/</strong>
  926. </li>
  927. <li>
  928. {{ t "page.integration.miniflux_api_username" }} = <strong>{{ .user.Username }}</strong>
  929. </li>
  930. <li>
  931. {{ t "page.integration.miniflux_api_password" }} = <strong>{{ t "page.integration.miniflux_api_password_value" }}</strong>
  932. </li>
  933. </ul>
  934. </div>
  935. <h3>{{ t "page.integration.bookmarklet" }}</h3>
  936. <div class="panel">
  937. <p>{{ t "page.integration.bookmarklet.help" }}</p>
  938. <div class="bookmarklet">
  939. <a href="javascript:location.href='{{ rootURL }}{{ route "bookmarklet" }}?uri='+encodeURIComponent(window.location.href)">{{ t "page.integration.bookmarklet.name" }}</a>
  940. </div>
  941. <p>{{ t "page.integration.bookmarklet.instructions" }}</p>
  942. </div>
  943. {{ end }}
  944. `,
  945. "login": `{{ define "title"}}{{ t "page.login.title" }}{{ end }}
  946. {{ define "content"}}
  947. <section class="login-form">
  948. <form action="{{ route "checkLogin" }}" method="post">
  949. <input type="hidden" name="csrf" value="{{ .csrf }}">
  950. {{ if .errorMessage }}
  951. <div class="alert alert-error">{{ t .errorMessage }}</div>
  952. {{ end }}
  953. <label for="form-username">{{ t "form.user.label.username" }}</label>
  954. <input type="text" name="username" id="form-username" value="{{ .form.Username }}" required autofocus>
  955. <label for="form-password">{{ t "form.user.label.password" }}</label>
  956. <input type="password" name="password" id="form-password" value="{{ .form.Password }}" required>
  957. <div class="buttons">
  958. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.loading" }}">{{ t "action.login" }}</button>
  959. </div>
  960. </form>
  961. {{ if hasOAuth2Provider "google" }}
  962. <div class="oauth2">
  963. <a href="{{ route "oauth2Redirect" "provider" "google" }}">{{ t "page.login.google_signin" }}</a>
  964. </div>
  965. {{ end }}
  966. </section>
  967. {{ end }}
  968. `,
  969. "search_entries": `{{ define "title"}}{{ t "page.search.title" }} ({{ .total }}){{ end }}
  970. {{ define "content"}}
  971. <section class="page-header">
  972. <h1>{{ t "page.search.title" }} ({{ .total }})</h1>
  973. </section>
  974. {{ if not .entries }}
  975. <p class="alert alert-info">{{ t "alert.no_search_result" }}</p>
  976. {{ else }}
  977. <div class="items">
  978. {{ range .entries }}
  979. <article class="item touch-item item-status-{{ .Status }}" data-id="{{ .ID }}">
  980. <div class="item-header">
  981. <span class="item-title">
  982. {{ if ne .Feed.Icon.IconID 0 }}
  983. <img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="{{ .Feed.Title }}">
  984. {{ end }}
  985. <a href="{{ route "searchEntry" "entryID" .ID }}?q={{ $.searchQuery }}">{{ .Title }}</a>
  986. </span>
  987. <span class="category"><a href="{{ route "categoryEntries" "categoryID" .Feed.Category.ID }}">{{ .Feed.Category.Title }}</a></span>
  988. </div>
  989. {{ template "item_meta" dict "user" $.user "entry" . "hasSaveEntry" $.hasSaveEntry }}
  990. </article>
  991. {{ end }}
  992. </div>
  993. {{ template "pagination" .pagination }}
  994. {{ end }}
  995. {{ end }}
  996. `,
  997. "sessions": `{{ define "title"}}{{ t "page.sessions.title" }}{{ end }}
  998. {{ define "content"}}
  999. <section class="page-header">
  1000. <h1>{{ t "page.sessions.title" }}</h1>
  1001. {{ template "settings_menu" dict "user" .user }}
  1002. </section>
  1003. <table>
  1004. <tr>
  1005. <th>{{ t "page.sessions.table.date" }}</th>
  1006. <th>{{ t "page.sessions.table.ip" }}</th>
  1007. <th>{{ t "page.sessions.table.user_agent" }}</th>
  1008. <th>{{ t "page.sessions.table.actions" }}</th>
  1009. </tr>
  1010. {{ range .sessions }}
  1011. <tr {{ if eq .Token $.currentSessionToken }}class="row-highlighted"{{ end }}>
  1012. <td class="column-20" title="{{ isodate .CreatedAt }}">{{ elapsed $.user.Timezone .CreatedAt }}</td>
  1013. <td class="column-20" title="{{ .IP }}">{{ .IP }}</td>
  1014. <td title="{{ .UserAgent }}">{{ .UserAgent }}</td>
  1015. <td class="column-20">
  1016. {{ if eq .Token $.currentSessionToken }}
  1017. {{ t "page.sessions.table.current_session" }}
  1018. {{ else }}
  1019. <a href="#"
  1020. data-confirm="true"
  1021. data-label-question="{{ t "confirm.question" }}"
  1022. data-label-yes="{{ t "confirm.yes" }}"
  1023. data-label-no="{{ t "confirm.no" }}"
  1024. data-label-loading="{{ t "confirm.loading" }}"
  1025. data-url="{{ route "removeSession" "sessionID" .ID }}">{{ t "action.remove" }}</a>
  1026. {{ end }}
  1027. </td>
  1028. </tr>
  1029. {{ end }}
  1030. </table>
  1031. {{ end }}
  1032. `,
  1033. "settings": `{{ define "title"}}{{ t "page.settings.title" }}{{ end }}
  1034. {{ define "content"}}
  1035. <section class="page-header">
  1036. <h1>{{ t "page.settings.title" }}</h1>
  1037. {{ template "settings_menu" dict "user" .user }}
  1038. </section>
  1039. <form method="post" autocomplete="off" action="{{ route "updateSettings" }}">
  1040. <input type="hidden" name="csrf" value="{{ .csrf }}">
  1041. {{ if .errorMessage }}
  1042. <div class="alert alert-error">{{ t .errorMessage }}</div>
  1043. {{ end }}
  1044. <label for="form-username">{{ t "form.user.label.username" }}</label>
  1045. <input type="text" name="username" id="form-username" value="{{ .form.Username }}" required>
  1046. <label for="form-password">{{ t "form.user.label.password" }}</label>
  1047. <input type="password" name="password" id="form-password" value="{{ .form.Password }}" autocomplete="new-password">
  1048. <label for="form-confirmation">{{ t "form.user.label.confirmation" }}</label>
  1049. <input type="password" name="confirmation" id="form-confirmation" value="{{ .form.Confirmation }}" autocomplete="new-password">
  1050. <label for="form-language">{{ t "form.prefs.label.language" }}</label>
  1051. <select id="form-language" name="language">
  1052. {{ range $key, $value := .languages }}
  1053. <option value="{{ $key }}" {{ if eq $key $.form.Language }}selected="selected"{{ end }}>{{ $value }}</option>
  1054. {{ end }}
  1055. </select>
  1056. <label for="form-timezone">{{ t "form.prefs.label.timezone" }}</label>
  1057. <select id="form-timezone" name="timezone">
  1058. {{ range $key, $value := .timezones }}
  1059. <option value="{{ $key }}" {{ if eq $key $.form.Timezone }}selected="selected"{{ end }}>{{ $value }}</option>
  1060. {{ end }}
  1061. </select>
  1062. <label for="form-theme">{{ t "form.prefs.label.theme" }}</label>
  1063. <select id="form-theme" name="theme">
  1064. {{ range $key, $value := .themes }}
  1065. <option value="{{ $key }}" {{ if eq $key $.form.Theme }}selected="selected"{{ end }}>{{ $value }}</option>
  1066. {{ end }}
  1067. </select>
  1068. <label for="form-entry-direction">{{ t "form.prefs.label.entry_sorting" }}</label>
  1069. <select id="form-entry-direction" name="entry_direction">
  1070. <option value="asc" {{ if eq "asc" $.form.EntryDirection }}selected="selected"{{ end }}>{{ t "form.prefs.select.older_first" }}</option>
  1071. <option value="desc" {{ if eq "desc" $.form.EntryDirection }}selected="selected"{{ end }}>{{ t "form.prefs.select.recent_first" }}</option>
  1072. </select>
  1073. <label><input type="checkbox" name="keyboard_shortcuts" value="1" {{ if .form.KeyboardShortcuts }}checked{{ end }}> {{ t "form.prefs.label.keyboard_shortcuts" }}</label>
  1074. <div class="buttons">
  1075. <button type="submit" class="button button-primary" data-label-loading="{{ t "form.submit.saving" }}">{{ t "action.update" }}</button>
  1076. </div>
  1077. </form>
  1078. {{ if hasOAuth2Provider "google" }}
  1079. <div class="panel">
  1080. {{ if hasKey .user.Extra "google_id" }}
  1081. <a href="{{ route "oauth2Unlink" "provider" "google" }}">{{ t "page.settings.unlink_google_account" }}</a>
  1082. {{ else }}
  1083. <a href="{{ route "oauth2Redirect" "provider" "google" }}">{{ t "page.settings.link_google_account" }}</a>
  1084. {{ end }}
  1085. </div>
  1086. {{ end }}
  1087. {{ end }}
  1088. `,
  1089. "unread_entries": `{{ define "title"}}{{ t "page.unread.title" }} {{ if gt .countUnread 0 }}({{ .countUnread }}){{ end }} {{ end }}
  1090. {{ define "content"}}
  1091. <section class="page-header">
  1092. <h1>{{ t "page.unread.title" }} (<span class="unread-counter">{{ .countUnread }}</span>)</h1>
  1093. {{ if .entries }}
  1094. <ul>
  1095. <li>
  1096. <a href="#"
  1097. data-action="markPageAsRead"
  1098. data-label-question="{{ t "confirm.question" }}"
  1099. data-label-yes="{{ t "confirm.yes" }}"
  1100. data-label-no="{{ t "confirm.no" }}"
  1101. data-label-loading="{{ t "confirm.loading" }}">{{ t "menu.mark_page_as_read" }}</a>
  1102. </li>
  1103. <li>
  1104. <a href="#"
  1105. data-confirm="true"
  1106. data-url="{{ route "markAllAsRead" }}"
  1107. data-redirect-url="{{ route "unread" }}"
  1108. data-label-question="{{ t "confirm.question" }}"
  1109. data-label-yes="{{ t "confirm.yes" }}"
  1110. data-label-no="{{ t "confirm.no" }}"
  1111. data-label-loading="{{ t "confirm.loading" }}">{{ t "menu.mark_all_as_read" }}</a>
  1112. </li>
  1113. </ul>
  1114. {{ end }}
  1115. </section>
  1116. {{ if not .entries }}
  1117. <p class="alert">{{ t "alert.no_unread_entry" }}</p>
  1118. {{ else }}
  1119. <div class="items hide-read-items">
  1120. {{ range .entries }}
  1121. <article class="item touch-item item-status-{{ .Status }}" data-id="{{ .ID }}">
  1122. <div class="item-header">
  1123. <span class="item-title">
  1124. {{ if ne .Feed.Icon.IconID 0 }}
  1125. <img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" loading="lazy" alt="{{ .Feed.Title }}">
  1126. {{ end }}
  1127. <a href="{{ route "unreadEntry" "entryID" .ID }}">{{ .Title }}</a>
  1128. </span>
  1129. <span class="category"><a href="{{ route "categoryEntries" "categoryID" .Feed.Category.ID }}">{{ .Feed.Category.Title }}</a></span>
  1130. </div>
  1131. {{ template "item_meta" dict "user" $.user "entry" . "hasSaveEntry" $.hasSaveEntry }}
  1132. </article>
  1133. {{ end }}
  1134. </div>
  1135. <section class="page-footer">
  1136. {{ if .entries }}
  1137. <ul>
  1138. <li>
  1139. <a href="#"
  1140. data-action="markPageAsRead"
  1141. data-label-question="{{ t "confirm.question" }}"
  1142. data-label-yes="{{ t "confirm.yes" }}"
  1143. data-label-no="{{ t "confirm.no" }}"
  1144. data-label-loading="{{ t "confirm.loading" }}">{{ t "menu.mark_page_as_read" }}</a>
  1145. </li>
  1146. </ul>
  1147. {{ end }}
  1148. </section>
  1149. {{ template "pagination" .pagination }}
  1150. {{ end }}
  1151. {{ end }}`,
  1152. "users": `{{ define "title"}}{{ t "page.users.title" }}{{ end }}
  1153. {{ define "content"}}
  1154. <section class="page-header">
  1155. <h1>{{ t "page.users.title" }}</h1>
  1156. {{ template "settings_menu" dict "user" .user }}
  1157. </section>
  1158. {{ if eq (len .users) 1 }}
  1159. <p class="alert">{{ t "alert.no_user" }}</p>
  1160. {{ else }}
  1161. <table>
  1162. <tr>
  1163. <th class="column-20">{{ t "page.users.username" }}</th>
  1164. <th>{{ t "page.users.is_admin" }}</th>
  1165. <th>{{ t "page.users.last_login" }}</th>
  1166. <th>{{ t "page.users.actions" }}</th>
  1167. </tr>
  1168. {{ range .users }}
  1169. {{ if ne .ID $.user.ID }}
  1170. <tr>
  1171. <td>{{ .Username }}</td>
  1172. <td>{{ if eq .IsAdmin true }}{{ t "page.users.admin.yes" }}{{ else }}{{ t "page.users.admin.no" }}{{ end }}</td>
  1173. <td>
  1174. {{ if .LastLoginAt }}
  1175. <time datetime="{{ isodate .LastLoginAt }}" title="{{ isodate .LastLoginAt }}">{{ elapsed $.user.Timezone .LastLoginAt }}</time>
  1176. {{ else }}
  1177. {{ t "page.users.never_logged" }}
  1178. {{ end }}
  1179. </td>
  1180. <td>
  1181. <a href="{{ route "editUser" "userID" .ID }}">{{ t "action.edit" }}</a>,
  1182. <a href="#"
  1183. data-confirm="true"
  1184. data-label-question="{{ t "confirm.question" }}"
  1185. data-label-yes="{{ t "confirm.yes" }}"
  1186. data-label-no="{{ t "confirm.no" }}"
  1187. data-label-loading="{{ t "confirm.loading" }}"
  1188. data-url="{{ route "removeUser" "userID" .ID }}">{{ t "action.remove" }}</a>
  1189. </td>
  1190. </tr>
  1191. {{ end }}
  1192. {{ end }}
  1193. </table>
  1194. {{ end }}
  1195. {{ end }}
  1196. `,
  1197. }
  1198. var templateViewsMapChecksums = map[string]string{
  1199. "about": "4035658497363d7af7f79be83190404eb21ec633fe8ec636bdfc219d9fc78cfc",
  1200. "add_subscription": "a0f1d2bc02b6adc83dbeae593f74d9b936102cd6dd73302cdbec2137cafdcdd9",
  1201. "bookmark_entries": "65588da78665699dd3f287f68325e9777d511f1a57fee4131a5bb6d00bb68df8",
  1202. "categories": "642ee3cddbd825ee6ab5a77caa0d371096b55de0f1bd4ae3055b8c8a70507d8d",
  1203. "category_entries": "3ec30d2cb97f29514ff61898a4f23d2aa73a24b3468b6d410b1c2d18c8808927",
  1204. "choose_subscription": "33c04843d7c1b608d034e605e52681822fc6d79bc6b900c04915dd9ebae584e2",
  1205. "create_category": "6b22b5ce51abf4e225e23a79f81be09a7fb90acb265e93a8faf9446dff74018d",
  1206. "create_user": "9b73a55233615e461d1f07d99ad1d4d3b54532588ab960097ba3e090c85aaf3a",
  1207. "edit_category": "daf073d2944a180ce5aaeb80b597eb69597a50dff55a9a1d6cf7938b48d768cb",
  1208. "edit_feed": "34aa0d668b3ea1a1b5fa480c20cebeae729b37010af3bb915d2a9eed73d3b996",
  1209. "edit_user": "c692db9de1a084c57b93e95a14b041d39bf489846cbb91fc982a62b72b77062a",
  1210. "entry": "24aeba26ef9a51ce585ca5c4af090f1de7d7bfd7f1e3ff1b63af520e2afa76bd",
  1211. "feed_entries": "9c70b82f55e4b311eff20be1641733612e3c1b406ce8010861e4c417d97b6dcc",
  1212. "feeds": "f11ba1c45cf3966843ddc406d96e048fc8f2235428e10111a1660a141ea2c42f",
  1213. "history_entries": "87e17d39de70eb3fdbc4000326283be610928758eae7924e4b08dcb446f3b6a9",
  1214. "import": "5eb56cecaa4d369b9acc991a82be7617710c551089a2e99d34ce8b6e5c37df0a",
  1215. "integrations": "6104ff6ff3ac3c1ae5e850c78250aab6e99e2342a337589f3848459fa333766a",
  1216. "login": "2e72d2d4b9786641b696bedbed5e10b04bdfd68254ddbbdb0a53cca621d200c7",
  1217. "search_entries": "274950d03298c24f3942e209c0faed580a6d57be9cf76a6c236175a7e766ac6a",
  1218. "sessions": "5d5c677bddbd027e0b0c9f7a0dd95b66d9d95b4e130959f31fb955b926c2201c",
  1219. "settings": "56f7c06f24eef317353582b0191aa9a5985f46ed755accf97e723ceb4bba4469",
  1220. "unread_entries": "e38f7ffce17dfad3151b08cd33771a2cefe8ca9db42df04fc98bd1d675dd6075",
  1221. "users": "17d0b7c760557e20f888d83d6a1b0d4506dab071a593cc42080ec0dbf16adf9e",
  1222. }