blog.html 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589
  1. {% load static %}{% load day_util %}
  2. <!DOCTYPE html>
  3. <html>
  4. <head>
  5. <meta charset="utf-8"/>
  6. <meta content="IE=edge,chrome=1" http-equiv="X-UA-Compatible"/>
  7. <meta
  8. name="viewport"
  9. content="width=device-width, initial-scale=1, maximum-scale=2, user-scalable=no"
  10. />
  11. <meta name="apple-mobile-web-app-capable" content="yes"/>
  12. <meta name="robots" content="index,follow"/>
  13. <meta
  14. name="description"
  15. content="{{ article.intro }}"
  16. />
  17. <meta name="keywords" content="{{ article.title }}"/>
  18. <meta name="author" content="CyberSicko"/>
  19. <meta name="theme-color" content="#ffffff"/>
  20. <title>{{ article.title }}-CyberSicko.net</title>
  21. <link rel="preconnect" href="https://fonts.googleapis.com">
  22. <link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
  23. <link href="https://fonts.googleapis.com/css2?family=Noto+Serif+SC:wght@200;300;400;500;600;700;900&display=swap"
  24. rel="stylesheet">
  25. <link
  26. rel="stylesheet"
  27. href="{% static 'semantic.min.css' %}"
  28. type="text/css"
  29. />
  30. <link
  31. rel="stylesheet"
  32. href="{% static 'photoswipe/photoswipe.css' %}"
  33. type="text/css"
  34. />
  35. <link rel="stylesheet" href="{% static 'highlightjs/styles/androidstudio.min.css' %}">
  36. <link rel="stylesheet" href="{% static 'css/fonts.css' %}">
  37. <link rel="shortcut icon" href="{% static 'favicon.ico' %}">
  38. <style>
  39. :root {
  40. --primary: #000000;
  41. --bg-light: #f8f8f8;
  42. --text: #333;
  43. --gray: #666;
  44. --line-height: 1.8;
  45. --max-width: 900px;
  46. --sidebar-width: 300px;
  47. }
  48. body {
  49. color: var(--text);
  50. background: whitesmoke;
  51. line-height: var(--line-height);
  52. }
  53. a {
  54. color: var(--primary);
  55. transition: color .2s;
  56. }
  57. a:hover {
  58. color: darken(var(--primary), 10%);
  59. }
  60. /* 主容器:限制最大宽度并居中 */
  61. .ui.grid.stackable.container {
  62. max-width: var(--max-width);
  63. margin: 2rem auto;
  64. padding: 0 1rem;
  65. }
  66. /* 段落与标题留白 */
  67. p {
  68. margin-bottom: 1rem;
  69. }
  70. h2.ui.header {
  71. margin-top: 2rem;
  72. margin-bottom: 1rem;
  73. }
  74. /* 卡片式内容区 */
  75. .eleven.wide.column {
  76. background: #fff;
  77. padding: 2rem;
  78. box-shadow: 0 2px 8px rgba(0, 0, 0, 0.05);
  79. border-radius: 4px;
  80. }
  81. /* 代码块滚动与样式 */
  82. .pre {
  83. overflow-x: auto;
  84. padding: 1rem;
  85. background: #fafafa;
  86. border-left: 3px solid var(--primary);
  87. margin: 1.5rem 0;
  88. }
  89. /* 顶部菜单去掉死板的灰色,增加一点阴影 */
  90. .ui.top.fixed.menu {
  91. background: #fff;
  92. box-shadow: 0 1px 4px rgba(0, 0, 0, 0.1);
  93. }
  94. /* 菜单项内边距更舒适 */
  95. .ui.menu .item {
  96. padding: 0.8rem 1.2rem;
  97. font-size: 1rem;
  98. }
  99. /* 移动端:展开动画更顺滑 */
  100. .ui.vertical.menu {
  101. transition: max-height .3s ease-out;
  102. overflow: hidden;
  103. }
  104. .ui.vertical.menu.collapsed {
  105. max-height: 0 !important;
  106. }
  107. .ui.vertical.menu.expanded {
  108. max-height: 500px !important;
  109. }
  110. /* 右侧 Archiv/Top 区域在大屏下保持粘性 */
  111. .four.wide.right.floated.column {
  112. position: sticky;
  113. top: 80px; /* 根据你的导航高度微调 */
  114. max-width: var(--sidebar-width);
  115. margin-left: 2rem;
  116. }
  117. /* 调整归档和标签列表间距 */
  118. .ui.list .item {
  119. padding: .5rem 0;
  120. }
  121. </style>
  122. <style type="text/css">
  123. p > a > img {
  124. width: 100%;
  125. height: 100%;
  126. }
  127. blockquote {
  128. margin: 0;
  129. }
  130. blockquote {
  131. padding: 15px;
  132. background: #eee;
  133. }
  134. blockquote ::before {
  135. content: '\201C';
  136. }
  137. blockquote ::after {
  138. content: '\201D';
  139. }
  140. p {
  141. font-size: 1.2em;
  142. }
  143. body {
  144. padding-top: 54px;
  145. -webkit-font-smoothing: antialiased;
  146. -moz-font-smoothing: grayscale;
  147. }
  148. .ui.borderless.menu {
  149. background-color: #f8f8f8;
  150. {#box-shadow: none;#} flex-wrap: wrap;
  151. border: none;
  152. padding-left: 0;
  153. padding-right: 0;
  154. }
  155. .ui.borderless.menu .header.item {
  156. font-size: 18px;
  157. font-weight: 400;
  158. }
  159. .ui.mobile.only.grid .ui.menu .ui.vertical.menu {
  160. display: none;
  161. }
  162. .ui.mobile.only.grid .ui.vertical.menu .dropdown.icon {
  163. float: unset;
  164. }
  165. .ui.mobile.only.grid .ui.vertical.menu .dropdown.icon:before {
  166. content: "\f0d7";
  167. }
  168. .ui.mobile.only.grid .ui.vertical.menu .ui.dropdown.item .menu {
  169. position: static;
  170. width: 100%;
  171. background-color: unset;
  172. border: none;
  173. box-shadow: none;
  174. }
  175. .ui.mobile.only.grid .ui.vertical.menu .ui.dropdown.item .menu {
  176. margin-top: 6px;
  177. }
  178. .ui.container > .ui.message {
  179. background-color: rga(238, 238, 238);
  180. box-shadow: none;
  181. padding: 5rem 4rem;
  182. margin-top: 1rem;
  183. }
  184. .ui.message h1.ui.header {
  185. font-size: 4.5rem;
  186. }
  187. .ui.message p.lead {
  188. font-size: 1.3rem;
  189. color: #333333;
  190. line-height: 1.4;
  191. font-weight: 300;
  192. }
  193. .pre {
  194. background-color: antiquewhite;
  195. border-radius: 4px;
  196. }
  197. </style>
  198. </head>
  199. <body id="root">
  200. {% if is_login %}
  201. {% csrf_token %}
  202. {% endif %}
  203. <div class="ui tablet computer only padded grid">
  204. <div class="ui top fixed borderless fluid huge menu">
  205. <div class="ui container">
  206. <a href="/" class="header item ">CyberSicko.net</a>
  207. <a href="/" class="item">主页</a>
  208. {% for item in category %}
  209. {# <li class="category_li"><a href="/category/{{ item.name }}?page=1">{{ item.name }}</a></li>#}
  210. <a class="{% if current_category != None and current_category == item.name %}
  211. active
  212. {% else %}
  213. {% endif %} item" href="/category/{{ item.name }}?page=1">{{ item.name }}</a>
  214. {% endfor %}
  215. <div class="right menu">
  216. <a href="/login" class="item"> <i class="icon share square"></i>
  217. Sign in</a>
  218. </div>
  219. </div>
  220. </div>
  221. </div>
  222. <div class="ui mobile only padded grid">
  223. <div class="ui top fixed borderless huge fluid menu">
  224. <a href="/" class="header item ">CyberSicko.net</a>
  225. <div class="right menu">
  226. <div class="item">
  227. <button class="ui icon toggle basic button">
  228. <i class="content icon"></i>
  229. </button>
  230. </div>
  231. </div>
  232. <div class="ui vertical borderless fluid menu">
  233. <a href="/" class="item">主页</a>
  234. {% for item in category %}
  235. {# <li class="category_li"><a href="/category/{{ item.name }}?page=1">{{ item.name }}</a></li>#}
  236. <a class="{% if current_category != None and current_category == item.name %}
  237. active
  238. {% else %}
  239. {% endif %} item" href="/category/{{ item.name }}?page=1">{{ item.name }}</a>
  240. {% endfor %}
  241. <a href="/login" class="item"> <i class="icon share square"></i>
  242. Sign in</a>
  243. </div>
  244. </div>
  245. </div>
  246. <div class="ui grid stackable container">
  247. <div class="row" id="article">
  248. <div class="eleven wide column">
  249. <h1 class="ui large header">
  250. <div class="content">{{ article.title }}</div>
  251. </h1>
  252. <div class="ui comments" style="margin-top: 0">
  253. <div class="comment">
  254. <a class="avatar">
  255. <img src="/user_avatar/{{ article.user_id }}">
  256. </a>
  257. <div class="content">
  258. <a class="author">{{ article.user__first_name }}{{ article.user__last_name }}</a>
  259. <div class="metadata">
  260. <div class="date">发布于: {{ article.created_time|days_until }}</div>
  261. </div>
  262. <div class="text">hava a nice day.</div>
  263. </div>
  264. </div>
  265. </div>
  266. {% autoescape off %}
  267. {{ article.html_text }}
  268. {% endautoescape %}
  269. <h3 class="ui dividing header" style="margin-top: 10vh">评论</h3>
  270. <div class="ui comments">
  271. {% if comments|length <= 0 %}
  272. <p style="text-align: center;font-style: italic">还没有评论</p>
  273. {% endif %}
  274. {% for comment in comments %}
  275. <div class="comment">
  276. <a class="avatar">
  277. <img src="{{ comment.avatar | get_avatar }}">
  278. </a>
  279. <div class="content">
  280. <a target="_blank" href="{{ comment.website }}" class="author">{{ comment.nick_name }}</a>
  281. <div class="text">{{ comment.comment }}</div>
  282. <div class="actions">
  283. <a href="mailto:{{ comment.email }}"> <i class="mail icon"></i>联系他</a>
  284. <a>
  285. <i class="time icon"></i>
  286. {{ comment.created_time }}
  287. </a>
  288. </div>
  289. </div>
  290. </div>
  291. {% endfor %}
  292. </div>
  293. <form method="post" action="/comment/add" id="comment_form" style="margin-top: 10px"
  294. class="ui form">
  295. {% csrf_token %}
  296. <input hidden="hidden" name="article_id" value="{{ article.id }}">
  297. <input hidden="hidden" name="article_title" value="{{ article.title }}">
  298. <h4 class="ui dividing header">发表评论</h4>
  299. <input id="veryCode" type="text" name="veryCode" hidden="hidden">
  300. <div class="field">
  301. <div class="two fields">
  302. <div class="field">
  303. <label>昵称<span style="color: red">*</span></label>
  304. <input id="nick_name" type="text" name="nick_name" placeholder="昵称">
  305. </div>
  306. <div class="field">
  307. <label>Email<span style="color: red">*</span></label>
  308. <input id="email" type="email" name="email" placeholder="Email">
  309. </div>
  310. </div>
  311. </div>
  312. <div class="field">
  313. <div class="two fields">
  314. <div class="twelve wide field">
  315. <label>网站(可不填)</label>
  316. <input id="website" type="url" name="website" placeholder="网站">
  317. </div>
  318. <div class="four wide field">
  319. <label>头像<span style="color: red">*</span></label>
  320. <div class="ui fluid selection dropdown">
  321. <input value="jenny" type="hidden" name="avatar">
  322. <i class="dropdown icon"></i>
  323. <div class="default text">
  324. Jenny Hess
  325. </div>
  326. <div class="menu">
  327. <div class="item" data-value="jenny">
  328. <img class="ui mini avatar image"
  329. src="https://semantic-ui.com/images/avatar/small/jenny.jpg">
  330. Jenny Hess
  331. </div>
  332. <div class="item" data-value="elliot">
  333. <img class="ui mini avatar image"
  334. src="https://semantic-ui.com/images/avatar/small/elliot.jpg">
  335. Elliot Fu
  336. </div>
  337. <div class="item" data-value="stevie">
  338. <img class="ui mini avatar image"
  339. src="https://semantic-ui.com/images/avatar/small/stevie.jpg">
  340. Stevie Feliciano
  341. </div>
  342. <div class="item" data-value="christian">
  343. <img class="ui mini avatar image"
  344. src="https://semantic-ui.com/images/avatar/small/christian.jpg">
  345. Christian
  346. </div>
  347. <div class="item" data-value="matt">
  348. <img class="ui mini avatar image"
  349. src="https://semantic-ui.com/images/avatar/small/matt.jpg">
  350. Matt
  351. </div>
  352. <div class="item" data-value="justen">
  353. <img class="ui mini avatar image"
  354. src="https://semantic-ui.com/images/avatar/small/justen.jpg">
  355. Justen Kitsune
  356. </div>
  357. </div>
  358. </div>
  359. </div>
  360. </div>
  361. </div>
  362. <div class="field">
  363. <textarea name="comment" placeholder="评论"></textarea>
  364. </div>
  365. <div style="color: red">{{ msg }}</div>
  366. <button class="ui blue labeled submit icon button">
  367. <i class="icon edit"></i> 提交
  368. </button>
  369. </form>
  370. <div class="ui hidden divider"></div>
  371. </div>
  372. <div class="four wide right floated column">
  373. <h4 class="ui header">搜索</h4>
  374. <div id="search" class="ui category search">
  375. <div class="ui icon input">
  376. <input class="prompt" type="text" placeholder="搜索">
  377. <i class="search icon"></i>
  378. </div>
  379. </div>
  380. <h4 class="ui header">归档</h4>
  381. <div class="ui list">
  382. {% for record in records %}
  383. <a class="item" href="/date/{{ record.datetime }}?page=1">{{ record.datetime }}
  384. ({{ record.count }})</a>
  385. {% endfor %}
  386. </div>
  387. <h4 class="ui header">Top</h4>
  388. <div class="ui list">
  389. {% for tag in tags %}
  390. <a class="item" href="/tag/{{ tag.name }}?page=1">{{ tag.name }} ({{ tag.count }})</a>
  391. {% endfor %}
  392. </div>
  393. </div>
  394. </div>
  395. </div>
  396. <div class="ui mini test modal">
  397. <div class="header">
  398. 请输入验证码
  399. </div>
  400. <div class="content">
  401. <p id="veryCodeSet"></p>
  402. <div class="ui input">
  403. <input id="veryCodeInput" type="text" placeholder="请输入验证码">
  404. </div>
  405. </div>
  406. <div class="actions">
  407. <a onclick="submitComment()" id="confirmBtn" class="ui positive right labeled icon button">
  408. 确认
  409. <i class="checkmark icon"></i>
  410. </a>
  411. </div>
  412. </div>
  413. <footer style="margin-top: 50px" class="ui secondary segment">
  414. <div class="ui two column stackable grid">
  415. <div class="ten wide column">
  416. <img style="width: 8rem" src="{% static '/img.png' %}">
  417. <p> © 2023 - 2023 cybersicko.net - All Rights Reserved.</p>
  418. </div>
  419. <div class="six wide column">
  420. <div class="ui left aligned container" style="margin-top:10px">
  421. <a target="_blank" href="http://www.beian.gov.cn/portal/registerSystemInfo?recordcode=50019002503732"
  422. style="display:inline-block;text-decoration:none;height:20px;line-height:20px;"><img
  423. src="{% static '/备案图标.png' %}" style="float:left;"/>
  424. <p style="float:left;height:20px;line-height:20px;margin: 0px 0px 0px 5px; color:#939393;"> 渝公网安备
  425. 50019002503732号</p></a>
  426. <a rel="nofollow"
  427. href="http://beian.miit.gov.cn/publish/query/indexFirst.action"
  428. target="_blank"><p>皖ICP备2023011943号</p></a>
  429. </div>
  430. </div>
  431. </div>
  432. </footer>
  433. <script type="application/javascript" src="{% static 'js/jquery.min.js' %}"></script>
  434. <script type="application/javascript" src="{% static 'semantic.js' %}"></script>
  435. <script src="{% static 'highlightjs/highlight.min.js' %}"></script>
  436. <script type="application/javascript" src="{% static 'photoswipe/photoswipe.umd.min.js' %}"></script>
  437. <script type="application/javascript" src="{% static 'photoswipe/photoswipe-lightbox.umd.min.js' %}"></script>
  438. <script>
  439. restoreComment()
  440. function restoreComment() {
  441. if (localStorage.getItem('nick_name') !== undefined && localStorage.getItem('nick_name') !== '') {
  442. $("#nick_name").val(localStorage.getItem('nick_name'))
  443. }
  444. if (localStorage.getItem('email') !== undefined && localStorage.getItem('email') !== '') {
  445. $("#email").val(localStorage.getItem('email'))
  446. }
  447. if (localStorage.getItem('website') !== undefined && localStorage.getItem('website') !== '') {
  448. $("#website").val(localStorage.getItem('website'))
  449. }
  450. }
  451. function submitComment() {
  452. var veryCodeInput = $("#veryCodeInput").val()
  453. $("#veryCode").attr("value", veryCodeInput)
  454. document.getElementById("comment_form").submit()
  455. }
  456. document.getElementById("comment_form").addEventListener("submit", function (event) {
  457. event.preventDefault(); // 阻止表单提交
  458. // 表单验证逻辑
  459. var veryCode = $('#veryCode').val()
  460. if (veryCode === undefined || veryCode === '') {
  461. getVeryCode()
  462. return
  463. }
  464. var data = $('#comment_form').serializeArray();
  465. localStorage.setItem('nick_name', $('#nick_name').val())
  466. localStorage.setItem('email', $('#email').val())
  467. localStorage.setItem('website', $('#website').val())
  468. return true
  469. // 其他处理逻辑
  470. });
  471. </script>
  472. <script>
  473. function getVeryCode() {
  474. $.ajax('/comment/getVeryCode', {
  475. method: 'get',
  476. contentType: false,
  477. processData: false,
  478. success: function (res) {
  479. $('.mini.modal').modal('show')
  480. $("#veryCodeSet").html("请输入数字: " + res.data)
  481. },
  482. error: function (err) {
  483. }
  484. })
  485. }
  486. $(document).ready(function () {
  487. $(".ui.toggle.button").click(function () {
  488. $(".mobile.only.grid .ui.vertical.menu").toggle(100);
  489. });
  490. $(".ui.dropdown").dropdown();
  491. });
  492. let imgs = $('#article').find('p>img')
  493. function getClass(width) {
  494. if (width > 35 && width < 150) {
  495. return 'ui small rounded image'
  496. } else if (width > 150 && width < 800) {
  497. return 'ui medium rounded image'
  498. } else if (width > 800) {
  499. return 'ui fluid rounded image'
  500. }
  501. }
  502. for (let i = 0; i < imgs.length; i++) {
  503. if ($(imgs[i]).attr('alt')) {
  504. let alt = $(imgs[i]).attr('alt')
  505. let src = $(imgs[i]).attr('src')
  506. $(imgs[i]).parent().html(`<a data-pswp-width="${alt.split(',')[0]}" data-pswp-height="${alt.split(',')[1]}" href="${src}"><img style="display: inline-block;" class="${getClass(alt.split(',')[0])}" width="${alt.split(',')[0]}" height="${alt.split(',')[1]}" data-magnify="gallery" data-src="${src}" src="${src}" data-href="${src}" "/></a>`)
  507. } else {
  508. let src = $(imgs[i]).attr('src')
  509. $(imgs[i]).parent().html(`<img style="display: inline-block;" class="ui fluid rounded image" src="${src}" />`)
  510. }
  511. }
  512. var lightbox = new PhotoSwipeLightbox({
  513. gallery: 'p',
  514. children: 'a[data-pswp-width]',
  515. // dynamic import is not supported in UMD version
  516. pswpModule: PhotoSwipe
  517. });
  518. lightbox.init();
  519. </script>
  520. <script>
  521. hljs.initHighlightingOnLoad();
  522. </script>
  523. </body>
  524. </html>