views.go 52 KB

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