VideoAudioCard.tsx 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265
  1. import React, { useState } from 'react'
  2. import useRequest from '@ahooksjs/use-request/es'
  3. import dayjs from 'dayjs'
  4. import { Modal, message } from 'antd'
  5. import { StarFilled, StarOutlined } from '@ant-design/icons'
  6. import videoIcon from 'assets/video-icon.svg'
  7. import audioIcon from 'assets/audio-icon.svg'
  8. import stopIcon from 'assets/audio_stop.png'
  9. import audioPlay from 'assets/audio_play.png'
  10. import videoPlay from 'assets/video-play.svg'
  11. import { default as CommentSvg } from 'assets/comment.svg'
  12. import { MaterialService } from 'Material/Material.service'
  13. import { MicroRoadshowService, IRoadshowistItem } from 'Material/MicroRoadshow.service'
  14. import AskAdd from './AskAdd'
  15. import { INewPermissionType, ITryType } from './NoPermission'
  16. import CategoryTag from 'Indepth/components/CategoryTag'
  17. import styles from '../css/ItemComponent.module.scss'
  18. export interface IVideoAudioCard {
  19. item: IRoadshowistItem
  20. audioPlayId?: number
  21. title?: string // 标红的标题
  22. onItemClick?: () => void
  23. onCollect?: (ID: number) => void // 收藏与取消收藏
  24. }
  25. export enum ILabelType {
  26. RoadshowReplay = 1,
  27. SurveyFeedback = 2,
  28. IndustryVideo = 3,
  29. QASeries = 4
  30. }
  31. const labelOption = [
  32. { key: ILabelType.RoadshowReplay, name: '路演回放' },
  33. { key: ILabelType.SurveyFeedback, name: '调研反馈' },
  34. { key: ILabelType.IndustryVideo, name: '产业视频' },
  35. { key: ILabelType.QASeries, name: '问答系列' }
  36. ]
  37. /**音视频卡片组件 */
  38. const VideoAudioCard: React.FC<IVideoAudioCard> = props => {
  39. const { item, audioPlayId, title, onItemClick, onCollect } = props
  40. const [visibleComment, setVisibleComment] = useState(false)
  41. const [videoItem, setVideoItem] = useState(item)
  42. const [commentPlaceholder, setCommentPlaceholder] = useState('可以留下您对视频内容的看法或者疑问')
  43. const placeholderOption = { 1: '可以留下您对音频内容的看法或者疑问', 2: '可以留下您对视频内容的看法或者疑问' }
  44. // 留言
  45. const { run: applyComment } = useRequest(MicroRoadshowService.postRoadshowComment, {
  46. manual: true,
  47. onSuccess: res => {
  48. res.data.Success ? message.success(res.data.Msg) : message.error(res.data.ErrMsg)
  49. }
  50. })
  51. // 收藏与取消收藏
  52. const { run: applyCollect } = useRequest(MicroRoadshowService.postRoadshowCollect, {
  53. manual: true,
  54. onSuccess: res => {
  55. res.data.Success ? message.success(res.data.Msg) : message.error(res.data.ErrMsg)
  56. if (res.data.Success) {
  57. onCollect && onCollect(item.Id)
  58. const newItem = JSON.parse(JSON.stringify(videoItem))
  59. newItem.IsCollect = !newItem.IsCollect
  60. setVideoItem(newItem)
  61. }
  62. }
  63. })
  64. const { run: applyTry } = useRequest(MaterialService.postApplyTry, {
  65. manual: true,
  66. onSuccess: res => {
  67. res.data.Ret === 200 ? message.success('提交成功,请等待销售人员与您联系') : message.info(res.data.Msg)
  68. }
  69. })
  70. // 打开留言
  71. const handleOpenComment = (e: React.MouseEvent<HTMLElement, MouseEvent>) => {
  72. e.stopPropagation()
  73. setVisibleComment(true)
  74. if (item.Type === 1) {
  75. setCommentPlaceholder(placeholderOption[1])
  76. return
  77. }
  78. setCommentPlaceholder(placeholderOption[2])
  79. }
  80. // 收藏与取消收藏
  81. const handleCollectOrNot = (e: React.MouseEvent<HTMLElement, MouseEvent>) => {
  82. // todo
  83. e.stopPropagation()
  84. applyCollect(item.SourceId || item.Id, item.Type)
  85. }
  86. // Type: number // 类型 1-音频; 2-活动视频; 3-产业视频
  87. // Source: string //资源类型 报告 :article 、图表 :newchart、微路演 :roadshow、活动 :activity、活动视频:activityvideo、活动音频:activityvoice、专项调研活动:activityspecial"
  88. const handleOkAsk = (e?: React.MouseEvent<HTMLElement, MouseEvent>) => {
  89. // e?.stopPropagation()
  90. setVisibleComment(false)
  91. }
  92. const handleApplyComment = (ID: number, content: string) => {
  93. applyComment(ID, item.Type, content)
  94. handleOkAsk()
  95. }
  96. // 标题与播放按钮的点击
  97. const handleToSomePage = (item: IRoadshowistItem, type: 'detail' | 'play') => {
  98. // e.stopPropagation()
  99. // 类型 1-音频-活动详情页; 2-活动视频-活动详情页; 3-产业视频-产业详情页,4-问答
  100. const urlType: {
  101. [key: number]: () => { url: string; tryType: ITryType; text: string }
  102. } = {
  103. 1: () => ({
  104. url: type === 'detail' ? `${window.location.origin}/activity/detail/${item.ActivityId}` : '',
  105. tryType: ITryType.MicroAudio,
  106. text: '音频'
  107. }),
  108. 2: () => ({
  109. url:
  110. type === 'detail'
  111. ? `${window.location.origin}/activity/detail/${item.ActivityId}`
  112. : `${window.location.origin}/activity/video/${item.Id}`,
  113. tryType: ITryType.ActivityVideo,
  114. text: '视频'
  115. }),
  116. 3: () => ({
  117. url:
  118. type === 'detail'
  119. ? `${window.location.origin}/indepth/info/${item.ChartPermissionId}/${item.IndustrialManagementId}`
  120. : `${window.location.origin}/indepth/video/${item.ChartPermissionId}/${item.IndustrialManagementId}/${item.Id}`,
  121. tryType: ITryType.MicroVideo,
  122. text: '视频'
  123. }),
  124. 4: () => ({
  125. url: '',
  126. tryType: ITryType.MicroAudio,
  127. text: '音频' // 暂时是音频
  128. })
  129. }
  130. if (item.AuthInfo.HasPermission !== INewPermissionType.OK) {
  131. let content: React.ReactNode = (
  132. <div>
  133. <div>您暂无权限查看此{urlType[item.Type]().text}</div>
  134. <div>若想查看可以联系对口销售</div>
  135. {item.AuthInfo?.SellerName && (
  136. <div>
  137. {item.AuthInfo?.SellerName}:{item.AuthInfo?.SellerMobile}
  138. </div>
  139. )}
  140. <div>申请开通对应的试用权限</div>
  141. </div>
  142. )
  143. if (
  144. item.AuthInfo.HasPermission === INewPermissionType.HasApplyPotential ||
  145. item.AuthInfo.HasPermission === INewPermissionType.NoApplyPotential ||
  146. item.AuthInfo.HasPermission === INewPermissionType.HasApplyFICC ||
  147. item.AuthInfo.HasPermission === INewPermissionType.NoApplyFICC
  148. ) {
  149. content = (
  150. <div>
  151. <p>您暂无权限查看此{urlType[item.Type]().text},若想查看可以申请开通对应的试用权限</p>
  152. </div>
  153. )
  154. }
  155. // 上传名片时-7,ApplyMethod为2,否则为1
  156. Modal.confirm({
  157. title: '提示',
  158. content: content,
  159. okText: '提交申请',
  160. cancelText: '取消',
  161. centered: true,
  162. closable: true,
  163. onOk: () => {
  164. applyTry({
  165. ApplyMethod: item.AuthInfo.HasPermission === INewPermissionType.NoApplyPotential ? 2 : 1,
  166. DetailId: item.Id,
  167. TryType: urlType[item.Type]().tryType
  168. })
  169. }
  170. })
  171. return
  172. }
  173. // 音频播放按钮点击抛出
  174. if (isAudioOrVideo(item) && type === 'play') {
  175. onItemClick && onItemClick()
  176. return
  177. }
  178. const url = urlType[item.Type]().url
  179. if (!url) return
  180. window.open(url)
  181. }
  182. // 判断是音频还是视频(返回true是音频)
  183. const isAudioOrVideo = (item: IRoadshowistItem) => {
  184. return item.Type === 1 || item.Type === 4
  185. }
  186. const formatLabel = (label: number) => {
  187. const labelItem = labelOption.find(item => item.key === label)
  188. return labelItem?.name || ''
  189. }
  190. if (!item) return null
  191. return (
  192. <div className={`${styles['roadshow-item']}`}>
  193. <div
  194. className={`roadshow-item-title ${item.Type === 4 ? '' : 'com-cursor-p'}`}
  195. onClick={handleToSomePage.bind(this, item, 'detail')}
  196. dangerouslySetInnerHTML={{
  197. __html: item?.LabelType
  198. ? `<span class="card-tag-lyhf no-wrap">${formatLabel(item.LabelType)}</span><span>${
  199. title || item?.Title
  200. }</span>`
  201. : title || item?.Title
  202. }}
  203. />
  204. <div className="video-line-box">
  205. <div className="video-img-wrapper">
  206. <img src={item.BackgroundImg} alt="图像" className="video-img-bg" />
  207. <div className={`right-top-bg ${isAudioOrVideo(item) ? 'right-top-blue' : 'right-top-orange'}`} />
  208. <img alt="图像" className="right-top-icon" src={isAudioOrVideo(item) ? audioIcon : videoIcon} />
  209. {isAudioOrVideo(item) ? (
  210. <img
  211. alt="图像"
  212. src={audioPlayId && audioPlayId === item.SourceId ? stopIcon : audioPlay}
  213. className="play-center"
  214. onClick={handleToSomePage.bind(this, item, 'play')}
  215. />
  216. ) : (
  217. <img
  218. alt="图像"
  219. src={videoPlay}
  220. className="play-center"
  221. onClick={handleToSomePage.bind(this, item, 'play')}
  222. />
  223. )}
  224. <span className="video-play-time">
  225. {dayjs(parseInt(item.PlaySeconds || '0', 10) * 1000 - 28800000).format(
  226. parseInt(item.PlaySeconds || '0', 10) >= 3600 ? 'HH:mm:ss' : 'mm:ss'
  227. )}
  228. </span>
  229. </div>
  230. </div>
  231. <div className="roadshow-footer">
  232. <div className="video-time">
  233. <CategoryTag content={item.ChartPermissionName} className="m-r-xs" />
  234. <span className="video-publishtime">{item.PublishTime}</span>
  235. </div>
  236. <div className="roadshow-right">
  237. <img src={CommentSvg} alt="图标" className="comment-icon" onClick={handleOpenComment} />
  238. {videoItem?.IsCollect ? (
  239. <StarFilled size={14} className="collect-icon " onClick={handleCollectOrNot} />
  240. ) : (
  241. <StarOutlined size={14} className="not-collect-icon" onClick={handleCollectOrNot} />
  242. )}
  243. </div>
  244. </div>
  245. <div onClick={e => e.stopPropagation()}>
  246. <AskAdd
  247. visible={visibleComment}
  248. ID={item?.SourceId || item.Id}
  249. placeholder={commentPlaceholder}
  250. onApply={handleApplyComment}
  251. onCloseModel={handleOkAsk}
  252. />
  253. </div>
  254. </div>
  255. )
  256. }
  257. export default VideoAudioCard