views.go 52 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337
  1. // Code generated by go generate; DO NOT EDIT.
  2. package 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. {{ end }}
  409. {{ end }}`,
  410. "edit_user": `{{ define "title"}}{{ t "Edit user: %s" .selected_user.Username }}{{ end }}
  411. {{ define "content"}}
  412. <section class="page-header">
  413. <h1>{{ t "Edit user %s" .selected_user.Username }}</h1>
  414. <ul>
  415. <li>
  416. <a href="{{ route "settings" }}">{{ t "Settings" }}</a>
  417. </li>
  418. <li>
  419. <a href="{{ route "integrations" }}">{{ t "Integrations" }}</a>
  420. </li>
  421. <li>
  422. <a href="{{ route "sessions" }}">{{ t "Sessions" }}</a>
  423. </li>
  424. <li>
  425. <a href="{{ route "users" }}">{{ t "Users" }}</a>
  426. </li>
  427. <li>
  428. <a href="{{ route "createUser" }}">{{ t "Add user" }}</a>
  429. </li>
  430. <li>
  431. <a href="{{ route "about" }}">{{ t "About" }}</a>
  432. </li>
  433. </ul>
  434. </section>
  435. <form action="{{ route "updateUser" "userID" .selected_user.ID }}" method="post" autocomplete="off">
  436. <input type="hidden" name="csrf" value="{{ .csrf }}">
  437. {{ if .errorMessage }}
  438. <div class="alert alert-error">{{ t .errorMessage }}</div>
  439. {{ end }}
  440. <label for="form-username">{{ t "Username" }}</label>
  441. <input type="text" name="username" id="form-username" value="{{ .form.Username }}" autocomplete="new-password" required autofocus>
  442. <label for="form-password">{{ t "Password" }}</label>
  443. <input type="password" name="password" id="form-password" value="{{ .form.Password }}" autocomplete="new-password">
  444. <label for="form-confirmation">{{ t "Confirmation" }}</label>
  445. <input type="password" name="confirmation" id="form-confirmation" value="{{ .form.Confirmation }}" autocomplete="new-password">
  446. <label><input type="checkbox" name="is_admin" value="1" {{ if .form.IsAdmin }}checked{{ end }}> {{ t "Administrator" }}</label>
  447. <div class="buttons">
  448. <button type="submit" class="button button-primary" data-label-loading="{{ t "Loading..." }}">{{ t "Update" }}</button> {{ t "or" }} <a href="{{ route "users" }}">{{ t "cancel" }}</a>
  449. </div>
  450. </form>
  451. {{ end }}
  452. `,
  453. "entry": `{{ define "title"}}{{ .entry.Title }}{{ end }}
  454. {{ define "content"}}
  455. <section class="entry">
  456. <header class="entry-header">
  457. <h1>
  458. <a href="{{ .entry.URL }}" target="_blank" rel="noopener noreferrer" referrerpolicy="no-referrer">{{ .entry.Title }}</a>
  459. </h1>
  460. <div class="entry-actions">
  461. <ul>
  462. <li>
  463. <a href="#"
  464. data-toggle-bookmark="true"
  465. data-bookmark-url="{{ route "toggleBookmark" "entryID" .entry.ID }}"
  466. data-label-loading="{{ t "Saving..." }}"
  467. data-label-star="☆ {{ t "Star" }}"
  468. data-label-unstar="★ {{ t "Unstar" }}"
  469. data-value="{{ if .entry.Starred }}star{{ else }}unstar{{ end }}"
  470. >{{ if .entry.Starred }}★ {{ t "Unstar" }}{{ else }}☆ {{ t "Star" }}{{ end }}</a>
  471. </li>
  472. {{ if .hasSaveEntry }}
  473. <li>
  474. <a href="#"
  475. title="{{ t "Save this article" }}"
  476. data-save-entry="true"
  477. data-save-url="{{ route "saveEntry" "entryID" .entry.ID }}"
  478. data-label-loading="{{ t "Saving..." }}"
  479. data-label-done="{{ t "Done!" }}"
  480. >{{ t "Save" }}</a>
  481. </li>
  482. {{ end }}
  483. <li>
  484. <a href="#"
  485. title="{{ t "Fetch original content" }}"
  486. data-fetch-content-entry="true"
  487. data-fetch-content-url="{{ route "fetchContent" "entryID" .entry.ID }}"
  488. data-label-loading="{{ t "Loading..." }}"
  489. data-label-done="{{ t "Done!" }}"
  490. >{{ t "Fetch original content" }}</a>
  491. </li>
  492. {{ if .entry.CommentsURL }}
  493. <li>
  494. <a href="{{ .entry.CommentsURL }}" title="{{ t "View Comments" }}" target="_blank" rel="noopener noreferrer" referrerpolicy="no-referrer">{{ t "Comments" }}</a>
  495. </li>
  496. {{ end }}
  497. </ul>
  498. </div>
  499. <div class="entry-meta">
  500. <span class="entry-website">
  501. {{ if ne .entry.Feed.Icon.IconID 0 }}
  502. <img src="{{ route "icon" "iconID" .entry.Feed.Icon.IconID }}" width="16" height="16" alt="{{ .entry.Feed.Title }}">
  503. {{ end }}
  504. <a href="{{ route "feedEntries" "feedID" .entry.Feed.ID }}">{{ .entry.Feed.Title }}</a>
  505. </span>
  506. {{ if .entry.Author }}
  507. <span class="entry-author">
  508. {{ if isEmail .entry.Author }}
  509. - <a href="mailto:{{ .entry.Author }}">{{ .entry.Author }}</a>
  510. {{ else }}
  511. – <em>{{ .entry.Author }}</em>
  512. {{ end }}
  513. </span>
  514. {{ end }}
  515. <span class="category">
  516. <a href="{{ route "categoryEntries" "categoryID" .entry.Feed.Category.ID }}">{{ .entry.Feed.Category.Title }}</a>
  517. </span>
  518. </div>
  519. <div class="entry-date">
  520. <time datetime="{{ isodate .entry.Date }}" title="{{ isodate .entry.Date }}">{{ elapsed $.user.Timezone .entry.Date }}</time>
  521. </div>
  522. </header>
  523. {{ if gt (len .entry.Content) 120 }}
  524. <div class="pagination-top">
  525. {{ template "entry_pagination" . }}
  526. </div>
  527. {{ end }}
  528. <article class="entry-content">
  529. {{ noescape (proxyFilter .entry.Content) }}
  530. </article>
  531. {{ if .entry.Enclosures }}
  532. <aside class="entry-enclosures">
  533. <h3>{{ t "Attachments" }}</h3>
  534. {{ range .entry.Enclosures }}
  535. <div class="entry-enclosure">
  536. {{ if hasPrefix .MimeType "audio/" }}
  537. <div class="enclosure-audio">
  538. <audio controls preload="metadata">
  539. <source src="{{ .URL }}" type="{{ .MimeType }}">
  540. </audio>
  541. </div>
  542. {{ else if hasPrefix .MimeType "video/" }}
  543. <div class="enclosure-video">
  544. <video controls preload="metadata">
  545. <source src="{{ .URL }}" type="{{ .MimeType }}">
  546. </video>
  547. </div>
  548. {{ else if hasPrefix .MimeType "image/" }}
  549. <div class="enclosure-image">
  550. <img src="{{ proxyURL .URL }}" title="{{ .URL }} ({{ .MimeType }})" alt="{{ .URL }} ({{ .MimeType }})">
  551. </div>
  552. {{ end }}
  553. <div class="entry-enclosure-download">
  554. <a href="{{ .URL }}" title="{{ .URL }} ({{ .MimeType }})" target="_blank" rel="noopener noreferrer" referrerpolicy="no-referrer">{{ t "Download" }}</a>
  555. <small>({{ .URL }})</small>
  556. </div>
  557. </div>
  558. {{ end }}
  559. </aside>
  560. {{ end }}
  561. </section>
  562. <div class="pagination-bottom">
  563. {{ template "entry_pagination" . }}
  564. </div>
  565. {{ end }}
  566. `,
  567. "feed_entries": `{{ define "title"}}{{ .feed.Title }} ({{ .total }}){{ end }}
  568. {{ define "content"}}
  569. <section class="page-header">
  570. <h1>{{ .feed.Title }} ({{ .total }})</h1>
  571. <ul>
  572. <li>
  573. <a href="{{ route "refreshFeed" "feedID" .feed.ID }}">{{ t "Refresh" }}</a>
  574. </li>
  575. <li>
  576. <a href="{{ route "editFeed" "feedID" .feed.ID }}">{{ t "Edit" }}</a>
  577. </li>
  578. {{ if .entries }}
  579. <li>
  580. <a href="#" data-on-click="markPageAsRead">{{ t "Mark this page as read" }}</a>
  581. </li>
  582. {{ end }}
  583. </ul>
  584. </section>
  585. {{ if ne .feed.ParsingErrorCount 0 }}
  586. <div class="alert alert-error">
  587. <h3>{{ t "There is a problem with this feed" }}</h3>
  588. <p>{{ t .feed.ParsingErrorMsg }}</p>
  589. </div>
  590. {{ else if not .entries }}
  591. <p class="alert">{{ t "There is no article for this feed." }}</p>
  592. {{ else }}
  593. <div class="items">
  594. {{ range .entries }}
  595. <article class="item touch-item item-status-{{ .Status }}" data-id="{{ .ID }}">
  596. <div class="item-header">
  597. <span class="item-title">
  598. {{ if ne .Feed.Icon.IconID 0 }}
  599. <img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" alt="{{ .Feed.Title }}">
  600. {{ end }}
  601. <a href="{{ route "feedEntry" "feedID" .Feed.ID "entryID" .ID }}">{{ .Title }}</a>
  602. </span>
  603. <span class="category"><a href="{{ route "categoryEntries" "categoryID" .Feed.Category.ID }}">{{ .Feed.Category.Title }}</a></span>
  604. </div>
  605. {{ template "item_meta" dict "user" $.user "entry" . "hasSaveEntry" $.hasSaveEntry }}
  606. </article>
  607. {{ end }}
  608. </div>
  609. {{ template "pagination" .pagination }}
  610. {{ end }}
  611. {{ end }}
  612. `,
  613. "feeds": `{{ define "title"}}{{ t "Feeds" }} ({{ .total }}){{ end }}
  614. {{ define "content"}}
  615. <section class="page-header">
  616. <h1>{{ t "Feeds" }} ({{ .total }})</h1>
  617. <ul>
  618. <li>
  619. <a href="{{ route "addSubscription" }}">{{ t "Add subscription" }}</a>
  620. </li>
  621. <li>
  622. <a href="{{ route "export" }}">{{ t "Export" }}</a>
  623. </li>
  624. <li>
  625. <a href="{{ route "import" }}">{{ t "Import" }}</a>
  626. </li>
  627. <li>
  628. <a href="{{ route "refreshAllFeeds" }}">{{ t "Refresh all feeds in background" }}</a>
  629. </li>
  630. </ul>
  631. </section>
  632. {{ if not .feeds }}
  633. <p class="alert">{{ t "You don't have any subscription." }}</p>
  634. {{ else }}
  635. <div class="items">
  636. {{ range .feeds }}
  637. <article class="item {{ if ne .ParsingErrorCount 0 }}feed-parsing-error{{ end }}">
  638. <div class="item-header">
  639. <span class="item-title">
  640. {{ if .Icon }}
  641. <img src="{{ route "icon" "iconID" .Icon.IconID }}" width="16" height="16" alt="{{ .Title }}">
  642. {{ end }}
  643. <a href="{{ route "feedEntries" "feedID" .ID }}">{{ .Title }}</a>
  644. </span>
  645. <span class="category">
  646. <a href="{{ route "categoryEntries" "categoryID" .Category.ID }}">{{ .Category.Title }}</a>
  647. </span>
  648. </div>
  649. <div class="item-meta">
  650. <ul>
  651. <li>
  652. <a href="{{ .SiteURL }}" title="{{ .SiteURL }}" target="_blank" rel="noopener noreferrer" referrerpolicy="no-referrer" data-original-link="true">{{ domain .SiteURL }}</a>
  653. </li>
  654. <li>
  655. {{ t "Last check:" }} <time datetime="{{ isodate .CheckedAt }}" title="{{ isodate .CheckedAt }}">{{ elapsed $.user.Timezone .CheckedAt }}</time>
  656. </li>
  657. </ul>
  658. <ul>
  659. <li>
  660. <a href="{{ route "refreshFeed" "feedID" .ID }}">{{ t "Refresh" }}</a>
  661. </li>
  662. <li>
  663. <a href="{{ route "editFeed" "feedID" .ID }}">{{ t "Edit" }}</a>
  664. </li>
  665. <li>
  666. <a href="#"
  667. data-confirm="true"
  668. data-label-question="{{ t "Are you sure?" }}"
  669. data-label-yes="{{ t "yes" }}"
  670. data-label-no="{{ t "no" }}"
  671. data-label-loading="{{ t "Work in progress..." }}"
  672. data-url="{{ route "removeFeed" "feedID" .ID }}">{{ t "Remove" }}</a>
  673. </li>
  674. </ul>
  675. </div>
  676. {{ if ne .ParsingErrorCount 0 }}
  677. <div class="parsing-error">
  678. <strong title="{{ .ParsingErrorMsg }}" class="parsing-error-count">{{ plural "plural.feed.error_count" .ParsingErrorCount .ParsingErrorCount }}</strong>
  679. - <small class="parsing-error-message">{{ .ParsingErrorMsg }}</small>
  680. </div>
  681. {{ end }}
  682. </article>
  683. {{ end }}
  684. </div>
  685. {{ end }}
  686. {{ end }}
  687. `,
  688. "history_entries": `{{ define "title"}}{{ t "History" }} ({{ .total }}){{ end }}
  689. {{ define "content"}}
  690. <section class="page-header">
  691. <h1>{{ t "History" }} ({{ .total }})</h1>
  692. {{ if .entries }}
  693. <ul>
  694. <li>
  695. <a href="{{ route "flushHistory" }}">{{ t "Flush history" }}</a>
  696. </li>
  697. </ul>
  698. {{ end }}
  699. </section>
  700. {{ if not .entries }}
  701. <p class="alert alert-info">{{ t "There is no history at the moment." }}</p>
  702. {{ else }}
  703. <div class="items">
  704. {{ range .entries }}
  705. <article class="item touch-item item-status-{{ .Status }}" data-id="{{ .ID }}">
  706. <div class="item-header">
  707. <span class="item-title">
  708. {{ if ne .Feed.Icon.IconID 0 }}
  709. <img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" alt="{{ .Feed.Title }}">
  710. {{ end }}
  711. <a href="{{ route "readEntry" "entryID" .ID }}">{{ .Title }}</a>
  712. </span>
  713. <span class="category"><a href="{{ route "categoryEntries" "categoryID" .Feed.Category.ID }}">{{ .Feed.Category.Title }}</a></span>
  714. </div>
  715. {{ template "item_meta" dict "user" $.user "entry" . "hasSaveEntry" $.hasSaveEntry }}
  716. </article>
  717. {{ end }}
  718. </div>
  719. {{ template "pagination" .pagination }}
  720. {{ end }}
  721. {{ end }}
  722. `,
  723. "import": `{{ define "title"}}{{ t "Import" }}{{ end }}
  724. {{ define "content"}}
  725. <section class="page-header">
  726. <h1>{{ t "Import" }}</h1>
  727. <ul>
  728. <li>
  729. <a href="{{ route "feeds" }}">{{ t "Feeds" }}</a>
  730. </li>
  731. <li>
  732. <a href="{{ route "addSubscription" }}">{{ t "Add subscription" }}</a>
  733. </li>
  734. <li>
  735. <a href="{{ route "export" }}">{{ t "Export" }}</a>
  736. </li>
  737. </ul>
  738. </section>
  739. <form action="{{ route "uploadOPML" }}" method="post" enctype="multipart/form-data">
  740. <input type="hidden" name="csrf" value="{{ .csrf }}">
  741. {{ if .errorMessage }}
  742. <div class="alert alert-error">{{ t .errorMessage }}</div>
  743. {{ end }}
  744. <label for="form-file">{{ t "OPML file" }}</label>
  745. <input type="file" name="file" id="form-file">
  746. <div class="buttons">
  747. <button type="submit" class="button button-primary" data-label-loading="{{ t "Loading..." }}">{{ t "Import" }}</button>
  748. </div>
  749. </form>
  750. {{ end }}
  751. `,
  752. "integrations": `{{ define "title"}}{{ t "Integrations" }}{{ end }}
  753. {{ define "content"}}
  754. <section class="page-header">
  755. <h1>{{ t "Integrations" }}</h1>
  756. <ul>
  757. <li>
  758. <a href="{{ route "settings" }}">{{ t "Settings" }}</a>
  759. </li>
  760. <li>
  761. <a href="{{ route "sessions" }}">{{ t "Sessions" }}</a>
  762. </li>
  763. {{ if .user.IsAdmin }}
  764. <li>
  765. <a href="{{ route "users" }}">{{ t "Users" }}</a>
  766. </li>
  767. {{ end }}
  768. <li>
  769. <a href="{{ route "about" }}">{{ t "About" }}</a>
  770. </li>
  771. </ul>
  772. </section>
  773. <form method="post" autocomplete="off" action="{{ route "updateIntegration" }}">
  774. <input type="hidden" name="csrf" value="{{ .csrf }}">
  775. {{ if .errorMessage }}
  776. <div class="alert alert-error">{{ t .errorMessage }}</div>
  777. {{ end }}
  778. <h3>Fever</h3>
  779. <div class="form-section">
  780. <label>
  781. <input type="checkbox" name="fever_enabled" value="1" {{ if .form.FeverEnabled }}checked{{ end }}> {{ t "Activate Fever API" }}
  782. </label>
  783. <label for="form-fever-username">{{ t "Fever Username" }}</label>
  784. <input type="text" name="fever_username" id="form-fever-username" value="{{ .form.FeverUsername }}">
  785. <label for="form-fever-password">{{ t "Fever Password" }}</label>
  786. <input type="password" name="fever_password" id="form-fever-password" value="{{ .form.FeverPassword }}" autocomplete="new-password">
  787. <p>{{ t "Fever API endpoint:" }} <strong>{{ rootURL }}{{ route "feverEndpoint" }}</strong></p>
  788. </div>
  789. <h3>Pinboard</h3>
  790. <div class="form-section">
  791. <label>
  792. <input type="checkbox" name="pinboard_enabled" value="1" {{ if .form.PinboardEnabled }}checked{{ end }}> {{ t "Save articles to Pinboard" }}
  793. </label>
  794. <label for="form-pinboard-token">{{ t "Pinboard API Token" }}</label>
  795. <input type="password" name="pinboard_token" id="form-pinboard-token" value="{{ .form.PinboardToken }}" autocomplete="new-password">
  796. <label for="form-pinboard-tags">{{ t "Pinboard Tags" }}</label>
  797. <input type="text" name="pinboard_tags" id="form-pinboard-tags" value="{{ .form.PinboardTags }}">
  798. <label>
  799. <input type="checkbox" name="pinboard_mark_as_unread" value="1" {{ if .form.PinboardMarkAsUnread }}checked{{ end }}> {{ t "Mark bookmark as unread" }}
  800. </label>
  801. </div>
  802. <h3>Instapaper</h3>
  803. <div class="form-section">
  804. <label>
  805. <input type="checkbox" name="instapaper_enabled" value="1" {{ if .form.InstapaperEnabled }}checked{{ end }}> {{ t "Save articles to Instapaper" }}
  806. </label>
  807. <label for="form-instapaper-username">{{ t "Instapaper Username" }}</label>
  808. <input type="text" name="instapaper_username" id="form-instapaper-username" value="{{ .form.InstapaperUsername }}">
  809. <label for="form-instapaper-password">{{ t "Instapaper Password" }}</label>
  810. <input type="password" name="instapaper_password" id="form-instapaper-password" value="{{ .form.InstapaperPassword }}" autocomplete="new-password">
  811. </div>
  812. <h3>Pocket</h3>
  813. <div class="form-section">
  814. <label>
  815. <input type="checkbox" name="pocket_enabled" value="1" {{ if .form.PocketEnabled }}checked{{ end }}> {{ t "Save articles to Pocket" }}
  816. </label>
  817. {{ if not .hasPocketConsumerKeyConfigured }}
  818. <label for="form-pocket-consumer-key">{{ t "Pocket Consumer Key" }}</label>
  819. <input type="text" name="pocket_consumer_key" id="form-pocket-consumer-key" value="{{ .form.PocketConsumerKey }}">
  820. {{ end }}
  821. <label for="form-pocket-access-token">{{ t "Pocket Access Token" }}</label>
  822. <input type="password" name="pocket_access_token" id="form-pocket-access-token" value="{{ .form.PocketAccessToken }}" autocomplete="new-password">
  823. {{ if not .form.PocketAccessToken }}
  824. <p><a href="{{ route "pocketAuthorize" }}">{{ t "Connect your Pocket account" }}</a></p>
  825. {{ end }}
  826. </div>
  827. <h3>Wallabag</h3>
  828. <div class="form-section">
  829. <label>
  830. <input type="checkbox" name="wallabag_enabled" value="1" {{ if .form.WallabagEnabled }}checked{{ end }}> {{ t "Save articles to Wallabag" }}
  831. </label>
  832. <label for="form-wallabag-url">{{ t "Wallabag API Endpoint" }}</label>
  833. <input type="url" name="wallabag_url" id="form-wallabag-url" value="{{ .form.WallabagURL }}" placeholder="http://v2.wallabag.org/">
  834. <label for="form-wallabag-client-id">{{ t "Wallabag Client ID" }}</label>
  835. <input type="text" name="wallabag_client_id" id="form-wallabag-client-id" value="{{ .form.WallabagClientID }}">
  836. <label for="form-wallabag-client-secret">{{ t "Wallabag Client Secret" }}</label>
  837. <input type="password" name="wallabag_client_secret" id="form-wallabag-client-secret" value="{{ .form.WallabagClientSecret }}" autocomplete="new-password">
  838. <label for="form-wallabag-username">{{ t "Wallabag Username" }}</label>
  839. <input type="text" name="wallabag_username" id="form-wallabag-username" value="{{ .form.WallabagUsername }}">
  840. <label for="form-wallabag-password">{{ t "Wallabag Password" }}</label>
  841. <input type="password" name="wallabag_password" id="form-wallabag-password" value="{{ .form.WallabagPassword }}" autocomplete="new-password">
  842. </div>
  843. <h3>Nunux Keeper</h3>
  844. <div class="form-section">
  845. <label>
  846. <input type="checkbox" name="nunux_keeper_enabled" value="1" {{ if .form.NunuxKeeperEnabled }}checked{{ end }}> {{ t "Save articles to Nunux Keeper" }}
  847. </label>
  848. <label for="form-nunux-keeper-url">{{ t "Nunux Keeper API Endpoint" }}</label>
  849. <input type="url" name="nunux_keeper_url" id="form-nunux-keeper-url" value="{{ .form.NunuxKeeperURL }}" placeholder="https://api.nunux.org/keeper">
  850. <label for="form-nunux-keeper-api-key">{{ t "Nunux Keeper API key" }}</label>
  851. <input type="text" name="nunux_keeper_api_key" id="form-nunux-keeper-api-key" value="{{ .form.NunuxKeeperAPIKey }}">
  852. </div>
  853. <div class="buttons">
  854. <button type="submit" class="button button-primary" data-label-loading="{{ t "Loading..." }}">{{ t "Update" }}</button>
  855. </div>
  856. </form>
  857. <h3>{{ t "Miniflux API" }}</h3>
  858. <div class="panel">
  859. <ul>
  860. <li>
  861. {{ t "API Endpoint" }} = <strong>{{ baseURL }}/v1/</strong>
  862. </li>
  863. <li>
  864. {{ t "Username" }} = <strong>{{ .user.Username }}</strong>
  865. </li>
  866. <li>
  867. {{ t "Password" }} = <strong>{{ t "Your account password" }}</strong>
  868. </li>
  869. </ul>
  870. </div>
  871. <h3>{{ t "Bookmarklet" }}</h3>
  872. <div class="panel">
  873. <p>{{ t "This special link allows you to subscribe to a website directly by using a bookmark in your web browser." }}</p>
  874. <div class="bookmarklet">
  875. <a href="javascript:location.href='{{ rootURL }}{{ route "bookmarklet" }}?uri='+encodeURIComponent(window.location.href)">{{ t "Add to Miniflux" }}</a>
  876. </div>
  877. <p>{{ t "Drag and drop this link to your bookmarks." }}</p>
  878. </div>
  879. {{ end }}
  880. `,
  881. "login": `{{ define "title"}}{{ t "Sign In" }}{{ end }}
  882. {{ define "content"}}
  883. <section class="login-form">
  884. <form action="{{ route "checkLogin" }}" method="post">
  885. <input type="hidden" name="csrf" value="{{ .csrf }}">
  886. {{ if .errorMessage }}
  887. <div class="alert alert-error">{{ t .errorMessage }}</div>
  888. {{ end }}
  889. <label for="form-username">{{ t "Username" }}</label>
  890. <input type="text" name="username" id="form-username" value="{{ .form.Username }}" required autofocus>
  891. <label for="form-password">{{ t "Password" }}</label>
  892. <input type="password" name="password" id="form-password" value="{{ .form.Password }}" required>
  893. <div class="buttons">
  894. <button type="submit" class="button button-primary" data-label-loading="{{ t "Loading..." }}">{{ t "Sign in" }}</button>
  895. </div>
  896. </form>
  897. {{ if hasOAuth2Provider "google" }}
  898. <div class="oauth2">
  899. <a href="{{ route "oauth2Redirect" "provider" "google" }}">{{ t "Sign in with Google" }}</a>
  900. </div>
  901. {{ end }}
  902. </section>
  903. {{ end }}
  904. `,
  905. "search_entries": `{{ define "title"}}{{ t "Search Results" }} ({{ .total }}){{ end }}
  906. {{ define "content"}}
  907. <section class="page-header">
  908. <h1>{{ t "Search Results" }} ({{ .total }})</h1>
  909. </section>
  910. {{ if not .entries }}
  911. <p class="alert alert-info">{{ t "There is no result for this search." }}</p>
  912. {{ else }}
  913. <div class="items">
  914. {{ range .entries }}
  915. <article class="item touch-item item-status-{{ .Status }}" data-id="{{ .ID }}">
  916. <div class="item-header">
  917. <span class="item-title">
  918. {{ if ne .Feed.Icon.IconID 0 }}
  919. <img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" alt="{{ .Feed.Title }}">
  920. {{ end }}
  921. <a href="{{ route "searchEntry" "entryID" .ID }}?q={{ $.searchQuery }}">{{ .Title }}</a>
  922. </span>
  923. <span class="category"><a href="{{ route "categoryEntries" "categoryID" .Feed.Category.ID }}">{{ .Feed.Category.Title }}</a></span>
  924. </div>
  925. {{ template "item_meta" dict "user" $.user "entry" . "hasSaveEntry" $.hasSaveEntry }}
  926. </article>
  927. {{ end }}
  928. </div>
  929. {{ template "pagination" .pagination }}
  930. {{ end }}
  931. {{ end }}
  932. `,
  933. "sessions": `{{ define "title"}}{{ t "Sessions" }}{{ end }}
  934. {{ define "content"}}
  935. <section class="page-header">
  936. <h1>{{ t "Sessions" }}</h1>
  937. <ul>
  938. <li>
  939. <a href="{{ route "settings" }}">{{ t "Settings" }}</a>
  940. </li>
  941. <li>
  942. <a href="{{ route "integrations" }}">{{ t "Integrations" }}</a>
  943. </li>
  944. {{ if .user.IsAdmin }}
  945. <li>
  946. <a href="{{ route "users" }}">{{ t "Users" }}</a>
  947. </li>
  948. <li>
  949. <a href="{{ route "createUser" }}">{{ t "Add user" }}</a>
  950. </li>
  951. {{ end }}
  952. <li>
  953. <a href="{{ route "about" }}">{{ t "About" }}</a>
  954. </li>
  955. </ul>
  956. </section>
  957. <table>
  958. <tr>
  959. <th>{{ t "Date" }}</th>
  960. <th>{{ t "IP Address" }}</th>
  961. <th>{{ t "User Agent" }}</th>
  962. <th>{{ t "Actions" }}</th>
  963. </tr>
  964. {{ range .sessions }}
  965. <tr {{ if eq .Token $.currentSessionToken }}class="row-highlighted"{{ end }}>
  966. <td class="column-20" title="{{ isodate .CreatedAt }}">{{ elapsed $.user.Timezone .CreatedAt }}</td>
  967. <td class="column-20" title="{{ .IP }}">{{ .IP }}</td>
  968. <td title="{{ .UserAgent }}">{{ .UserAgent }}</td>
  969. <td class="column-20">
  970. {{ if eq .Token $.currentSessionToken }}
  971. {{ t "Current session" }}
  972. {{ else }}
  973. <a href="#"
  974. data-confirm="true"
  975. data-label-question="{{ t "Are you sure?" }}"
  976. data-label-yes="{{ t "yes" }}"
  977. data-label-no="{{ t "no" }}"
  978. data-label-loading="{{ t "Work in progress..." }}"
  979. data-url="{{ route "removeSession" "sessionID" .ID }}">{{ t "Remove" }}</a>
  980. {{ end }}
  981. </td>
  982. </tr>
  983. {{ end }}
  984. </table>
  985. {{ end }}
  986. `,
  987. "settings": `{{ define "title"}}{{ t "Settings" }}{{ end }}
  988. {{ define "content"}}
  989. <section class="page-header">
  990. <h1>{{ t "Settings" }}</h1>
  991. <ul>
  992. <li>
  993. <a href="{{ route "integrations" }}">{{ t "Integrations" }}</a>
  994. </li>
  995. <li>
  996. <a href="{{ route "sessions" }}">{{ t "Sessions" }}</a>
  997. </li>
  998. {{ if .user.IsAdmin }}
  999. <li>
  1000. <a href="{{ route "users" }}">{{ t "Users" }}</a>
  1001. </li>
  1002. {{ end }}
  1003. <li>
  1004. <a href="{{ route "about" }}">{{ t "About" }}</a>
  1005. </li>
  1006. </ul>
  1007. </section>
  1008. <form method="post" autocomplete="off" action="{{ route "updateSettings" }}">
  1009. <input type="hidden" name="csrf" value="{{ .csrf }}">
  1010. {{ if .errorMessage }}
  1011. <div class="alert alert-error">{{ t .errorMessage }}</div>
  1012. {{ end }}
  1013. <label for="form-username">{{ t "Username" }}</label>
  1014. <input type="text" name="username" id="form-username" value="{{ .form.Username }}" required>
  1015. <label for="form-password">{{ t "Password" }}</label>
  1016. <input type="password" name="password" id="form-password" value="{{ .form.Password }}" autocomplete="new-password">
  1017. <label for="form-confirmation">{{ t "Confirmation" }}</label>
  1018. <input type="password" name="confirmation" id="form-confirmation" value="{{ .form.Confirmation }}" autocomplete="new-password">
  1019. <label for="form-language">{{ t "Language" }}</label>
  1020. <select id="form-language" name="language">
  1021. {{ range $key, $value := .languages }}
  1022. <option value="{{ $key }}" {{ if eq $key $.form.Language }}selected="selected"{{ end }}>{{ $value }}</option>
  1023. {{ end }}
  1024. </select>
  1025. <label for="form-timezone">{{ t "Timezone" }}</label>
  1026. <select id="form-timezone" name="timezone">
  1027. {{ range $key, $value := .timezones }}
  1028. <option value="{{ $key }}" {{ if eq $key $.form.Timezone }}selected="selected"{{ end }}>{{ $value }}</option>
  1029. {{ end }}
  1030. </select>
  1031. <label for="form-theme">{{ t "Theme" }}</label>
  1032. <select id="form-theme" name="theme">
  1033. {{ range $key, $value := .themes }}
  1034. <option value="{{ $key }}" {{ if eq $key $.form.Theme }}selected="selected"{{ end }}>{{ $value }}</option>
  1035. {{ end }}
  1036. </select>
  1037. <label for="form-entry-direction">{{ t "Entry Sorting" }}</label>
  1038. <select id="form-entry-direction" name="entry_direction">
  1039. <option value="asc" {{ if eq "asc" $.form.EntryDirection }}selected="selected"{{ end }}>{{ t "Older entries first" }}</option>
  1040. <option value="desc" {{ if eq "desc" $.form.EntryDirection }}selected="selected"{{ end }}>{{ t "Recent entries first" }}</option>
  1041. </select>
  1042. <div class="buttons">
  1043. <button type="submit" class="button button-primary" data-label-loading="{{ t "Loading..." }}">{{ t "Update" }}</button>
  1044. </div>
  1045. </form>
  1046. {{ if hasOAuth2Provider "google" }}
  1047. <div class="panel">
  1048. {{ if hasKey .user.Extra "google_id" }}
  1049. <a href="{{ route "oauth2Unlink" "provider" "google" }}">{{ t "Unlink my Google account" }}</a>
  1050. {{ else }}
  1051. <a href="{{ route "oauth2Redirect" "provider" "google" }}">{{ t "Link my Google account" }}</a>
  1052. {{ end }}
  1053. </div>
  1054. {{ end }}
  1055. {{ end }}
  1056. `,
  1057. "unread_entries": `{{ define "title"}}{{ t "Unread Items" }} {{ if gt .countUnread 0 }}({{ .countUnread }}){{ end }} {{ end }}
  1058. {{ define "content"}}
  1059. <section class="page-header">
  1060. <h1>{{ t "Unread" }} (<span class="unread-counter">{{ .countUnread }}</span>)</h1>
  1061. {{ if .entries }}
  1062. <ul>
  1063. <li>
  1064. <a href="#" data-on-click="markPageAsRead">{{ t "Mark this page as read" }}</a>
  1065. </li>
  1066. <li>
  1067. <a href="{{ route "markAllAsRead" }}">{{ t "Mark all as read" }}</a>
  1068. </li>
  1069. </ul>
  1070. {{ end }}
  1071. </section>
  1072. {{ if not .entries }}
  1073. <p class="alert">{{ t "There is no unread article." }}</p>
  1074. {{ else }}
  1075. <div class="items hide-read-items">
  1076. {{ range .entries }}
  1077. <article class="item touch-item item-status-{{ .Status }}" data-id="{{ .ID }}">
  1078. <div class="item-header">
  1079. <span class="item-title">
  1080. {{ if ne .Feed.Icon.IconID 0 }}
  1081. <img src="{{ route "icon" "iconID" .Feed.Icon.IconID }}" width="16" height="16" alt="{{ .Feed.Title }}">
  1082. {{ end }}
  1083. <a href="{{ route "unreadEntry" "entryID" .ID }}">{{ .Title }}</a>
  1084. </span>
  1085. <span class="category"><a href="{{ route "categoryEntries" "categoryID" .Feed.Category.ID }}">{{ .Feed.Category.Title }}</a></span>
  1086. </div>
  1087. {{ template "item_meta" dict "user" $.user "entry" . "hasSaveEntry" $.hasSaveEntry }}
  1088. </article>
  1089. {{ end }}
  1090. </div>
  1091. {{ template "pagination" .pagination }}
  1092. {{ end }}
  1093. {{ end }}`,
  1094. "users": `{{ define "title"}}{{ t "Users" }}{{ end }}
  1095. {{ define "content"}}
  1096. <section class="page-header">
  1097. <h1>{{ t "Users" }}</h1>
  1098. <ul>
  1099. <li>
  1100. <a href="{{ route "settings" }}">{{ t "Settings" }}</a>
  1101. </li>
  1102. <li>
  1103. <a href="{{ route "integrations" }}">{{ t "Integrations" }}</a>
  1104. </li>
  1105. <li>
  1106. <a href="{{ route "sessions" }}">{{ t "Sessions" }}</a>
  1107. </li>
  1108. <li>
  1109. <a href="{{ route "createUser" }}">{{ t "Add user" }}</a>
  1110. </li>
  1111. <li>
  1112. <a href="{{ route "about" }}">{{ t "About" }}</a>
  1113. </li>
  1114. </ul>
  1115. </section>
  1116. {{ if eq (len .users) 1 }}
  1117. <p class="alert">{{ t "You are the only user." }}</p>
  1118. {{ else }}
  1119. <table>
  1120. <tr>
  1121. <th class="column-20">{{ t "Username" }}</th>
  1122. <th>{{ t "Administrator" }}</th>
  1123. <th>{{ t "Last Login" }}</th>
  1124. <th>{{ t "Actions" }}</th>
  1125. </tr>
  1126. {{ range .users }}
  1127. {{ if ne .ID $.user.ID }}
  1128. <tr>
  1129. <td>{{ .Username }}</td>
  1130. <td>{{ if eq .IsAdmin true }}{{ t "Yes" }}{{ else }}{{ t "No" }}{{ end }}</td>
  1131. <td>
  1132. {{ if .LastLoginAt }}
  1133. <time datetime="{{ isodate .LastLoginAt }}" title="{{ isodate .LastLoginAt }}">{{ elapsed $.user.Timezone .LastLoginAt }}</time>
  1134. {{ else }}
  1135. {{ t "Never" }}
  1136. {{ end }}
  1137. </td>
  1138. <td>
  1139. <a href="{{ route "editUser" "userID" .ID }}">{{ t "Edit" }}</a>,
  1140. <a href="#"
  1141. data-confirm="true"
  1142. data-label-question="{{ t "Are you sure?" }}"
  1143. data-label-yes="{{ t "yes" }}"
  1144. data-label-no="{{ t "no" }}"
  1145. data-label-loading="{{ t "Work in progress..." }}"
  1146. data-url="{{ route "removeUser" "userID" .ID }}">{{ t "Remove" }}</a>
  1147. </td>
  1148. </tr>
  1149. {{ end }}
  1150. {{ end }}
  1151. </table>
  1152. {{ end }}
  1153. {{ end }}
  1154. `,
  1155. }
  1156. var templateViewsMapChecksums = map[string]string{
  1157. "about": "ad2fb778fc73c39b733b3f81b13e5c7d689b041fadd24ee2d4577f545aa788ad",
  1158. "add_subscription": "3fbcffefc94fb0fccfcf870d602f5ba78ce3ab7ebaeacd04198a6e529143cb29",
  1159. "bookmark_entries": "49423f84c05d77368e20c8e14c53ad237308cdaf4143413487d1b0e11c18d148",
  1160. "categories": "ca1280cd157bb527d4fc907da67b05a8347378f6dce965b9389d4bcdf3600a11",
  1161. "category_entries": "6d8f790b61abdf56edd6a6c88c9bf1eff34424fb47fee0463ee7cb198b34e309",
  1162. "choose_subscription": "7266b269ddbe145e757a24a57f3fbc7611e34a20383fbd887988204cebce2681",
  1163. "create_category": "2b82af5d2dcd67898dc5daa57a6461e6ff8121a6089b2a2a1be909f35e4a2275",
  1164. "create_user": "1ef0a1f9bf119d44929c81f13073a257d69650cf5064960cf06a63fe51923e86",
  1165. "edit_category": "cee720faadcec58289b707ad30af623d2ee66c1ce23a732965463250d7ff41c5",
  1166. "edit_feed": "992e315c244da588d5e9e6912d12d933534c2a86e8e771e7dbd5908564abf616",
  1167. "edit_user": "7373e09f805e6c017167001519b9feb04226be6c81c2875cbacd5ce94f2c24bf",
  1168. "entry": "c838ae7f19d775d473ee92c4999eb28a5063869411c834d26a3f4b686ce88dbf",
  1169. "feed_entries": "76e7e32cdd6552304e38931459fe12cefdb5354f65a7ac3bbed52a2979896d3e",
  1170. "feeds": "1006698abfe0962b656f27794bc44568515392da72b6fac0c03316de06024237",
  1171. "history_entries": "f94e15d37d7604500cede7b583e03bf79c06be81c6597a4a43693f5712af2e13",
  1172. "import": "73b5112e20bfd232bf73334544186ea419505936bc237d481517a8622901878f",
  1173. "integrations": "20c1c82070b93235d189b10acccd0cda5694cc5684d0b3be23de2ba5ae83e73f",
  1174. "login": "7d83c3067c02f1f6aafdd8816c7f97a4eb5a5a4bdaaaa4cc1e2fbb9c17ea65e8",
  1175. "search_entries": "a6e69c3edf414558541e8a23bf197d7580b043f8ddaf53a5b609bcd678fd6f3d",
  1176. "sessions": "3fa79031dd883847eba92fbafe5f535fa3a4e1614bb610f20588b6f8fc8b3624",
  1177. "settings": "d435dc37e82896ce9a7a573b3c2aeda1db71eec62349e2472ebbf1d5c3e0bc21",
  1178. "unread_entries": "adbddbdd0ce70f5c622a2a2687b261121e397e15536dcb551f71bb11550ca46b",
  1179. "users": "c6d91b0b29984b4cb3073bec6a2933cfb72981ec60f54b6c7aa05194f0e860bd",
  1180. }