views.go 67 KB

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