views.go 59 KB

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