views.go 72 KB

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