Form.svelte 3.2 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. <script lang="ts">
  2. import { getContext } from 'svelte';
  3. import snarkdown from 'snarkdown';
  4. import insane from 'insane';
  5. export let parentId = "";
  6. export let comments = [];
  7. export let count: number;
  8. export let formOpened = true;
  9. let showPreview = false;
  10. const reqUrl:string = getContext("reqUrl");
  11. let newComment = {
  12. uri: getContext("pageUri"),
  13. author: "",
  14. email: "",
  15. website: "",
  16. content: "",
  17. parent: parentId
  18. };
  19. async function sendComment() {
  20. const submitBtn = this.querySelector('button[type="submit"]');
  21. submitBtn.disabled = true;
  22. const response = await fetch(reqUrl, {
  23. method: "POST",
  24. headers: {
  25. "Content-Type": "application/json; charset=UTF-8",
  26. },
  27. body: JSON.stringify(newComment)
  28. });
  29. const data = await response.json();
  30. // Put new comment to page
  31. count++
  32. if (parentId === "") {
  33. comments = [{
  34. id: data.id,
  35. author: data.author,
  36. avatar: data.avatar,
  37. website: data.website,
  38. content: data.content,
  39. created: data.created,
  40. reply: []
  41. }, ...comments];
  42. } else {
  43. const index = comments.findIndex(e => e.id === parentId)
  44. comments[index].reply.push({
  45. id: data.id,
  46. author: data.author,
  47. avatar: data.avatar,
  48. website: data.website,
  49. content: data.content,
  50. created: data.created,
  51. });
  52. };
  53. newComment.content = "";
  54. submitBtn.disabled = false;
  55. formOpened=false;
  56. };
  57. </script>
  58. <form class="comment-form" on:submit|preventDefault={sendComment}>
  59. <textarea name="content" placeholder="欢迎评论……(支持 Markdown 语法)" rows="6" bind:value={newComment.content} required></textarea>
  60. {#if showPreview}
  61. <div class="comment-preview">
  62. {@html insane(snarkdown(newComment.content))}
  63. </div>
  64. {/if}
  65. <div class=form-wrapper>
  66. <label for="name">
  67. 名字<span class="required" aria-hidden="true">*</span>
  68. <input type="text" name="author" placeholder="John Doe" bind:value={newComment.author} required>
  69. </label>
  70. <label for="email">
  71. 邮箱<span class="required" aria-hidden="true">*</span>
  72. <input type="email" name="email" placeholder="someone@example.com" bind:value={newComment.email} required>
  73. </label>
  74. </div>
  75. <label for="website">
  76. 网址
  77. <input type="url" name="website" placeholder="https://example.com" bind:value={newComment.website}>
  78. </label>
  79. <div class=form-wrapper>
  80. <button type="button" on:click={()=>{showPreview = !showPreview}}>
  81. <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="feather feather-eye"><path d="M1 12s4-8 11-8 11 8 11 8-4 8-11 8-11-8-11-8z"></path><circle cx="12" cy="12" r="3"></circle></svg> 预览
  82. </button>
  83. <button type="submit">
  84. <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="feather feather-send"><line x1="22" y1="2" x2="11" y2="13"></line><polygon points="22 2 15 22 11 13 2 9 22 2"></polygon></svg> 发送
  85. </button>
  86. </div>
  87. </form>