views.go 68 KB

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