views.go 71 KB

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