views.go 58 KB

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