views.go 71 KB

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