4
0

views.go 73 KB

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