TestECPay.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468
  1. <script setup lang="ts">
  2. import { ref, reactive } from "vue";
  3. import { required } from "@/utils";
  4. import { useAdminStore } from "@/stores/admin";
  5. import axios from "axios";
  6. import { useMainStore } from "@/stores/main";
  7. const mainStore = useMainStore();
  8. const items = reactive([
  9. { title: "100% 真人觀看" },
  10. { title: "包含影片設定費" },
  11. { title: "開發票" },
  12. { title: "包含成效報表" },
  13. ]);
  14. const cardItems = reactive([
  15. { view: "5,000", price: "2,700", originalPrice: "3,500", param: 2700 },
  16. { view: "10,000", price: "4,400", originalPrice: "5,000", param: 4400 },
  17. { view: "30,000", price: "12,400", originalPrice: "13,000", param: 12400 },
  18. { view: "50,000", price: "20,400", originalPrice: "21,000", param: 20400 },
  19. ]);
  20. const ageOptions = [
  21. { label: "18 - 24 歲" },
  22. { label: "25 - 34 歲" },
  23. { label: "35 - 44 歲" },
  24. { label: "45 - 54 歲" },
  25. { label: "55 - 64 歲" },
  26. { label: "65 歲以上" },
  27. ];
  28. const objectOptions = [
  29. { label: "交通工具與運輸" },
  30. { label: "媒體和娛樂" },
  31. { label: "家居與園藝" },
  32. { label: "新聞與政治" },
  33. { label: "旅遊" },
  34. { label: "生活型態與興趣" },
  35. { label: "科技" },
  36. { label: "美容與健康" },
  37. { label: "美食與餐飲" },
  38. { label: "購物愛好者" },
  39. { label: "運動與健身" },
  40. { label: "銀行與金融" },
  41. ];
  42. const themeOptions = [
  43. { label: "人文與社會" },
  44. { label: "保健" },
  45. { label: "全球地點(各地區)" },
  46. { label: "參考資料(圖書館、博物館與目錄、清單等)" },
  47. { label: "圖書與文學" },
  48. { label: "家居與園藝" },
  49. { label: "寵物與動物" },
  50. { label: "工作與教育" },
  51. { label: "工商業" },
  52. { label: "房地產" },
  53. { label: "新聞" },
  54. { label: "旅遊與交通" },
  55. { label: "汽車與交通工具" },
  56. { label: "法律與政府" },
  57. { label: "科學" },
  58. { label: "網路社群" },
  59. { label: "網際網路與電信" },
  60. { label: "美容與健身" },
  61. { label: "美食佳飲" },
  62. { label: "興趣與休閒" },
  63. { label: "藝術與娛樂" },
  64. { label: "購物" },
  65. { label: "遊戲" },
  66. { label: "運動" },
  67. { label: "金融" },
  68. { label: "電腦和電子產品" },
  69. ];
  70. let chooseError = ref(false);
  71. let selectedBtn = ref();
  72. function activeBtn(param: any) {
  73. selectedBtn.value = param;
  74. chooseError.value = false;
  75. }
  76. let userData = reactive({
  77. email: "",
  78. name: "",
  79. company: "",
  80. url: "",
  81. area: "",
  82. language: "",
  83. age: [],
  84. object: "",
  85. theme: "",
  86. tax: "",
  87. });
  88. // 其他選項
  89. let object = ref("");
  90. let theme = ref("");
  91. let otherObject = ref("");
  92. let otherTheme = ref("");
  93. async function ECPaySubmit() {
  94. /*
  95. const timestamp = Date.now()
  96. const date = new Date(timestamp);
  97. const year = date.getFullYear();
  98. const month = date.getMonth() + 1;
  99. const day = date.getDate();
  100. const hour = date.getHours();
  101. const minute = date.getMinutes();
  102. const second = date.getSeconds();
  103. console.log("ECPay button pushed")
  104. const formData = new URLSearchParams();
  105. formData.append("MerchantID", "3226141") //必填
  106. formData.append("MerchantTradeNo", "AAA000") //必填
  107. formData.append("MerchantTradeDate", `${year}/${month}/${day} ${hour}:${minute}:${second}`) //必填
  108. formData.append("PaymentType", "aio") //必填
  109. formData.append("TotalAmount", "600") //必填
  110. formData.append("TradeDesc", "choozmo SaaS") //必填
  111. formData.append("ItemName", "charge600") //必填
  112. formData.append("ReturnURL", "https:cloud.choozmo.com") //必填
  113. formData.append("ChoosePayment", "ALL") //必填
  114. formData.append("CheckMacValue", "choaho")
  115. formData.append("EncryptType", "1") //必填
  116. formData.append("StoreID", "")
  117. formData.append("ClientBackURL", "")
  118. formData.append("ItemURL", "")
  119. formData.append("Remark", "")
  120. formData.append("ChooseSubPayment", "")
  121. formData.append("OrderResultURL", "")
  122. formData.append("NeedExtraPaidInfo", "")
  123. formData.append("IgnorePayment", "")
  124. formData.append("PlatformID", "")
  125. formData.append("IgnorePayment", "")
  126. formData.append("CustomField1 ", "")
  127. formData.append("CustomField2", "")
  128. formData.append("CustomField3", "")
  129. formData.append("CustomField4", "")
  130. formData.append("Language", "")
  131. return axios.post("https://payment-stage.ecpay.com.tw/Cashier/AioCheckOut/V5", formData)
  132. */
  133. // 是否選擇方案
  134. if (!selectedBtn.value) {
  135. chooseError.value = true;
  136. window.scrollTo({ top: 0, behavior: "smooth" }); // 捲動至上方
  137. } else {
  138. chooseError.value = false;
  139. }
  140. if (object.value === "其他" && otherObject.value !== "") {
  141. userData.object = otherObject.value;
  142. } else {
  143. userData.object = object.value;
  144. }
  145. if (theme.value === "其他" && otherTheme.value !== "") {
  146. userData.theme = otherTheme.value;
  147. } else {
  148. userData.theme = theme.value;
  149. }
  150. await saveData();
  151. const originalHTML = await mainStore.ecpayPaymentHTML(selectedBtn.value);
  152. let formHTML = originalHTML?.replace(
  153. '<script type="text/javascript">document.getElementById("data_set").submit();</scr',
  154. ""
  155. );
  156. formHTML = formHTML?.replace("ipt>", "");
  157. const payFormElement = document.getElementById("pay-form");
  158. payFormElement!.innerHTML = formHTML!;
  159. const ecpayForm: HTMLFormElement = <HTMLFormElement>(
  160. document.getElementById("data_set")
  161. );
  162. console.log(ecpayForm);
  163. ecpayForm.submit();
  164. }
  165. // 資料存進 Google Sheets
  166. async function saveData() {
  167. const scriptURL =
  168. "https://script.google.com/macros/s/AKfycbxCcfiOQ695DaxIa3peClqRRTWNj2aUNLbx7ty8U2wKlyU7wreQLioHG-sls5MPKBdlRQ/exec";
  169. let formdata = new FormData();
  170. formdata.append("email", userData.email);
  171. formdata.append("name", userData.name);
  172. formdata.append("company", userData.company);
  173. formdata.append("url", userData.url);
  174. formdata.append("area", userData.area);
  175. formdata.append("language", userData.language);
  176. formdata.append("age", userData.age.join("、"));
  177. formdata.append("object", userData.object);
  178. formdata.append("theme", userData.theme);
  179. formdata.append("tax", userData.tax);
  180. axios
  181. .post(scriptURL, formdata)
  182. .then(function (response) {
  183. console.log(response.data);
  184. })
  185. .catch(function (error) {
  186. console.log(error);
  187. });
  188. }
  189. </script>
  190. <template>
  191. <v-container fluid>
  192. <v-card class="ma-3 pa-3">
  193. <v-card-title primary-title class="mb-3">
  194. <h3 class="headline primary--text">YouTube 觀看數</h3>
  195. </v-card-title>
  196. <v-card-text>
  197. <p class="ms-3">請選擇方案:</p>
  198. <v-row no-gutters class="pay-card">
  199. <v-col
  200. xs="12"
  201. sm="6"
  202. lg="3"
  203. v-for="(item, index) in cardItems"
  204. :key="index"
  205. >
  206. <button @click="activeBtn(item.param)" class="w-100">
  207. <v-card
  208. class="ma-3 py-3"
  209. :class="{ active: selectedBtn === item.param }"
  210. >
  211. <v-card-title primary-title class="pa-0">
  212. <div class="d-flex flex-column">
  213. <section class="d-flex mx-auto">
  214. <img
  215. width="30"
  216. height="30"
  217. src="@/assets/img/icon/play-button.png"
  218. alt=""
  219. class="me-2"
  220. />
  221. <h5 class="m-0">{{ item.view }}</h5>
  222. </section>
  223. <span class="text-center" style="color: #7c8ba7"
  224. >Views</span
  225. >
  226. </div>
  227. <p class="price">
  228. NT${{ item.price }} <br />
  229. <small>NT${{ item.originalPrice }}</small>
  230. </p>
  231. </v-card-title>
  232. <v-card-text class="d-flex align-center justify-center mt-3">
  233. <ul>
  234. <li
  235. v-for="(item, index) in items"
  236. :key="index"
  237. class="d-flex align-center"
  238. >
  239. <img
  240. width="30"
  241. src="@/assets/img/icon/check.png"
  242. alt=""
  243. />
  244. {{ item.title }}
  245. </li>
  246. </ul>
  247. </v-card-text>
  248. <!-- <v-card-actions class="d-flex justify-center">
  249. <v-btn @click="ECPaySubmit(item.param)"> Buy Now </v-btn>
  250. </v-card-actions> -->
  251. </v-card>
  252. </button>
  253. </v-col>
  254. <p class="ms-3 error" v-show="chooseError">尚未選擇方案</p>
  255. </v-row>
  256. <v-sheet max-width="500" class="mx-auto mt-10">
  257. <v-form @submit.prevent class="ECPay-form">
  258. <v-text-field
  259. v-model="userData.email"
  260. :rules="[(v) => !!v || '請輸入您的電子郵件']"
  261. label="電子郵件"
  262. required
  263. ></v-text-field>
  264. <v-text-field
  265. v-model="userData.name"
  266. :rules="[(v) => !!v || '請輸入您的姓名']"
  267. label="姓名"
  268. required
  269. ></v-text-field>
  270. <v-text-field
  271. v-model="userData.company"
  272. label="公司 / 所屬產業"
  273. ></v-text-field>
  274. <v-text-field
  275. v-model="userData.url"
  276. :rules="[(v) => !!v || '請輸入 YouTube 影片網址']"
  277. label="YouTube 影片網址"
  278. required
  279. ></v-text-field>
  280. <v-text-field
  281. v-model="userData.area"
  282. :rules="[(v) => !!v || '請輸入影片放送地區']"
  283. label="影片放送地區(國家 / 縣市)"
  284. required
  285. ></v-text-field>
  286. <v-text-field
  287. v-model="userData.language"
  288. :rules="[(v) => !!v || '請輸入受眾語言']"
  289. label="受眾語言"
  290. required
  291. ></v-text-field>
  292. <p class="mt-5">客層(未選擇的話視為全部)</p>
  293. <div class="checkbox ms-5">
  294. <v-checkbox
  295. v-for="option in ageOptions"
  296. v-model="userData.age"
  297. :key="option.label"
  298. :label="option.label"
  299. :value="option.label"
  300. color="primary"
  301. ></v-checkbox>
  302. </div>
  303. <p class="mt-10 mb-3">
  304. 目標對象區隔(興趣、習慣)<span class="text-red-darken-1">*</span>
  305. </p>
  306. <div class="ms-5">
  307. <v-radio-group v-model="object">
  308. <v-radio
  309. v-for="option in objectOptions"
  310. :key="option.label"
  311. :label="option.label"
  312. :value="option.label"
  313. color="primary"
  314. ></v-radio>
  315. <v-radio label="其他" value="其他" color="primary"></v-radio>
  316. <input v-model="otherObject" type="text" class="other" />
  317. </v-radio-group>
  318. </div>
  319. <p class="mt-5 mb-3">
  320. 影片主題 <span class="text-red-darken-1">*</span>
  321. </p>
  322. <div class="ms-5">
  323. <v-radio-group v-model="theme">
  324. <v-radio
  325. v-for="option in themeOptions"
  326. :key="option.label"
  327. :label="option.label"
  328. :value="option.label"
  329. color="primary"
  330. ></v-radio>
  331. <v-radio label="其他" value="其他" color="primary"></v-radio>
  332. <input v-model="otherTheme" type="text" class="other" />
  333. </v-radio-group>
  334. </div>
  335. <v-text-field
  336. type="number"
  337. label="是否需要統編(可填寫統編號碼)"
  338. v-model="userData.tax"
  339. ></v-text-field>
  340. <v-btn
  341. @click="ECPaySubmit()"
  342. type="submit"
  343. block
  344. class="mt-2 submit-btn"
  345. >送出</v-btn
  346. >
  347. </v-form>
  348. </v-sheet>
  349. </v-card-text>
  350. </v-card>
  351. </v-container>
  352. <div id="pay-form"></div>
  353. </template>
  354. <style lang="scss">
  355. .pay-card {
  356. .v-card-title {
  357. h5 {
  358. font-size: 20px;
  359. }
  360. span {
  361. font-size: 16px;
  362. letter-spacing: 1px;
  363. }
  364. .price {
  365. padding: 10px 0;
  366. font-size: 26px;
  367. font-weight: 600;
  368. text-align: center;
  369. color: #fff;
  370. background-color: var(--main-color);
  371. letter-spacing: 2px;
  372. small {
  373. display: block;
  374. margin-top: -3px;
  375. font-size: 18px;
  376. font-weight: 100;
  377. text-decoration: line-through;
  378. }
  379. }
  380. }
  381. .v-card-text {
  382. ul {
  383. padding: 0;
  384. list-style: none;
  385. }
  386. }
  387. .v-card-actions {
  388. button {
  389. padding: 5px 15px;
  390. color: #fff;
  391. border-radius: 100px;
  392. background-color: var(--main-color);
  393. border: 1px solid transparent;
  394. &:hover {
  395. color: var(--main-color);
  396. background-color: #fff;
  397. border: 1px solid var(--main-color);
  398. }
  399. }
  400. }
  401. .active {
  402. border: 3px solid var(--main-color);
  403. }
  404. .error {
  405. color: #b00020;
  406. }
  407. }
  408. .ECPay-form {
  409. font-size: 16px;
  410. .checkbox {
  411. margin-left: 5px;
  412. list-style: none;
  413. .v-input {
  414. height: 40px;
  415. }
  416. }
  417. .v-input__details {
  418. padding-top: 0;
  419. padding-bottom: 3px;
  420. }
  421. .other {
  422. margin-left: 40px;
  423. border-bottom: 1px solid #333;
  424. &:focus-visible {
  425. outline: none !important;
  426. }
  427. }
  428. .submit-btn {
  429. color: #fff;
  430. background-color: var(--main-color);
  431. }
  432. }
  433. </style>