views.go 57 KB

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