views.go 58 KB

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