index.tsx 50 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894
  1. import React, { useCallback, useEffect, useMemo, useState } from 'react'
  2. import { Modal, Form, Input, Divider, Select, Radio, Switch, Spin, List, Space, Button, message, Image, Empty } from 'antd'
  3. import styles from './index.less'
  4. import { useAjax } from '@/Hook/useAjax'
  5. import { getText, get_adcreative_template, get_adcreative_template_list, get_tools_video_capture } from '@/services/launchAdq/global'
  6. import { AdcreativeTemplate, AdcreativeTemplateList } from '@/services/launchAdq'
  7. import { mySet } from '@/utils/arrFn'
  8. import SelectCloud from '@/pages/launchSystemNew/components/selectCloud'
  9. import { useModel } from 'umi'
  10. import { ModalConfig } from '../../ad';
  11. import { outAdcreativeTemplateIdFun } from '../../../localAd/adenum'
  12. import { CreateAdProps } from '@/services/launchAdq/createAd'
  13. import { creativeConfig, overrideCanvasHeadOptionEnum } from './config'
  14. import BrandImage from './brandImage'
  15. import HeadNickJump from './headNickJump'
  16. import moment from 'moment'
  17. interface Props {
  18. queryForm: Partial<CreateAdProps>,
  19. title?: string,
  20. visible: boolean,
  21. PupFn: (arg: ModalConfig) => void,
  22. callback: (params: any, material: { label: string, name: string, restriction: any }[], textData: any[]) => void,
  23. confirmLoading?: boolean,
  24. type?: 'add' | 'look' | 'edit',//新增,查看,编辑
  25. dataInfo?: any
  26. }
  27. /**创意组模板*/
  28. function CreativePup(props: Props) {
  29. /***************************************/
  30. let { visible, confirmLoading, PupFn, callback, type, dataInfo, queryForm } = props
  31. const { currentUser }: any = useModel('@@initialState', model => ({ currentUser: model.initialState?.currentUser }))
  32. let [template_checked, settemplate_checked] = useState<boolean>(dataInfo?.isTemplate || false)
  33. let { promotedObjectType, sysAdgroup } = queryForm
  34. let { siteSet } = sysAdgroup
  35. let arg = type === 'look' ? { footer: null } : {}
  36. // 请求
  37. const getAdcreativeTemplate = useAjax((params) => get_adcreative_template(params))
  38. const getAdcreativeTemplateList = useAjax((params) => get_adcreative_template_list(params))
  39. const getTextLsit = useAjax((params) => getText(params))
  40. const getVideoCapture = useAjax((params) => get_tools_video_capture(params))
  41. // 变量
  42. const [adcreative_template, set_adcreative_template] = useState<AdcreativeTemplate>()
  43. const [adcreative_template_list, set_adcreative_template_list] = useState<AdcreativeTemplateList[]>([])
  44. const [selectImgVisible, set_selectImgVisible] = useState(false)
  45. const [selectVideoVisible, set_selectVideoVisible] = useState(false)
  46. const [videoImgsVisbile, set_videoImgsVisbile] = useState(false)
  47. const [endPageDescShow, setendPageDescnshow] = useState(false)
  48. const [materialData, setMaterialData] = useState<{ label: string, name: string, restriction: any }[]>([])
  49. const [textData, setTextData] = useState<any[]>([])
  50. const [isShowSc, set_isShowSc] = useState(false)//是否展示素材选项
  51. const [infoSet, set_infoSet] = useState(false)//回填设置已完成
  52. const [isShowXd, setIsShowXd] = useState(false)
  53. const [videoImgs, set_videoImgs] = useState<{//视频封面图设置
  54. activeUrl: string,//选中的视频封面图地址
  55. preview: boolean,//是否开启图片点击预览
  56. urlList: any[],//生成的视频封面列表
  57. }>({
  58. activeUrl: '',
  59. preview: false,
  60. urlList: [
  61. 'https://test-adq-media.oss-cn-hangzhou.aliyuncs.com/image/21D8D51AD98C4FF8BF41F1C2D28EA39F.jpg',
  62. 'https://test-adq-media.oss-cn-hangzhou.aliyuncs.com/image/80DBE1AB3EDE4E85ABAE5F1670D9FED0.jpg',
  63. 'https://test-adq-media.oss-cn-hangzhou.aliyuncs.com/image/BCB2DAB86BDB4549BCB8E493C4F29E82.jpg',
  64. 'https://test-adq-media.oss-cn-hangzhou.aliyuncs.com/image/545A4C2A5B874C82A9D1C0C063624AE5.jpg'
  65. ]
  66. })
  67. const [form] = Form.useForm();
  68. const [pupState, setPupState] = useState({
  69. kp_show: false,
  70. xd_show: false,
  71. sj_show: false,
  72. bq_show: false,
  73. sp_show: false
  74. })
  75. const [imgMaterialConfig, setImgMaterialConfig] = useState<{
  76. adcreativeTemplateId?: number,
  77. type: string,
  78. cloudSize: { relation: string, width: number, height: number }[],
  79. list: any[],
  80. max: number
  81. }>({
  82. type: '',//类型
  83. cloudSize: [],//素材搜索条件
  84. list: [],//素材
  85. max: 1,//素材数量
  86. })//图片素材配置
  87. const [videoMaterialConfig, setVideoMaterialConfig] = useState<{
  88. adcreativeTemplateId?: number,
  89. type: string,
  90. cloudSize: { relation: string, width: number, height: number }[],
  91. list: any[],
  92. max: number
  93. }>({
  94. type: '',//类型
  95. cloudSize: [],//素材搜索条件
  96. list: [],//素材
  97. max: 1,//素材数量
  98. })//图片素材配置
  99. const [conversionList, setConversionList] = useState<any>(null)
  100. let pageType = Form.useWatch('pageType', form)
  101. let adcreativeTemplateId = Form.useWatch('adcreativeTemplateId', form)
  102. let actionBtn = Form.useWatch('actionBtn', form)
  103. // let siteSet = Form.useWatch('siteSet', form)
  104. let overrideCanvasHeadOption = Form.useWatch('overrideCanvasHeadOption', form)
  105. let adcreativeElementsType = Form.useWatch('adcreativeElementsType', form)
  106. let dataShow = Form.useWatch('dataShow', form)
  107. let conversionDataType = Form.useWatch('conversionDataType', form)
  108. let titles = Form.useWatch('title', form)
  109. let description = Form.useWatch('description', form)
  110. let videoOver = Form.useWatch('videoOver', form)
  111. let endPageDesc = Form.useWatch('endPageDesc', form)
  112. let linkPageType = Form.useWatch('linkPageType', form)
  113. /***************************************/
  114. // 确定事件
  115. const handleOk = useCallback(() => {
  116. form.validateFields().then(values => {
  117. console.log('values=>1', values)
  118. let newValues = JSON.parse(JSON.stringify(values))
  119. for (let key in newValues) {
  120. switch (key) {
  121. case 'image'://图素材
  122. newValues.adcreativeElements = {
  123. ...newValues.adcreativeElements,
  124. imageUrl: imgMaterialConfig?.list[0]?.url,
  125. }
  126. delete newValues[key]
  127. break;
  128. case 'video'://视频素材
  129. newValues.adcreativeElements = {
  130. ...newValues.adcreativeElements,
  131. videoUrl: videoMaterialConfig?.list[0]?.url,
  132. }
  133. delete newValues[key]
  134. break;
  135. case 'image_list'://图素材
  136. newValues.adcreativeElements = {
  137. ...newValues.adcreativeElements,
  138. imageUrlList: imgMaterialConfig.list?.map(item => item.url),
  139. description: newValues.description,
  140. }
  141. delete newValues[key]
  142. break;
  143. case 'short_video1'://视频素材
  144. newValues.adcreativeElements = {
  145. ...newValues.adcreativeElements,
  146. shortVideoStruct: {
  147. shortVideo1Url: videoMaterialConfig?.list[0]?.url
  148. },
  149. description: newValues.description,
  150. }
  151. delete newValues[key]
  152. break;
  153. case 'description'://文案
  154. newValues.adcreativeElements = { ...newValues.adcreativeElements, description: newValues.description }
  155. break;
  156. case 'title'://文案
  157. newValues.adcreativeElements = { ...newValues.adcreativeElements, title: newValues.title }
  158. break;
  159. case 'endPageType'://视频结束l类型
  160. newValues.adcreativeElements = { ...newValues.adcreativeElements, endPage: { ...newValues.adcreativeElements.endPage, endPageType: newValues.endPageType } }
  161. delete newValues[key]
  162. break;
  163. case 'endPageDesc'://视频结束文案
  164. newValues.adcreativeElements = { ...newValues.adcreativeElements, endPage: { ...newValues.adcreativeElements.endPage, endPageDesc: newValues.endPageDesc } }
  165. delete newValues[key]
  166. break;
  167. case 'buttonText'://特殊行动按钮
  168. newValues.adcreativeElements = { ...newValues.adcreativeElements, buttonText: newValues.buttonText }
  169. delete newValues[key]
  170. break;
  171. case 'brand'://品牌形象
  172. newValues.adcreativeElements = {
  173. ...newValues.adcreativeElements, brand: {
  174. brandName: newValues.brand.split('_')[0],
  175. brandImgUrl: newValues.brand.split('_')[1]
  176. }
  177. }
  178. break;
  179. case 'profile':
  180. newValues.adcreativeElements = {
  181. ...newValues.adcreativeElements, brand: {
  182. brandName: newValues.profile.split('_')[0],
  183. brandImgUrl: newValues.profile.split('_')[1]
  184. }
  185. }
  186. newValues.profile = {
  187. headImageUrl: newValues.profile.split('_')[1],
  188. profileName: newValues.profile.split('_')[0],
  189. description: newValues.profile.split('_')[2]
  190. }
  191. break
  192. case 'pageUrl'://跳转落地页
  193. newValues.linkPageSpec = {
  194. ...newValues.linkPageSpec,
  195. pageUrl: newValues.pageUrl
  196. }
  197. delete newValues.pageUrl
  198. break;
  199. case 'miniProgramId':
  200. newValues.linkPageSpec = {
  201. ...newValues.linkPageSpec,
  202. miniProgramSpec: {
  203. miniProgramId: newValues.miniProgramId,
  204. miniProgramPath: newValues.miniProgramPath
  205. }
  206. }
  207. delete newValues.miniProgramId
  208. delete newValues.miniProgramPath
  209. break;
  210. }
  211. }
  212. if (!newValues.adcreativeElements) {
  213. newValues.adcreativeElements = {}
  214. }
  215. //假如不存在promotedObjectType
  216. if (!newValues?.promotedObjectType) {
  217. newValues['promotedObjectType'] = queryForm.promotedObjectType
  218. }
  219. // 假如不存在siteSet
  220. if (!newValues?.siteSet) {
  221. newValues['siteSet'] = queryForm.sysAdgroup.siteSet
  222. }
  223. delete newValues.description //删除外层文案
  224. delete newValues.title //删除外层文案
  225. delete newValues.adcreativeElementsType //删除创意形式
  226. delete newValues.dataShow //删除数据开关
  227. delete newValues.actionBtn //删除行动开关
  228. delete newValues.brand //品牌形象
  229. // 假如使用了落地页顶部素材替换外部素材
  230. if (newValues.overrideCanvasHeadOption === 'OPTION_CANVAS_OVERRIDE_CREATIVE') {
  231. console.log(adcreative_template?.adcreativeElements)
  232. adcreative_template?.adcreativeElements?.filter(item => item.required && item.name === 'image_list' || item.name === 'short_video1' || item.name === 'video' || item.name === 'image').forEach(item => {
  233. switch (item.name) {
  234. case 'image'://图素材
  235. newValues.adcreativeElements = {
  236. ...newValues.adcreativeElements,
  237. imageUrl: '',
  238. }
  239. break;
  240. case 'video'://视频素材
  241. newValues.adcreativeElements = {
  242. ...newValues.adcreativeElements,
  243. videoUrl: '',
  244. }
  245. break;
  246. case 'image_list'://图素材
  247. newValues.adcreativeElements = {
  248. ...newValues.adcreativeElements,
  249. imageUrlList: [],
  250. }
  251. break;
  252. case 'short_video1'://视频素材
  253. newValues.adcreativeElements = {
  254. ...newValues.adcreativeElements,
  255. shortVideoStruct: {
  256. shortVideo1Url: ''
  257. },
  258. }
  259. break;
  260. }
  261. })
  262. }
  263. console.log('newValues=>2', newValues)
  264. newValues['isTemplate'] = template_checked
  265. // // 开启存为模板开关执行
  266. callback(newValues, materialData, textData)
  267. })
  268. }, [form, imgMaterialConfig, videoMaterialConfig, materialData, textData, queryForm, template_checked, adcreative_template, isShowSc])
  269. // 获取创意形式列表
  270. useEffect(() => {
  271. if (siteSet?.length > 0 && promotedObjectType) {
  272. getAdcreativeTemplateList.run({
  273. siteSet,
  274. promotedObjectType,
  275. campaignType: 'CAMPAIGN_TYPE_NORMAL',
  276. }).then(res => {
  277. let newArr: any = []
  278. let newData: any[] = []
  279. // 过滤掉相同的和即将下线的
  280. if (!res) {
  281. return
  282. }
  283. //
  284. Object.values(res)?.forEach((arr: any) => {
  285. newData.push(arr)
  286. Array.isArray(arr) && arr?.forEach((item: any) => {
  287. if (newArr.length > 0) {//假如已存在ID,需要过滤相同
  288. if (outAdcreativeTemplateIdFun(item.adcreativeTemplateId) && newArr.every((i: { adcreativeTemplateId: any }) => i.adcreativeTemplateId !== item.adcreativeTemplateId)) {//不重复的添加
  289. newArr.push(item)
  290. }
  291. } else {//不存在ID直接过滤掉即将下线的
  292. if (outAdcreativeTemplateIdFun(item.adcreativeTemplateId)) {
  293. newArr.push(item)
  294. }
  295. }
  296. })
  297. })
  298. /*****暂时排除激励和banner有问题******/
  299. if (siteSet.some((i: string) => i === 'SITE_SET_MOMENTS')) {
  300. newArr = newArr.filter((item: { adcreativeTemplateId: number }) => item.adcreativeTemplateId !== 910 && item.adcreativeTemplateId !== 925)
  301. }
  302. /*****暂时排除出框形态 视频合约广告******/
  303. if (siteSet.some((i: string) => i === 'SITE_SET_WECHAT')) {
  304. newArr = newArr.filter((item: { adcreativeTemplateId: number }) => item.adcreativeTemplateId !== 1945)
  305. }
  306. let newArr1: any[] = []
  307. let newArr2: any[] = []
  308. newArr?.forEach((arr: { adcreativeTemplateId: any, isGeneral?: boolean }) => {
  309. if (newData.every((item: { adcreativeTemplateId: any }[]) => item.find(i => i.adcreativeTemplateId === arr.adcreativeTemplateId))) {
  310. newArr1.push({ ...arr, isGeneral: true })
  311. } else {
  312. newArr2.push(arr)
  313. }
  314. })
  315. set_adcreative_template_list([...newArr1, ...newArr2])
  316. })
  317. }
  318. }, [siteSet, promotedObjectType])
  319. // 获取创意形式详情
  320. const getTemplate = (id: any, ok?: any) => {
  321. // CAMPAIGN_TYPE_NORMAL
  322. if (siteSet?.length > 0 && promotedObjectType && id) {
  323. if (id) {
  324. getAdcreativeTemplate.run({
  325. siteSet,
  326. promotedObjectType,
  327. adcreativeTemplateId: id
  328. }).then(res => {
  329. if (res?.length > 0) {
  330. form.setFieldsValue({ adcreativeName: res[0]?.adcreativeTemplateAppellation + '_' + moment().format('YYYYMMDDhhmmss') + '_' + currentUser.userId })
  331. set_adcreative_template(res[0])
  332. if (siteSet?.some((name: string) => name === 'SITE_SET_MOMENTS')) {
  333. let id = res[0].adcreativeTemplateId
  334. set_isShowSc(!!creativeConfig[id])//判定当前创意是否需要展示替换素材选项
  335. if (creativeConfig[id] && !ok) {//假如不等于回填元素的ID
  336. let overrideCanvasHeadOption = creativeConfig[id].overrideCanvasHeadOption
  337. form.setFieldsValue({ overrideCanvasHeadOption: overrideCanvasHeadOption?.includes('OPTION_CREATIVE_OVERRIDE_CANVAS') ? 'OPTION_CREATIVE_OVERRIDE_CANVAS' : overrideCanvasHeadOption[0] })
  338. }
  339. }
  340. templateChange(res[0], ok)
  341. // 处理素材
  342. setMaterialData(res[0]?.adcreativeElements?.filter((item: any) => item.required && item.name === 'image_list' || item.name === 'short_video1' || item.name === 'video' || item.name === 'image').map((item: any) => {
  343. return {
  344. label: item.description === '图片' && res[0]?.adcreativeElements?.some((item: any) => item.name === 'video') ? '视频封面图' : item.description,
  345. name: item.name,
  346. restriction: item.restriction,
  347. arrayProperty: item?.arrayProperty
  348. }
  349. }))
  350. // 处理文案
  351. setTextData(res[0]?.adcreativeElements?.filter((item: any) => item.name === 'title' || (item.required && item.name === 'description')).map((item: any) => ({ ...item, pupState })))
  352. }
  353. })
  354. }
  355. }
  356. }
  357. // 获取对应落地页按钮
  358. const pageTypeList = useMemo(() => {
  359. if (adcreativeTemplateId) {
  360. let arr: any = adcreative_template?.landingPageConfig?.supportPageTypeList
  361. return arr
  362. }
  363. return null
  364. }, [adcreativeTemplateId, adcreative_template])
  365. // 获取对应行动按钮数据
  366. const linkNameList = useMemo(() => {
  367. if (pageType) {
  368. let arr = (pageTypeList?.filter((item: { pageType: any; }) => item.pageType === pageType)[0] as any)?.supportLinkNameType?.list
  369. return arr
  370. }
  371. return null
  372. }, [pageType, pageTypeList])
  373. // 跳转落地页
  374. const linkPageList = useMemo(() => {
  375. if (pageType) {
  376. let arr = (pageTypeList?.filter((item: { pageType: any; }) => item.pageType === pageType)[0] as any)?.supportLinkPageType?.list
  377. return arr
  378. }
  379. return null
  380. }, [pageType, pageTypeList])
  381. const typeChange = useCallback((adcreativeElementsType) => {
  382. if (adcreativeElementsType && adcreative_template_list?.length > 0) {
  383. let adcreativeTemplateIdArr = adcreative_template_list?.filter(item => item.adcreativeTemplateStyle === adcreativeElementsType)
  384. console.log('typeChange====>', adcreativeTemplateIdArr[0].adcreativeTemplateId)
  385. getTemplate(adcreativeTemplateIdArr[0].adcreativeTemplateId)
  386. form.setFieldsValue({ adcreativeTemplateId: adcreativeTemplateIdArr[0].adcreativeTemplateId })
  387. }
  388. }, [adcreative_template_list])
  389. //每次选中创意设置该展示的界面
  390. const templateChange = (adcreative_template: any, ok?: any) => {
  391. let states = {
  392. kp_show: false,
  393. xd_show: true,
  394. sj_show: false,
  395. bq_show: false,
  396. sp_show: false
  397. }
  398. let values: any = { pageType: 'PAGE_TYPE_CANVAS_WECHAT', }
  399. if (adcreative_template) {
  400. let pageList = adcreative_template?.landingPageConfig?.supportPageTypeList?.filter((i: { description: string | string[] }) => i.description.includes('微信原生推广页'))//当前版本只获取微信原生页,后期改进
  401. let pageType = pageList?.length ? pageList[0]?.pageType : null
  402. //数据展示组件
  403. if (adcreative_template.adcreativeAttributes.some((item: { name: string }) => item.name === 'conversion_data_type' || item.name === 'conversion_target_type')) {
  404. let arr = adcreative_template.adcreativeAttributes?.filter((item: { name: string; }) => item.name === 'conversion_data_type' || item.name === 'conversion_target_type')
  405. let newObj: any = {}
  406. arr.forEach((item: { propertyDetail: { enumDetail: { enumeration: any[] } }; name: string | number }) => {
  407. let arr: any[] = mySet(item.propertyDetail.enumDetail.enumeration)
  408. newObj[item.name] = arr
  409. })
  410. setConversionList(newObj)
  411. states = { ...states, sj_show: true }
  412. if (newObj.conversion_data_type) {
  413. values = { ...values, conversionDataType: newObj.conversion_data_type[0].value }
  414. }
  415. if (newObj.conversion_target_type) {
  416. values = { ...values, conversionTargetType: newObj.conversion_target_type[0].value }
  417. }
  418. }
  419. //行动按钮组件存在
  420. if (states.xd_show) {
  421. let supportLinkNameTypeData = (pageList?.filter((item: { pageType: any; }) => item.pageType === pageType)[0] as any)?.supportLinkNameType
  422. let linkNameList = supportLinkNameTypeData?.list
  423. let linkPageList = (pageList?.filter((item: { pageType: any; }) => item.pageType === pageType)[0] as any)?.supportLinkPageType?.list
  424. if (linkNameList) { // && !linkPageType
  425. if (!ok) {
  426. let linkNameType = linkNameList[0]?.linkNameType
  427. let linkPageType = linkPageList?.some((i: { linkPageType: string }) => i.linkPageType === "LINK_PAGE_TYPE_CANVAS_WECHAT") ? "LINK_PAGE_TYPE_CANVAS_WECHAT" : linkPageList[0]?.linkPageType
  428. values = { ...values, linkNameType, linkPageType, actionBtn: true }
  429. }
  430. } else {
  431. states = { ...states, xd_show: false }
  432. }
  433. if (supportLinkNameTypeData?.required) {
  434. states = { ...states, xd_show: true }
  435. values = { ...values, actionBtn: true }
  436. setIsShowXd(true)
  437. } else {
  438. setIsShowXd(false)
  439. }
  440. }
  441. // 特殊行动按钮
  442. if (adcreative_template.adcreativeElements?.find((item: { name: string }) => item.name === 'button_text') && !ok) {
  443. values = { ...values, buttonText: adcreative_template?.adcreativeElements?.find((item: { name: string }) => item.name === 'button_text')?.enumProperty?.enumeration[0].value }
  444. }
  445. // 视频结束页 end_page
  446. if (adcreative_template.adcreativeElements.some((item: { name: string }) => item.name === 'end_page')) {
  447. // let endPageType =adcreative_template?.adcreativeElements?.filter(item=>item.name === 'end_page_type')[0]?.enumProperty?.enumeration
  448. if (!ok) {
  449. values = { ...values, endPageType: 'END_PAGE_AVATAR_NICKNAME_HIGHLIGHT' }
  450. }
  451. states = { ...states, sp_show: true }
  452. }
  453. setPupState(states)
  454. form.setFieldsValue(values)
  455. }
  456. }
  457. // 版位改变清空数据
  458. useEffect(() => {
  459. if (imgMaterialConfig.adcreativeTemplateId && adcreativeTemplateId !== imgMaterialConfig.adcreativeTemplateId) {
  460. setImgMaterialConfig({ ...imgMaterialConfig, adcreativeTemplateId: undefined, list: [] })
  461. }
  462. if (videoMaterialConfig.adcreativeTemplateId && adcreativeTemplateId !== videoMaterialConfig.adcreativeTemplateId) {
  463. setVideoMaterialConfig({ ...videoMaterialConfig, adcreativeTemplateId: undefined, list: [] })
  464. }
  465. }, [adcreativeTemplateId, imgMaterialConfig, videoMaterialConfig])
  466. // 文案助手
  467. const textList = useCallback((arg: { maxTextLength: number, keyword?: string }) => {
  468. let { maxTextLength, keyword } = arg
  469. getTextLsit.run({ keyword: keyword || titles || description, maxTextLength })
  470. }, [titles, description])
  471. // 监听点击取消文案助手弹窗
  472. useEffect(() => {
  473. let modal = document.querySelector('.myModal')
  474. let onBiurdescription = (e: any) => {
  475. let p = document.querySelector('.my_endPageDesc')
  476. if (!p?.contains(e.target)) {
  477. setendPageDescnshow(false)
  478. }
  479. }
  480. modal?.addEventListener('click', onBiurdescription)
  481. return () => {
  482. modal?.removeEventListener('click', onBiurdescription)
  483. }
  484. }, [])
  485. // 数据回填
  486. useEffect(() => {
  487. if (!infoSet && dataInfo && adcreative_template_list?.length > 0) {
  488. let { adcreativeName, adcreativeTemplateId, conversionDataType, conversionTargetType, linkNameType, linkPageType, pageType, promotedObjectType, siteSet, profile, adcreativeElements, overrideCanvasHeadOption, linkPageSpec } = dataInfo
  489. let { description, imageUrl, title, videoUrl, imageUrlList, endPage, shortVideoStruct, brand, buttonText } = adcreativeElements
  490. let obj: any = {
  491. adcreativeName,
  492. siteSet,
  493. promotedObjectType,
  494. adcreativeTemplateId,
  495. }
  496. getTemplate(adcreativeTemplateId, true)
  497. console.log(2222, dataInfo)
  498. if ([720, 721, 618, 1708, 722, 1529].some(n => n === adcreativeTemplateId)) {
  499. obj = { ...obj, adcreativeElementsType: '视频' }
  500. } else {
  501. obj = { ...obj, adcreativeElementsType: '图片' }
  502. }
  503. if (conversionDataType) {
  504. obj = { ...obj, conversionDataType, dataShow: true }
  505. }
  506. if (conversionTargetType) {
  507. obj = { ...obj, conversionTargetType, dataShow: true }
  508. }
  509. if (linkNameType) {
  510. obj = { ...obj, linkNameType, actionBtn: true }
  511. }
  512. if (linkPageType) {
  513. obj = { ...obj, linkPageType, actionBtn: true }
  514. }
  515. if (pageType) {
  516. obj = { ...obj, pageType }
  517. }
  518. if (description) {
  519. obj = { ...obj, description }
  520. }
  521. if (title) {
  522. obj = { ...obj, title }
  523. }
  524. if (endPage) {
  525. obj = { ...obj, videoOver: true, ...endPage }
  526. }
  527. if (overrideCanvasHeadOption) {
  528. obj = { ...obj, overrideCanvasHeadOption }
  529. }
  530. if (linkPageSpec?.pageUrl) {
  531. obj = { ...obj, pageUrl: linkPageSpec?.pageUrl }
  532. }
  533. if (linkPageSpec?.miniProgramSpec && linkPageSpec?.miniProgramSpec?.miniProgramPath) {
  534. obj = { ...obj, miniProgramPath: linkPageSpec?.miniProgramSpec?.miniProgramPath, miniProgramId: linkPageSpec?.miniProgramSpec?.miniProgramId }
  535. }
  536. if (brand && brand.brandImgUrl && brand.brandName) {
  537. obj = { ...obj, brand: brand.brandName + '_' + brand.brandImgUrl }
  538. }
  539. if (profile && profile.headImageUrl && profile.profileName && profile.description) {
  540. obj = { ...obj, profile: profile.profileName + '_' + profile.headImageUrl + '_' + profile.description }
  541. }
  542. if (buttonText) {
  543. obj = { ...obj, buttonText }
  544. }
  545. if (videoUrl) {
  546. setVideoMaterialConfig({
  547. cloudSize: [],
  548. list: [{ url: videoUrl }],
  549. max: 1,
  550. type: 'video',
  551. adcreativeTemplateId
  552. })
  553. obj = { ...obj, video: videoUrl }
  554. }
  555. if (imageUrl) {
  556. setImgMaterialConfig({
  557. cloudSize: [],
  558. list: [{ url: imageUrl }],
  559. max: 1,
  560. type: 'image',
  561. adcreativeTemplateId
  562. })
  563. obj = { ...obj, image: imageUrl }
  564. }
  565. if (imageUrlList) {
  566. setImgMaterialConfig({
  567. cloudSize: [],
  568. list: imageUrlList?.map((url: any) => ({ url })),
  569. max: imageUrlList.length,
  570. type: 'image_list',
  571. adcreativeTemplateId
  572. })
  573. obj = { ...obj, image_list: imageUrlList }
  574. }
  575. if (shortVideoStruct) {
  576. setVideoMaterialConfig({
  577. cloudSize: [],
  578. list: [{ url: shortVideoStruct.shortVideo1Url }],
  579. max: 1,
  580. type: 'short_video1',
  581. adcreativeTemplateId
  582. })
  583. obj = { ...obj, short_video1: shortVideoStruct.shortVideo1Url }
  584. }
  585. console.log('数据回填====>', obj)
  586. form.setFieldsValue(obj)
  587. set_infoSet(true)
  588. }
  589. // 不是数据回填首次打开界面选中视频
  590. if (!infoSet && !dataInfo && adcreative_template_list?.length > 0) {
  591. typeChange('视频')
  592. set_infoSet(true)
  593. }
  594. }, [dataInfo, adcreative_template_list, adcreative_template, infoSet])
  595. return <Modal
  596. visible={visible}
  597. title={type === 'add' ? '新建创意' : type === 'look' ? '创意详情' : '编辑创意'}
  598. onCancel={() => { PupFn({ visible: false, dataInfo: null, type: 'add' }) }}
  599. // onOk={handleOk}
  600. width={1200}
  601. confirmLoading={confirmLoading}
  602. footer={<Space>
  603. <Button onClick={() => { PupFn({ visible: false, dataInfo: null, type: 'add' }) }}>取消</Button>
  604. <Button type='primary' onClick={handleOk}>确定</Button>
  605. </Space>}
  606. className='myModal'
  607. {...arg}
  608. >
  609. <Form
  610. form={form}
  611. labelCol={{ span: 5 }}
  612. labelWrap={true}
  613. className='ad_form_style'
  614. initialValues={
  615. {
  616. adcreativeElementsType: '视频',
  617. }
  618. }
  619. >
  620. {/* ============================================================创意形式============================================================= */}
  621. <Divider orientation='center'>创意形式</Divider>
  622. {/* ============================================================创意形式============================================================= */}
  623. <Form.Item label={<strong>创意形式</strong>} name='adcreativeElementsType'>
  624. <Radio.Group onChange={(e) => {
  625. let value = e.target.value
  626. typeChange(value)
  627. }}>
  628. <Radio.Button value="视频">视频</Radio.Button>
  629. <Radio.Button value="图片">图片</Radio.Button>
  630. </Radio.Group>
  631. </Form.Item>
  632. {
  633. getAdcreativeTemplateList?.loading ? <Spin tip="Loading..." style={{ width: '100%' }}></Spin> :
  634. <>
  635. <Form.Item style={{ marginLeft: 177 }} name='adcreativeTemplateId'>
  636. <Radio.Group className={styles.adcreative_template} onChange={(e) => {
  637. let id = e.target.value
  638. getTemplate(id)
  639. }}>
  640. {adcreative_template_list?.filter(item => item.adcreativeTemplateStyle === adcreativeElementsType && item.supportBidModeList.includes(queryForm?.sysAdgroup?.bidMode))?.map((item: any) => {
  641. return <Radio.Button value={item.adcreativeTemplateId} key={item.adcreativeTemplateId}>
  642. <div className={styles.adcreative_template_item}>
  643. {item.isGeneral && <span style={{ color: '#4080ff', fontSize: 10 }}>所选版位通投</span>}
  644. <img src={item.adcreativeSampleImage} />
  645. <span style={{ fontSize: 12, height: 20, lineHeight: '20px' }}>{item.adcreativeTemplateAppellation}</span>
  646. <span style={{ fontSize: 12, height: 20, lineHeight: '20px' }}>{item.adcreativeTemplateId}</span>
  647. </div>
  648. </Radio.Button>
  649. })}
  650. </Radio.Group>
  651. </Form.Item>
  652. {/* ============================================================创意内容============================================================= */}
  653. <Divider orientation='center'>创意内容</Divider>
  654. {/* =============================================================头像及昵称跳转页===================================================================== */}
  655. {queryForm.promotedObjectType === 'PROMOTED_OBJECT_TYPE_LEAD_AD' ? adcreative_template?.adcreativeAttributes?.find(item => item.name === 'profile_id') ? <Form.Item label={<strong>头像及昵称跳转页</strong>} name='profile' rules={[{ required: true, message: '请选择一个头像及昵称跳转页,与广告创意一起展示' }]}>
  656. <HeadNickJump />
  657. </Form.Item> : <Form.Item label={<strong>品牌形象</strong>} name='brand' rules={[{ required: true, message: '请选择一个头像及昵称跳转页,与广告创意一起展示' }]}>
  658. <BrandImage />
  659. </Form.Item> : null}
  660. {/* ============================================================落地页============================================================= */}
  661. {adcreativeTemplateId ? <Form.Item label={<strong>落地页</strong>} name='pageType' >
  662. <Radio.Group>
  663. {pageTypeList?.map((item: any) => {
  664. return <Radio.Button value={item.pageType} key={item.pageType} disabled={!item.description.includes('微信原生推广页')}>{item.description.includes('微信原生推广页') ? '微信原生推广页' : item.description}</Radio.Button>
  665. })}
  666. </Radio.Group>
  667. </Form.Item> : <div style={{ minHeight: 400, display: 'flex', justifyContent: 'center', alignItems: 'center' }}>
  668. <Empty description="请先选择创意形式" />
  669. </div>}
  670. {
  671. pageType === 'PAGE_TYPE_CANVAS_WECHAT' && isShowSc && <Form.Item label={<strong>素材选项</strong>} name='overrideCanvasHeadOption'>
  672. <Radio.Group disabled>
  673. {adcreativeTemplateId && creativeConfig[adcreativeTemplateId]?.overrideCanvasHeadOption?.map((item: string | number) => {
  674. return <Radio value={item} key={item}>{overrideCanvasHeadOptionEnum[item]}</Radio>
  675. })}
  676. </Radio.Group>
  677. </Form.Item>
  678. }
  679. {/* ============================================================普通行动按钮============================================================= */}
  680. {
  681. pupState.xd_show && <Form.Item hidden={isShowXd} label={<strong>行动按钮</strong>} name='actionBtn' valuePropName="checked">
  682. <Switch checkedChildren="开启" unCheckedChildren="关闭" disabled={isShowXd}/>
  683. </Form.Item>
  684. }
  685. {
  686. actionBtn && <>
  687. <Form.Item name='linkNameType' label={<strong>按钮文案</strong>}>
  688. <Select style={{ width: 200 }} showSearch filterOption={(input, option) =>
  689. (option!.children as unknown as string).toLowerCase().includes(input.toLowerCase())
  690. } allowClear>
  691. {linkNameList?.map((item: any) => {
  692. return <Select.Option value={item.linkNameType} key={item.linkNameType}>{item.description}</Select.Option>
  693. })}
  694. </Select>
  695. </Form.Item>
  696. <Form.Item label={<strong>跳转落地页</strong>}>
  697. <Form.Item name='linkPageType' noStyle>
  698. <Radio.Group style={{ display: 'flex' }}>
  699. {linkPageList?.map((item: { linkPageType: string; description: string; }, index: number) => {
  700. return <Radio.Button value={item.linkPageType} key={item.linkPageType} >{item.description}</Radio.Button>
  701. })}
  702. </Radio.Group>
  703. </Form.Item>
  704. {/* 自定义落地页地址 */}
  705. {linkPageType === "LINK_PAGE_TYPE_DEFAULT" && <Form.Item name='pageUrl' rules={[{ required: true, message: '请输入自定义落地页地址' }]} style={{ marginTop: 10, marginBottom: 0 }}>
  706. <Input placeholder='请输入自定义落地页地址' style={{ width: 300 }} />
  707. </Form.Item>}
  708. {/* 小程序 */}
  709. {linkPageType === "LINK_PAGE_TYPE_MINI_PROGRAM_WECHAT" && <Form.Item noStyle >
  710. <Form.Item rules={[{ required: true, message: '请输入小程序原始ID' }]} name='miniProgramId' style={{ marginTop: 10, marginBottom: 0 }} >
  711. <Input placeholder='请输入小程序原始ID' style={{ width: 300 }} />
  712. </Form.Item>
  713. <Form.Item rules={[{ required: true, message: '请输入小程序链接' }]} name='miniProgramPath' style={{ marginTop: 10, marginBottom: 0 }}>
  714. <Input placeholder='请输入小程序链接' style={{ width: 300 }} />
  715. </Form.Item>
  716. </Form.Item>}
  717. </Form.Item>
  718. {/* 落地页 */}
  719. </>
  720. }
  721. {/* ============================================================特殊行动按钮============================================================= */}
  722. {
  723. !pupState.xd_show && adcreative_template?.adcreativeElements?.find(item => item.name === 'button_text') && <Form.Item label={<strong>行动按钮</strong>} >
  724. <Form.Item valuePropName="checked" noStyle >
  725. <Switch checkedChildren="开启" unCheckedChildren="关闭" checked={true} disabled defaultChecked={true} />
  726. </Form.Item>
  727. </Form.Item>
  728. }
  729. {
  730. !pupState.xd_show && adcreative_template?.adcreativeElements?.find(item => item.name === 'button_text') && <Form.Item name='buttonText' label={<strong>按钮文案</strong>} rules={[{ required: true, message: '请选择按钮文案!' }]}>
  731. <Select style={{ width: 200 }} showSearch filterOption={(input, option) =>
  732. (option!.children as unknown as string).toLowerCase().includes(input.toLowerCase())
  733. } allowClear>
  734. {adcreative_template?.adcreativeElements?.find(item => item.name === 'button_text')?.enumProperty?.enumeration?.map((item: any) => {
  735. return <Select.Option value={item.value} key={item.value}>{item.value}</Select.Option>
  736. })}
  737. </Select>
  738. </Form.Item>
  739. }
  740. {/* ============================================================数据展示============================================================= */}
  741. {pupState.sj_show && <Form.Item label={<strong>数据展示</strong>} name='dataShow' valuePropName="checked">
  742. <Switch checkedChildren="开启" unCheckedChildren="关闭" />
  743. </Form.Item>}
  744. {
  745. dataShow && <>
  746. <Form.Item name='conversionDataType' label={<strong>数据类型</strong>}>
  747. <Radio.Group>
  748. {conversionList?.conversion_data_type?.map((item: { value: string; description: string; }, index: number) => {
  749. return <Radio.Button value={item.value} key={item.value}>{item.description}</Radio.Button>
  750. })}
  751. </Radio.Group>
  752. </Form.Item>
  753. {conversionList?.conversion_target_type && conversionDataType === 'CONVERSION_DATA_ADMETRIC' && <Form.Item name='conversionTargetType' label={<strong>转化行为</strong>}>
  754. <Radio.Group>
  755. {conversionList?.conversion_target_type?.map((item: { value: string; description: string; }, index: number) => {
  756. return <Radio.Button value={item.value} key={item.value}>{item.description}</Radio.Button>
  757. })}
  758. </Radio.Group>
  759. </Form.Item>}
  760. </>
  761. }
  762. {/* ============================================================视频结束页============================================================= */}
  763. {pupState.sp_show && <Form.Item label={<strong>视频结束页</strong>} name='videoOver' valuePropName="checked">
  764. <Switch checkedChildren="开启" unCheckedChildren="关闭" />
  765. </Form.Item>}
  766. {
  767. videoOver && <>
  768. <Form.Item name='endPageType' label={<strong>视频结束页类型</strong>} >
  769. <Radio.Group>
  770. {
  771. adcreative_template?.adcreativeElements?.filter(item => item.name === 'end_page_type')[0]?.enumProperty?.enumeration?.map((item) => {
  772. return <Radio.Button value={item.value} key={item.value}>{item.description}</Radio.Button>
  773. })
  774. }
  775. </Radio.Group>
  776. </Form.Item>
  777. <div className={'my_endPageDesc'} >
  778. <Form.Item label={<strong>结束文案</strong>}>
  779. <Form.Item name='endPageDesc' rules={[{ required: true, pattern: RegExp("^[^\\<\\>\\&'\\\"\\/\\x08\\x09\\x0A\\x0D\\\\]{1,12}$"), message: '请输入正确的结束页文案' }]} noStyle>
  780. <Input
  781. placeholder='请输入结束页文案'
  782. style={{ width: 300 }}
  783. onFocus={() => {
  784. setendPageDescnshow(true)
  785. textList({ maxTextLength: 12 })
  786. }}
  787. onChange={(e) => {
  788. let value = e.target.value
  789. textList({ maxTextLength: 12, keyword: value })
  790. }}
  791. allowClear
  792. />
  793. </Form.Item>
  794. <span>{endPageDesc?.length || 0}/12</span>
  795. {
  796. endPageDescShow && <List
  797. loading={getTextLsit?.loading}
  798. size="small"
  799. style={{ maxHeight: 300, maxWidth: 300, overflowX: 'auto' }}
  800. bordered
  801. dataSource={getTextLsit?.data?.returnTexts}
  802. renderItem={(item: any) => <List.Item onClick={(e: any) => {
  803. form.setFieldsValue({ endPageDesc: item.text })
  804. setendPageDescnshow(false)
  805. }}><span >{item.text}{item.tag && <span className={styles.crt}>{'CTR 高'}</span>}</span></List.Item>}
  806. />
  807. }
  808. </Form.Item>
  809. </div>
  810. </>
  811. }
  812. </>
  813. }
  814. {/* ============================================================基本信息============================================================= */}
  815. <Divider orientation='center'>基本信息</Divider>
  816. {/* ============================================================创意名称============================================================= */}
  817. <Form.Item label={<strong>创意名称</strong>} name='adcreativeName' rules={[{ required: true, message: '请输入广告名称!' }]}>
  818. <Input placeholder='创意名称' style={{ width: 300 }} />
  819. </Form.Item>
  820. </Form>
  821. {/* 选择图片素材 */}
  822. {
  823. selectImgVisible && <SelectCloud
  824. visible={selectImgVisible}
  825. onClose={() => {
  826. set_selectImgVisible(false)
  827. }}
  828. sliderImgContent={imgMaterialConfig.list}
  829. onChange={(content) => {
  830. if (content.length > 0) {
  831. form.setFieldsValue({ [imgMaterialConfig.type]: imgMaterialConfig.type })
  832. }
  833. setImgMaterialConfig({ ...imgMaterialConfig, list: content })
  834. set_selectImgVisible(false)
  835. }} />
  836. }
  837. {/* 选择视频素材 */}
  838. {
  839. selectVideoVisible && <SelectCloud
  840. visible={selectVideoVisible}
  841. onClose={() => set_selectVideoVisible(false)}
  842. sliderImgContent={videoMaterialConfig.list}
  843. onChange={(content) => {
  844. if (content.length > 0) {
  845. form.setFieldsValue({ [videoMaterialConfig.type]: videoMaterialConfig.type })
  846. }
  847. setVideoMaterialConfig({ ...videoMaterialConfig, list: content })
  848. set_selectVideoVisible(false)
  849. }} />
  850. }
  851. {/* 视频封面图弹窗 */}
  852. {
  853. videoImgsVisbile && <Modal
  854. visible={videoImgsVisbile}
  855. title={<div>生成封面图 <Switch checkedChildren="开启预览" unCheckedChildren="关闭预览" checked={videoImgs.preview} onChange={(checked) => { set_videoImgs({ ...videoImgs, preview: checked }) }} /></div>}
  856. onOk={() => {
  857. if (videoImgs.activeUrl) {
  858. setImgMaterialConfig({ ...imgMaterialConfig, list: [{ url: videoImgs.activeUrl }] })
  859. set_videoImgsVisbile(false)
  860. } else {
  861. message.error('请选择图片,获取使用取消按钮关闭弹窗!')
  862. }
  863. }}
  864. onCancel={() => { set_videoImgsVisbile(false) }}
  865. confirmLoading={getVideoCapture.loading}
  866. width={600}
  867. >
  868. <Radio.Group className={styles.videoImgs} onChange={(e) => {
  869. let url = e.target.value
  870. set_videoImgs({ ...videoImgs, activeUrl: url })
  871. }}>
  872. {
  873. videoImgs?.urlList?.map((item: any, index: number) => {
  874. return <Radio.Button value={item} key={index}>
  875. <Image src={item} preview={videoImgs.preview} />
  876. </Radio.Button>
  877. })
  878. }
  879. </Radio.Group>
  880. </Modal>
  881. }
  882. </Modal >
  883. }
  884. export default CreativePup