Courses.vue 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086
  1. <script setup>
  2. import { ref, reactive, watch } from "vue";
  3. import { useMainStore } from "@/stores/store";
  4. import { Loader } from "@googlemaps/js-api-loader";
  5. import { VDataTable } from "vuetify/labs/VDataTable";
  6. import axios from "axios";
  7. import moment from "moment";
  8. const store = useMainStore();
  9. let token = store.token;
  10. let loading = ref(false);
  11. let courses = reactive({
  12. list: [],
  13. });
  14. let classes = reactive({
  15. list: [],
  16. });
  17. // 取得開課紀錄
  18. async function getClass() {
  19. console.log("getClass");
  20. loading.value = true;
  21. let token = store.token;
  22. try {
  23. let response = await axios.get(
  24. `https://cmm.ai:8088/api/get_class_name?access_token=${token}`
  25. );
  26. classes.list = response.data.classes;
  27. // let coursesData = response.data.classes[0];
  28. // classNameId.value = coursesData.class_name_id;
  29. // 存入課程資料
  30. // for (const key in coursesData) {
  31. // if (course.hasOwnProperty(key) && coursesData[key] !== undefined) {
  32. // course[key] = coursesData[key];
  33. // }
  34. // if (coursesData["cover_img"] !== "") {
  35. // coverImgUrl.value = `https://ntcri.org/${coursesData["cover_img"]}`;
  36. // }
  37. // }
  38. // console.log('course',course);
  39. loading.value = false;
  40. console.log("開課資料", classes.list);
  41. } catch (error) {
  42. console.error(error);
  43. }
  44. }
  45. getClass();
  46. // 彈跳視窗狀態
  47. let schoolDialog = ref(false); // 據點
  48. let courseDialog = ref(false); // 課程
  49. let enrolledDialog = reactive([]); // 報名清單
  50. let sessionsDialog = ref(false); // 場次
  51. watch(schoolDialog, () => {
  52. initMap();
  53. });
  54. // 據點參數
  55. let location = reactive({
  56. location_name: "",
  57. Lng: "",
  58. Lat: "",
  59. address: "",
  60. introduction: "",
  61. email: "",
  62. phone: "",
  63. access_token: token,
  64. });
  65. // Google Map
  66. const states = reactive({
  67. google: null,
  68. map: null,
  69. markers: null,
  70. });
  71. const initMap = async () => {
  72. const loader = new Loader({
  73. apiKey: "AIzaSyAzDeviZ-TpwzT1atlnshNJRjBgndP05Mw",
  74. version: "weekly",
  75. libraries: ["places"],
  76. language: "zh-TW",
  77. });
  78. states.google = await loader.load();
  79. let dom = document.getElementById("map");
  80. states.map = new states.google.maps.Map(document.getElementById("map"), {
  81. center: { lat: 25.0425, lng: 121.5468 },
  82. zoom: 11,
  83. mapTypeControl: false,
  84. fullscreenControl: false,
  85. });
  86. };
  87. // 查詢 Google 地圖
  88. const getCoordinates = async () => {
  89. const geocoder = new states.google.maps.Geocoder();
  90. geocoder.geocode({ address: location.address }, (results, status) => {
  91. if (status === states.google.maps.GeocoderStatus.OK) {
  92. location.Lat = results[0].geometry.location.lat();
  93. location.Lng = results[0].geometry.location.lng();
  94. }
  95. // 將地圖中心設為取得的經緯度,並調整縮放等級
  96. states.map.setCenter({ lat: location.Lat, lng: location.Lng });
  97. states.map.setZoom(15);
  98. // 設定地址圖標
  99. const marker = new google.maps.Marker({
  100. position: { lat: location.Lat, lng: location.Lng },
  101. map: states.map,
  102. title: location.address,
  103. icon: store.getImageUrl("map-icon/icon_house05.png"),
  104. });
  105. });
  106. };
  107. // 更新按鈕狀態
  108. function updateButtonStatus(name) {
  109. if (name === "location") {
  110. locationBtnLoading.value = false;
  111. locationBtnText.value = "儲存成功!";
  112. setTimeout(() => {
  113. // schoolDialog.value = false; // 關閉彈跳視窗
  114. locationBtnText.value = "儲存修改";
  115. }, 3000);
  116. } else if (name === "course") {
  117. courseBtnLoading.value = false;
  118. courseBtnText.value = "儲存成功!";
  119. setTimeout(() => {
  120. // courseDialog.value = false; // 關閉彈跳視窗
  121. courseBtnText.value = "儲存修改";
  122. }, 3000);
  123. }
  124. }
  125. let schoolData = reactive({});
  126. let locationId = ref(null);
  127. // 取得據點
  128. async function getSchool(id) {
  129. console.log("getSchool id", id);
  130. locationId.value = id;
  131. try {
  132. const response = await axios.get(
  133. // `https://cmm.ai:8088/api/get_school?location_id=${id}&access_token=${token}` // 暫時不帶 token
  134. `https://cmm.ai:8088/api/get_school?location_id=${id}`
  135. );
  136. schoolData = response.data.schools[0];
  137. console.log("據點", schoolData);
  138. for (const key in schoolData) {
  139. if (location.hasOwnProperty(key) && schoolData[key] !== undefined) {
  140. location[key] = schoolData[key];
  141. }
  142. if (key === "school_introduction") {
  143. location["introduction"] = schoolData["school_introduction"];
  144. }
  145. }
  146. // 設定地圖
  147. getCoordinates();
  148. } catch (error) {
  149. console.error(error);
  150. }
  151. }
  152. // 儲存按鈕狀態
  153. let locationBtnLoading = ref(false);
  154. let locationBtnText = ref("儲存修改");
  155. // 更新據點
  156. async function saveLocation() {
  157. locationBtnLoading.value = true;
  158. let data = reactive({
  159. location_id: locationId.value,
  160. location_name: "",
  161. Lng: "",
  162. Lat: "",
  163. address: "",
  164. introduction: "",
  165. email: "",
  166. phone: "",
  167. });
  168. for (const key in location) {
  169. if (data.hasOwnProperty(key) && location[key] !== undefined) {
  170. data[key] = location[key];
  171. }
  172. }
  173. const formData = new FormData();
  174. for (const key in data) {
  175. formData.append(key, data[key]);
  176. }
  177. try {
  178. await axios.post("https://cmm.ai:8088/api/update_school", formData);
  179. setTimeout(() => {
  180. updateButtonStatus("location");
  181. }, 500);
  182. } catch (error) {
  183. console.error(error);
  184. }
  185. }
  186. // 課程參數
  187. let course = reactive({
  188. name: "", // 課程名稱
  189. location_id: "", // 據點編號
  190. category: "", // 工藝類別
  191. introduction: "", // 課程簡介
  192. organizer: "", // 主辦單位
  193. cover_img_file: "", // 課程圖片
  194. group_id: 2, // 學群編號(技藝)
  195. group_sort: "", // 學群細分
  196. special_class_list_name: "",
  197. recommend: 0, // 是否推薦
  198. is_inner: 1, // 內課課程
  199. is_check: 0, // 審核結果
  200. address: "", // 上課地點
  201. access_token: token,
  202. });
  203. let classNameId = ref(null);
  204. let courseLoading = ref(false);
  205. let coverImgLoading = ref(false);
  206. // 取得課程
  207. async function getCourse(id) {
  208. classNameId.value = id;
  209. console.log("取得課程", id);
  210. courseLoading.value = true;
  211. let token = store.token;
  212. try {
  213. let response = await axios.get(
  214. `https://cmm.ai:8088/api/get_class_name?class_name_id=${id}&access_token=${token}`
  215. );
  216. // console.log('課程資料',response);
  217. courses.list = response.data.classes;
  218. console.log("課程資料", courses.list);
  219. let coursesData = response.data.classes[0];
  220. console.log("coursesData", coursesData);
  221. // classNameId.value = coursesData.class_name_id;
  222. // 存入課程資料
  223. for (const key in coursesData) {
  224. if (course.hasOwnProperty(key) && coursesData[key] !== undefined) {
  225. course[key] = coursesData[key];
  226. }
  227. if (coursesData["cover_img"] !== "") {
  228. coverImgLoading.value = true;
  229. coverImgUrl.value = `https://ntcri.org/${coursesData["cover_img"]}`;
  230. setTimeout(() => {
  231. coverImgLoading.value = false;
  232. }, 1500);
  233. }
  234. }
  235. console.log("最終 course", course);
  236. loading.value = false;
  237. } catch (error) {
  238. console.error(error);
  239. }
  240. }
  241. // 儲存按鈕狀態
  242. let courseBtnLoading = ref(false);
  243. let courseBtnText = ref("儲存修改");
  244. // 更新課程
  245. async function saveCourse() {
  246. courseBtnLoading.value = true;
  247. delete course.access_token;
  248. course["class_name_id"] = classNameId.value;
  249. if (coverImg.value) {
  250. course["cover_img_file"] = coverImg.value;
  251. }
  252. const formData = new FormData();
  253. for (const key in course) {
  254. formData.append(key, course[key]);
  255. }
  256. try {
  257. await axios.post("https://cmm.ai:8088/api/update_class_name", formData);
  258. setTimeout(() => {
  259. updateButtonStatus("course");
  260. }, 500);
  261. } catch (error) {
  262. console.error(error);
  263. }
  264. }
  265. // 上傳圖片 Input
  266. const coverImgRef = ref(null);
  267. const fileInputClick = (ref) => {
  268. if (ref === "cover") {
  269. if (coverImgRef.value) {
  270. coverImgRef.value[0].click();
  271. }
  272. }
  273. };
  274. // 封面圖片
  275. let coverImg = ref("");
  276. let coverImgUrl = ref("");
  277. const handleCoverImg = (event) => {
  278. const file = event.target.files[0];
  279. if (file) {
  280. coverImg.value = file;
  281. coverImgUrl.value = URL.createObjectURL(file); // 設定圖片預覽 URL
  282. }
  283. };
  284. // 取得場次
  285. async function getEvent(id) {
  286. console.log("getEvent id", id);
  287. try {
  288. const response = await axios.get(
  289. `https://cmm.ai:8088/api/get_event?class_name_id=${id}`
  290. );
  291. // event_id
  292. console.log("場次資料", response);
  293. let eventId = response.data.classes[0].event_id;
  294. getEnrolledData(eventId);
  295. } catch (error) {
  296. console.error(error);
  297. }
  298. }
  299. let enrolledLoading = ref(false);
  300. let registrations = reactive({
  301. list: [],
  302. });
  303. // 取得報名清單
  304. async function getEnrolledData(eventId) {
  305. enrolledLoading.value = true;
  306. console.log("getEnrolledData eventId", eventId);
  307. try {
  308. const response = await axios.get(
  309. `https://cmm.ai:8088/api/get_registration_class?event_id=${eventId}`
  310. );
  311. console.log("取得報名", response);
  312. registrations.list = response.data.registrations;
  313. console.log("報名清單", registrations.list);
  314. registrations.list.map((item) => {
  315. if (item.payment_status) {
  316. item.payment_status = true;
  317. } else {
  318. item.payment_status = false;
  319. }
  320. });
  321. setTimeout(() => {
  322. enrolledLoading.value = false;
  323. }, 1000);
  324. } catch (error) {
  325. console.error(error);
  326. }
  327. }
  328. // 場次
  329. let eventData = reactive({
  330. list: [],
  331. });
  332. // 場次參數
  333. let event = reactive({
  334. name_id: "", // 課程名稱編號(需先新增課程取得Id)
  335. event: "", // 場次名稱
  336. start_time: "", // 課程起始日
  337. end_time: "", // 課程結束日
  338. contact: "", // 聯絡資訊
  339. lecturer: "", // 課程講師
  340. location: "",
  341. content: "", // 課程內容
  342. URL: "", // 外部連結
  343. people: "", // 對象
  344. fee_method: "", // 收費方式
  345. registration_way: "", // 報名方式
  346. registration_start: "", // 報名起始日
  347. registration_end: "", // 報名截止日
  348. number_limit: "", // 人數限制
  349. remark: "", // 備註
  350. ATM_address: "", // 匯款帳號(銀行代碼+帳號)
  351. access_token: token,
  352. });
  353. let date = reactive({
  354. start_date: "", // 起始日期
  355. start_time: "", // 起始時間
  356. end_date: "", // 結束日期
  357. end_time: "", // 結束時間
  358. registration_start_date: "", // 報名起始日期
  359. registration_start_time: "", // 報名起始時間
  360. registration_end_date: "", // 報名截止日期
  361. registration_end_time: "", // 報名截止時間
  362. });
  363. let itemsPerPage = ref(5); // 每頁顯示筆數(查看出缺席)
  364. let headers = reactive([
  365. {
  366. title: "姓名",
  367. align: "start",
  368. key: "real_name",
  369. },
  370. { title: "電話", key: "phone" },
  371. { title: "信箱", key: "email" },
  372. { title: "出席", key: "enrolled" },
  373. { title: "付款狀態", key: "payment" },
  374. ]);
  375. </script>
  376. <template>
  377. <v-card class="h-100 user-courses">
  378. <div class="title">
  379. <h4>我的開課</h4>
  380. </div>
  381. <div class="d-flex justify-center my-10" v-if="loading">
  382. <v-progress-circular
  383. color="grey-lighten-4"
  384. indeterminate
  385. ></v-progress-circular>
  386. </div>
  387. <div v-else class="main-table mt-3">
  388. <h6 class="table-title">開課紀錄</h6>
  389. <router-link
  390. v-if="!classes.list.length"
  391. to="/setup-courses"
  392. class="hint-item mb-7"
  393. >點此進入開課專區</router-link
  394. >
  395. <table v-else>
  396. <thead>
  397. <tr>
  398. <th></th>
  399. <th>名稱</th>
  400. <th>開課日期</th>
  401. <th width="20%">狀態</th>
  402. <th>出/缺席</th>
  403. <th></th>
  404. </tr>
  405. </thead>
  406. <tbody>
  407. <tr v-for="(item, index) in classes.list" :key="index">
  408. <td>
  409. <v-chip v-if="!item.is_check"> 未審核 </v-chip>
  410. <v-chip
  411. v-else
  412. variant="outlined"
  413. color="green"
  414. text-color="white"
  415. >
  416. 已審核
  417. </v-chip>
  418. </td>
  419. <td>{{ item.name }}</td>
  420. <td>
  421. {{ moment(`${item.update_time}`).format("YYYY/MM/DD") }}
  422. </td>
  423. <td>
  424. 開放報名
  425. <!-- {{ item.state }} -->
  426. </td>
  427. <td>
  428. <v-dialog v-model="enrolledDialog[index]" width="1000px">
  429. <template v-slot:activator="{ props }">
  430. <v-btn
  431. @click="getEvent(item.class_name_id)"
  432. v-bind="props"
  433. color="blue"
  434. variant="flat"
  435. rounded="xl"
  436. class="me-3"
  437. >
  438. <p class="text-white">查看</p>
  439. </v-btn>
  440. </template>
  441. <v-card class="pa-5">
  442. <v-card-text>
  443. <div class="d-flex justify-center" v-if="enrolledLoading">
  444. <v-progress-circular
  445. color="grey-lighten-4"
  446. indeterminate
  447. ></v-progress-circular>
  448. </div>
  449. <v-data-table
  450. v-else
  451. v-model:items-per-page="itemsPerPage"
  452. :headers="headers"
  453. :items="registrations.list"
  454. item-value="name"
  455. class="elevation-1 courses-table"
  456. >
  457. <template v-slot:item.enrolled="{ item }">
  458. <v-checkbox hide-details></v-checkbox>
  459. </template>
  460. <template v-slot:item.payment="{ item }">
  461. <!-- <v-select
  462. :items="['已付款', '未付款']"
  463. density="compact"
  464. variant="outlined"
  465. hide-details
  466. class="my-3"
  467. ></v-select> -->
  468. <div class="d-flex align-center my-2">
  469. <v-checkbox
  470. v-model="item.selectable.payment_status"
  471. color="purple"
  472. hide-details
  473. label="已付款"
  474. ></v-checkbox>
  475. <v-text-field
  476. density="compact"
  477. variant="outlined"
  478. hide-details
  479. label="帳號末五碼"
  480. class="account-item"
  481. :disabled="!item.selectable.payment_status"
  482. ></v-text-field>
  483. </div>
  484. </template>
  485. </v-data-table>
  486. </v-card-text>
  487. <v-card-actions class="d-flex justify-center mt-3">
  488. <v-btn
  489. color="purple"
  490. variant="outlined"
  491. class="me-3"
  492. @click="enrolledDialog[index] = false"
  493. >關閉</v-btn
  494. >
  495. <v-btn variant="flat" color="purple" class="px-6"
  496. >儲存修改</v-btn
  497. >
  498. </v-card-actions>
  499. </v-card>
  500. </v-dialog>
  501. </td>
  502. <td>
  503. <v-menu>
  504. <template v-slot:activator="{ props }">
  505. <v-btn color="blue" v-bind="props" variant="outlined">
  506. <p>編輯</p>
  507. </v-btn>
  508. <v-btn color="blue" variant="outlined" class="ms-3">
  509. <a
  510. :href="
  511. $router.resolve(`/course-detail/${item.class_name_id}`)
  512. .href
  513. "
  514. class="text-blue"
  515. target="_blank"
  516. >前往課程</a
  517. >
  518. <!-- <router-link :to="class_name_id" class="text-white">前往課程</router-link> -->
  519. </v-btn>
  520. </template>
  521. <v-list>
  522. <v-list-item>
  523. <v-list-item-title>
  524. <v-btn
  525. variant="text"
  526. @click="getSchool(item.location_id)"
  527. >
  528. 據點
  529. <v-dialog
  530. v-model="schoolDialog"
  531. activator="parent"
  532. width="auto"
  533. >
  534. <v-card class="pa-5">
  535. <v-card-title class="ps-5 pb-5"
  536. >編輯據點</v-card-title
  537. >
  538. <v-card-text>
  539. <v-label class="d-flex align-center pb-3">
  540. <p class="pb-5 pe-3">
  541. 據點名稱<span class="mark">*</span>
  542. </p>
  543. <v-text-field
  544. v-model="location.location_name"
  545. :rules="[requiredRule]"
  546. placeholder="可以是您的工作室或品牌名稱/教學單位/您的姓名"
  547. density="compact"
  548. variant="outlined"
  549. counter
  550. maxlength="60"
  551. ></v-text-field>
  552. </v-label>
  553. <v-label class="d-flex align-center pb-3 w-100">
  554. <p class="pb-5 pe-3">
  555. 據點地址<span class="mark">*</span>
  556. </p>
  557. <v-text-field
  558. v-model="location.address"
  559. :rules="[requiredRule]"
  560. density="compact"
  561. variant="outlined"
  562. placeholder="需在安全且便於學徒到達之地點開課"
  563. ></v-text-field>
  564. <v-btn
  565. @click="getCoordinates"
  566. color="purple"
  567. variant="flat"
  568. class="ms-3 mb-6 px-8"
  569. >
  570. 查詢
  571. </v-btn>
  572. </v-label>
  573. <div
  574. class="d-flex flex-column justify-end ms-16 ps-5"
  575. >
  576. <div
  577. class="map"
  578. id="map"
  579. style="width: 100%; height: 500px"
  580. ></div>
  581. <v-row class="mt-3">
  582. <v-col cols="12" md="6">
  583. <v-label class="d-flex align-center">
  584. <p class="pb-5 pe-3">經度</p>
  585. <v-text-field
  586. v-model="location.Lng"
  587. density="compact"
  588. variant="outlined"
  589. disabled
  590. ></v-text-field>
  591. </v-label>
  592. </v-col>
  593. <v-col cols="12" md="6">
  594. <v-label class="d-flex align-center">
  595. <p class="pb-5 pe-3">緯度</p>
  596. <v-text-field
  597. v-model="location.Lat"
  598. density="compact"
  599. variant="outlined"
  600. disabled
  601. ></v-text-field>
  602. </v-label>
  603. </v-col>
  604. </v-row>
  605. </div>
  606. <v-label class="d-block">
  607. <p class="d-flex mb-5">
  608. 據點 Email<span class="mark">*</span>
  609. </p>
  610. <v-text-field
  611. v-model="location.email"
  612. :rules="[requiredRule]"
  613. density="compact"
  614. variant="outlined"
  615. placeholder="請填寫 Email"
  616. ></v-text-field>
  617. </v-label>
  618. <v-row class="mt-3">
  619. <v-col cols="12" md="12">
  620. <v-label class="d-block">
  621. <p class="d-flex">
  622. 公開電話<span class="mark">*</span>
  623. </p>
  624. <span class="d-block pt-1 pb-3 hint"
  625. >會顯示於課程介紹中,想了解課程資訊者聯繫用途</span
  626. >
  627. <v-text-field
  628. v-model="location.phone"
  629. :rules="[requiredRule]"
  630. density="compact"
  631. variant="outlined"
  632. ></v-text-field>
  633. </v-label>
  634. </v-col>
  635. <v-col cols="12" md="12">
  636. <v-label class="d-block">
  637. <p class="d-flex mb-5">
  638. 據點簡介<span class="mark">*</span>
  639. </p>
  640. <v-textarea
  641. v-model="location.introduction"
  642. rows="5"
  643. variant="outlined"
  644. ></v-textarea>
  645. </v-label>
  646. </v-col>
  647. </v-row>
  648. </v-card-text>
  649. <v-card-actions class="d-flex justify-center">
  650. <v-btn
  651. variant="outlined"
  652. color="purple"
  653. @click="schoolDialog = false"
  654. class="me-2 px-3"
  655. >關閉</v-btn
  656. >
  657. <v-btn
  658. variant="flat"
  659. color="purple"
  660. class="px-6"
  661. :loading="locationBtnLoading"
  662. @click="saveLocation()"
  663. >{{ locationBtnText }}</v-btn
  664. >
  665. </v-card-actions>
  666. </v-card>
  667. </v-dialog>
  668. </v-btn>
  669. </v-list-item-title>
  670. </v-list-item>
  671. <v-list-item>
  672. <v-list-item-title>
  673. <!-- {{ item.title }} -->
  674. <v-btn
  675. @click="getCourse(item.class_name_id)"
  676. variant="text"
  677. >
  678. 課程
  679. <v-dialog
  680. v-model="courseDialog"
  681. activator="parent"
  682. width="600"
  683. >
  684. <v-card class="pa-5">
  685. <v-card-title class="ps-5 pb-8"
  686. >編輯課程</v-card-title
  687. >
  688. <v-card-text class="mb-7">
  689. <v-row class="justify-space-evenly">
  690. <v-col cols="12" class="py-0">
  691. <v-label class="d-block pb-3">
  692. <p class="pb-5 pe-3">
  693. 課程名稱<span class="mark">*</span>
  694. </p>
  695. <v-text-field
  696. v-model="course.name"
  697. :rules="[requiredRule]"
  698. density="compact"
  699. variant="outlined"
  700. ></v-text-field>
  701. </v-label>
  702. </v-col>
  703. <!-- <v-col cols="12" md="6">
  704. <v-label class="d-block pb-3">
  705. <p class="pb-5 pe-3">
  706. 課程價格<span class="mark">*</span>
  707. </p>
  708. <v-text-field
  709. :rules="[requiredRule]"
  710. density="compact"
  711. variant="outlined"
  712. ></v-text-field>
  713. </v-label>
  714. </v-col> -->
  715. <v-col cols="12" class="py-0">
  716. <v-label class="d-block pb-3">
  717. <p class="pb-5 pe-3">
  718. 工藝類別<span class="mark">*</span>
  719. </p>
  720. <v-select
  721. v-model="course.category"
  722. placeholder="請選擇類別"
  723. :items="[
  724. '樹藝',
  725. '漆藝',
  726. '藍染',
  727. '蠟雕',
  728. '竹工藝籃',
  729. '金工/飾品',
  730. '蠟燭/香氛/調香',
  731. '植栽/花藝',
  732. '插畫/繪畫/寫字',
  733. '皮件/皮革',
  734. '木工/竹藝',
  735. '陶藝/玻璃',
  736. '編織/羊毛氈/縫紉',
  737. '其他',
  738. ]"
  739. :rules="[requiredRule]"
  740. density="compact"
  741. variant="outlined"
  742. ></v-select>
  743. </v-label>
  744. </v-col>
  745. <!-- <v-col cols="12" md="6">
  746. <v-label class="d-block pb-3">
  747. <p class="pb-5 pe-3">
  748. 適合報名對象(年紀)<span class="mark"
  749. >*</span
  750. >
  751. </p>
  752. <v-select
  753. v-model="age"
  754. :items="[
  755. '不拘',
  756. '7 歲以下',
  757. '7-18 歲',
  758. '18-65 歲',
  759. '65 歲以上',
  760. ]"
  761. :rules="[requiredRule]"
  762. density="compact"
  763. variant="outlined"
  764. outlined
  765. ></v-select>
  766. </v-label>
  767. </v-col> -->
  768. <v-col cols="12" class="py-0">
  769. <v-label class="d-block pb-3">
  770. <p class="pb-5 pe-3">上課地點</p>
  771. <v-text-field
  772. v-model="course.address"
  773. density="compact"
  774. variant="outlined"
  775. ></v-text-field>
  776. </v-label>
  777. </v-col>
  778. <v-col cols="12" class="py-0">
  779. <v-label class="d-block pb-3">
  780. <p class="pb-5 pe-3">主辦單位</p>
  781. <v-text-field
  782. v-model="course.organizer"
  783. density="compact"
  784. variant="outlined"
  785. ></v-text-field>
  786. </v-label>
  787. </v-col>
  788. <v-col cols="12" class="py-0">
  789. <v-label class="d-block">
  790. <p class="d-flex">
  791. 課程簡介<span class="mark">*</span>
  792. </p>
  793. <small class="d-block text-gray my-2"
  794. >內容不得少於 100
  795. 字元,且不得含有工藝無關之資訊</small
  796. >
  797. <v-textarea
  798. v-model="course.introduction"
  799. rows="5"
  800. variant="outlined"
  801. ></v-textarea>
  802. </v-label>
  803. </v-col>
  804. <v-col cols="12" class="me-auto">
  805. <v-label class="d-block">
  806. <p class="d-flex">
  807. 課程封面<span class="mark">*</span>
  808. </p>
  809. <v-file-input
  810. multiple
  811. v-model="coverImg"
  812. id="coverImgRef"
  813. ref="coverImgRef"
  814. label="File input"
  815. variant="outlined"
  816. placeholder="選擇相片上傳"
  817. @change="handleCoverImg"
  818. style="display: none"
  819. ></v-file-input>
  820. </v-label>
  821. <v-btn
  822. @click="fileInputClick('cover')"
  823. color="purple"
  824. class="my-5"
  825. >選擇相片上傳</v-btn
  826. >
  827. <div class="step-01 image-preview">
  828. <div
  829. class="d-flex justify-center"
  830. v-if="coverImgLoading"
  831. >
  832. <v-progress-circular
  833. color="grey-lighten-4"
  834. indeterminate
  835. ></v-progress-circular>
  836. </div>
  837. <img
  838. v-if="
  839. coverImgUrl !== '' && !coverImgLoading
  840. "
  841. :src="coverImgUrl"
  842. alt="上傳圖片預覽"
  843. />
  844. </div>
  845. </v-col>
  846. </v-row>
  847. </v-card-text>
  848. <v-card-actions class="d-flex justify-center">
  849. <v-btn
  850. variant="outlined"
  851. color="purple"
  852. @click="courseDialog = false"
  853. class="me-2 px-3"
  854. >關閉</v-btn
  855. >
  856. <v-btn
  857. variant="flat"
  858. color="purple"
  859. class="px-6"
  860. :loading="courseBtnLoading"
  861. @click="saveCourse()"
  862. >{{ courseBtnText }}</v-btn
  863. >
  864. </v-card-actions>
  865. </v-card>
  866. </v-dialog>
  867. </v-btn>
  868. </v-list-item-title>
  869. </v-list-item>
  870. </v-list>
  871. </v-menu>
  872. </td>
  873. </tr>
  874. </tbody>
  875. </table>
  876. </div>
  877. </v-card>
  878. </template>
  879. <style lang="scss">
  880. .user-courses {
  881. p,
  882. a {
  883. font-size: 16px;
  884. line-height: 30px;
  885. }
  886. .v-table {
  887. padding: 20px;
  888. }
  889. .main-info {
  890. margin: 30px;
  891. padding: 50px;
  892. position: relative;
  893. border: 2px solid var(--blue);
  894. border-radius: 20px;
  895. @media (max-width: 600px) {
  896. margin: 0;
  897. padding: 50px 20px 100px;
  898. }
  899. h5 {
  900. display: flex;
  901. flex-direction: column;
  902. align-items: center;
  903. font-size: 20px;
  904. font-weight: 500;
  905. letter-spacing: 1px;
  906. span {
  907. display: block;
  908. margin: 5px 0 15px;
  909. font-size: 36px;
  910. }
  911. }
  912. p {
  913. text-align: center;
  914. line-height: 22px;
  915. @media (max-width: 490px) {
  916. font-size: 14px;
  917. }
  918. }
  919. .icon {
  920. width: 50px;
  921. position: absolute;
  922. right: -50px;
  923. }
  924. .record-item {
  925. margin-top: 20px;
  926. position: relative;
  927. p {
  928. width: 100%;
  929. position: absolute;
  930. z-index: 100;
  931. top: 50%;
  932. left: 50%;
  933. transform: translate(-50%, -50%);
  934. color: #fff;
  935. font-size: 30px;
  936. }
  937. strong {
  938. display: inline-block;
  939. margin: 25px 0;
  940. font-size: 56px;
  941. font-weight: 500;
  942. @media (max-width: 600px) {
  943. font-size: 50px;
  944. }
  945. }
  946. small {
  947. font-size: 20px;
  948. }
  949. img {
  950. max-width: 260px;
  951. @media (max-width: 600px) {
  952. max-width: 220px;
  953. }
  954. }
  955. }
  956. }
  957. .table-title {
  958. background-color: var(--blue);
  959. }
  960. table {
  961. thead {
  962. border-bottom: 2px solid var(--blue);
  963. }
  964. tbody {
  965. td {
  966. border-bottom: 1px solid var(--blue);
  967. }
  968. }
  969. .finish-icon {
  970. padding: 8px;
  971. border: 2px solid var(--blue);
  972. border-radius: 100px;
  973. .v-icon {
  974. color: var(--blue);
  975. }
  976. }
  977. }
  978. }
  979. .courses-table {
  980. .v-data-table-footer {
  981. padding: 15px 0;
  982. }
  983. .account-item {
  984. .v-field__input {
  985. padding: 0 20px;
  986. }
  987. .v-label {
  988. color: var(--gray);
  989. }
  990. }
  991. }
  992. </style>