index.vue 81 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199
  1. <template>
  2. <div class="p-2">
  3. <transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave">
  4. <div v-show="showSearch" class="mb-[20px]">
  5. <el-card shadow="hover">
  6. <el-form ref="queryFormRef" :model="queryParams" :inline="true">
  7. <el-form-item label="比赛ID" prop="name">
  8. <el-input v-model="queryParams.id" placeholder="请输入比赛ID" clearable @keyup.enter="handleQuery" />
  9. </el-form-item>
  10. <el-form-item label="比赛名称" prop="name">
  11. <el-input v-model="queryParams.name" placeholder="请输入比赛名称" clearable @keyup.enter="handleQuery" />
  12. </el-form-item>
  13. <el-form-item label="比赛状态" prop="gameType">
  14. <el-select aria-required="true" v-model="queryParams.status" placeholder="请选择" :disabled="dialog.mode === 'view'">
  15. <!-- 添加"全部"选项 -->
  16. <el-option :value="null" :label="'全部'" />
  17. <el-option v-for="dict in tournaments_status" :key="dict.value" :label="dict.label" :value="dict.value"> </el-option>
  18. </el-select>
  19. </el-form-item>
  20. <el-form-item label="开始时间" prop="startTime">
  21. <el-date-picker clearable v-model="queryParams.startTime" type="date" value-format="YYYY-MM-DD" placeholder="请选择比赛开始时间" />
  22. </el-form-item>
  23. <el-form-item>
  24. <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
  25. <el-button icon="Refresh" @click="resetQuery">重置</el-button>
  26. </el-form-item>
  27. </el-form>
  28. </el-card>
  29. </div>
  30. </transition>
  31. <el-card shadow="never">
  32. <template #header>
  33. <el-row :gutter="10" class="mb8">
  34. <el-col :span="1.5">
  35. <el-button type="primary" plain icon="Plus" @click="handleAdd" v-hasPermi="['business:tournaments:add']">创立比赛</el-button>
  36. </el-col>
  37. <!-- <el-col :span="1.5">
  38. <el-button type="success" plain icon="Edit" :disabled="single" @click="handleUpdate()" v-hasPermi="['business:tournaments:edit']"
  39. >编辑</el-button
  40. >
  41. </el-col>-->
  42. <!-- <el-col :span="1.5">
  43. <el-button type="danger" plain icon="Delete" :disabled="multiple" @click="handleDelete()" v-hasPermi="['business:tournaments:remove']"
  44. >删除</el-button
  45. >service/catory
  46. </el-col>-->
  47. <el-col :span="1.5">
  48. <el-button type="warning" plain icon="Download" @click="handleExport" v-hasPermi="['business:tournaments:export']">导出</el-button>
  49. </el-col>
  50. <el-col :span="1.5">
  51. <el-button type="warning" plain icon="Operation">
  52. <router-link to="/service/catory">比赛类目管理</router-link>
  53. </el-button>
  54. </el-col>
  55. <el-col :span="1.5">
  56. <el-button type="success" plain icon="Operation">
  57. <router-link to="/service/tag">比赛标签管理</router-link>
  58. </el-button>
  59. </el-col>
  60. <!-- 新增的 “盲注管理” 按钮 -->
  61. <!-- <el-col :span="1.5">
  62. <el-button type="warning" plain icon="Operation" v-hasPermi="['business:tournaments:manager']">
  63. <router-link to="/tournament/structures">盲注表管理</router-link>
  64. </el-button>
  65. </el-col>
  66. <el-col :span="1.5">
  67. <el-button type="success" plain icon="Operation" v-hasPermi="['business:tournaments:manager']">
  68. <router-link to="/tournament/config">自动比赛管理</router-link>
  69. </el-button>
  70. </el-col>-->
  71. <right-toolbar v-model:showSearch="showSearch" @queryTable="getList"></right-toolbar>
  72. </el-row>
  73. </template>
  74. <el-table
  75. v-loading="loading"
  76. border
  77. :data="tournamentsList"
  78. @selection-change="handleSelectionChange"
  79. @sort-change="handleSortChange"
  80. :default-sort="{ prop: 'startTime', order: sortData.order }"
  81. >
  82. <!-- <el-table-column type="selection" width="55" align="center" />-->
  83. <el-table-column label="比赛ID" align="center" prop="id" width="70" />
  84. <el-table-column label="比赛名" align="center" prop="name" />
  85. <!-- <el-table-column label="比赛项目" align="center" prop="projectName" width="150" />-->
  86. <!-- <el-table-column label="比赛项目" align="center" prop="gameVariant">
  87. <template #default="scope">
  88. {{ getGameVariantText(scope.row.gameVariant) }}
  89. </template>
  90. </el-table-column>-->
  91. <!-- <el-table-column label="比赛Logo" align="center" width="90">
  92. <template #default="scope">
  93. <el-image
  94. v-if="scope.row.competitionIcon"
  95. :src="scope.row.competitionIcon"
  96. style="width: 40px; height: 40px; border-radius: 4px; cursor: zoom-in"
  97. :preview-src-list="[scope.row.competitionIcon]"
  98. :preview-teleported="true"
  99. fit="cover"
  100. />
  101. <span v-else></span>
  102. </template>
  103. </el-table-column>-->
  104. <!-- <el-table-column label="比赛背景" align="center" width="90">
  105. <template #default="scope">
  106. <el-image
  107. v-if="scope.row.competitionBg"
  108. :src="scope.row.competitionBg"
  109. style="width: 40px; height: 40px; border-radius: 4px; cursor: zoom-in"
  110. :preview-src-list="[scope.row.competitionBg]"
  111. :preview-teleported="true"
  112. fit="cover"
  113. />
  114. <span v-else></span>
  115. </template>
  116. </el-table-column>-->
  117. <el-table-column label="开始时间" align="center" prop="startTime" width="150" sortable="custom"> </el-table-column>
  118. <el-table-column label="结束时间" align="center" prop="endTime" width="150"> </el-table-column>
  119. <el-table-column label="报名人数" align="center" prop="signNum" width="80"> </el-table-column>
  120. <el-table-column label="总手数" align="center" prop="totalSignup" width="70"> </el-table-column>
  121. <!-- <el-table-column label="报名要求" align="center">
  122. <template #default="scope">
  123. {{
  124. scope.row.itemsName && scope.row.itemsNum
  125. ? `${scope.row.itemsName} x ${scope.row.itemsNum}`
  126. : scope.row.itemsName || scope.row.itemsNum || '—'
  127. }}
  128. </template>
  129. </el-table-column>
  130. <el-table-column label="比赛类目" align="center">
  131. <template #default="scope">
  132. <span v-if="scope.row.categoryName && scope.row.categoryName.length > 0">
  133. <el-tag v-for="(category, index) in scope.row.categoryName" :key="index" type="primary" style="margin-right: 4px">
  134. {{ category }}
  135. </el-tag>
  136. </span>
  137. <span v-else>-</span>
  138. </template>
  139. </el-table-column>
  140. <el-table-column label="比赛标签" align="center">
  141. <template #default="scope">
  142. <span v-if="scope.row.tagName && scope.row.tagName.length > 0">
  143. <el-tag v-for="(tag, index) in scope.row.tagName" :key="index" type="success" style="margin-right: 4px">
  144. {{ tag }}
  145. </el-tag>
  146. </span>
  147. <span v-else>-</span>
  148. </template>
  149. </el-table-column>
  150. <el-table-column label="截止盲注等级" align="center" prop="lateRegistrationLevel" />
  151. &lt;!&ndash; <el-table-column label="盲注表" align="center" prop="blindStructuresName" />&ndash;&gt;
  152. <el-table-column label="奖励" align="center">
  153. <template #default="scope">
  154. <div v-if="scope.row.itemsPrizeList && scope.row.itemsPrizeList.length > 0">
  155. &lt;!&ndash; 显示第一名奖励 &ndash;&gt;
  156. <div>
  157. 第{{ scope.row.itemsPrizeList[0].ranking }}名:{{ scope.row.itemsPrizeList[0].quantity }} {{ scope.row.itemsPrizeList[0].itemsName }}
  158. </div>
  159. &lt;!&ndash; 如果有更多奖励,用 tooltip 显示 &ndash;&gt;
  160. <el-tooltip v-if="scope.row.itemsPrizeList.length > 1" :content="getRewardTooltipContent(scope.row.itemsPrizeList)" placement="top">
  161. <span class="more-rewards">更多</span>
  162. </el-tooltip>
  163. </div>
  164. <span v-else>—</span>
  165. </template>
  166. </el-table-column>-->
  167. <!-- <el-table-column label="最小参赛人数" align="center" prop="minPlayers" />-->
  168. <el-table-column label="状态" align="center">
  169. <template #default="scope">
  170. <span
  171. :style="{
  172. color: scope.row.isDelete ? 'red' : ''
  173. }"
  174. >
  175. {{ scope.row.isDelete ? '已删除' : scope.row.statusText || '' }}
  176. </span>
  177. </template>
  178. </el-table-column>
  179. <el-table-column label="创建时间" align="center" prop="createdAt" width="150"> </el-table-column>
  180. <!-- <el-table-column label="是否删除" align="center">
  181. <template #default="scope">
  182. <span
  183. :style="{
  184. color: [null, undefined, '', ' '].includes(scope.row.isDelete) || scope.row.isDelete === false ? '' : 'red'
  185. }"
  186. >
  187. {{ [null, undefined, '', ' '].includes(scope.row.isDelete) || scope.row.isDelete === false ? '未删除' : '已删除' }}
  188. </span>
  189. </template>
  190. </el-table-column>-->
  191. <el-table-column label="操作" align="center" class-name="small-padding fixed-width" width="320">
  192. <template #default="scope">
  193. <div class="operation-buttons-vertical" v-if="scope.row.isDelete === true">
  194. <el-tooltip content="查看" placement="top" v-hasPermi="['business:tournaments:query']">
  195. <el-button link type="primary" icon="View" @click="openAuditDialog(scope.row.id, 'view')">查看</el-button>
  196. </el-tooltip>
  197. <el-tooltip content="复制" placement="top" v-hasPermi="['business:tournaments:copy']">
  198. <el-button link type="primary" icon="Files" @click="handleCopy(scope.row)"> 复制 </el-button>
  199. </el-tooltip>
  200. <!-- <el-tooltip content="恢复" placement="top">
  201. <el-button link type="primary" icon="Files" @click="handleRecoverTournament(scope.row)" v-hasPermi="['business:tournaments:query']"
  202. >恢复</el-button
  203. >
  204. </el-tooltip>-->
  205. <el-tooltip content="粉碎删除" placement="top">
  206. <el-button link type="primary" icon="Delete" @click="deleteAllData(scope.row)" v-hasPermi="['business:tournaments:query']"
  207. >粉碎删除</el-button
  208. >
  209. </el-tooltip>
  210. </div>
  211. <div class="operation-buttons-vertical" v-else>
  212. <el-tooltip content="查看" placement="top" v-hasPermi="['business:tournaments:query']">
  213. <el-button link type="primary" icon="View" @click="openAuditDialog(scope.row.id, 'view')">查看</el-button>
  214. </el-tooltip>
  215. <el-tooltip content="删除" v-if="scope.row.status === 0 && scope.row.isDelete != true" placement="top">
  216. <el-button link type="primary" icon="Delete" @click="handleDelete(scope.row)" v-hasPermi="['business:tournaments:query']"
  217. >删除</el-button
  218. >
  219. </el-tooltip>
  220. <el-tooltip content="编辑" v-if="scope.row.status === 0" placement="top" v-hasPermi="['business:tournaments:query']">
  221. <el-button link type="primary" icon="Edit" @click="handleUpdate(scope.row, 'edit')">编辑</el-button>
  222. </el-tooltip>
  223. <el-tooltip content="复制" placement="top" v-hasPermi="['business:tournaments:query']">
  224. <el-button link type="primary" icon="Files" @click="handleCopy(scope.row)"> 复制 </el-button>
  225. </el-tooltip>
  226. <el-tooltip content="复盘" placement="top" v-hasPermi="['business:tournaments:query']">
  227. <el-button link type="primary" icon="DocumentChecked" @click="handleViewPublicHistory(scope.row)"> 复盘 </el-button>
  228. </el-tooltip>
  229. <el-tooltip content="领奖审核" placement="top" v-hasPermi="['business:tournaments:query']">
  230. <el-button link type="primary" icon="DocumentChecked" @click="openAuditDialog(scope.row.id, 'audit')">领奖审核</el-button>
  231. </el-tooltip>
  232. <el-tooltip content="关闭" v-if="scope.row.status === 1" placement="top" v-hasPermi="['business:tournaments:query']">
  233. <el-button link type="primary" icon="Close" @click="closeSendTournamentOperate(scope.row)">关闭</el-button>
  234. </el-tooltip>
  235. <el-tooltip content="发布" v-if="scope.row.status === -1" placement="top" v-hasPermi="['business:tournaments:query']">
  236. <el-button link type="primary" icon="DocumentChecked" @click="publishToTournament2(scope.row)">发布</el-button>
  237. </el-tooltip>
  238. </div>
  239. </template>
  240. </el-table-column>
  241. </el-table>
  242. <pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" />
  243. </el-card>
  244. <!-- 添加或修改【请填写功能名称】对话框 -->
  245. <!-- <el-dialog :title="dialog.title" v-model="dialog.visible" width="500px" append-to-body>
  246. <el-form ref="tournamentsFormRef" :model="form" :rules="rules" label-width="80px">
  247. <el-form-item label="赛事名称" prop="name">
  248. <el-input v-model="form.name" placeholder="请输入赛事名称" />
  249. </el-form-item>
  250. <el-form-item label="比赛开始时间" prop="startTime">
  251. <el-date-picker clearable v-model="form.startTime" type="datetime" value-format="YYYY-MM-DD HH:mm:ss" placeholder="请选择比赛开始时间">
  252. </el-date-picker>
  253. </el-form-item>
  254. <el-form-item label="起始记分牌数量" prop="startingChips">
  255. <el-input v-model="form.startingChips" placeholder="请输入起始记分牌数量" />
  256. </el-form-item>
  257. <el-form-item label="级别持续时间" prop="levelDuration">
  258. <el-input v-model="form.levelDuration" placeholder="请输入级别持续时间" />
  259. </el-form-item>
  260. <el-form-item label="截止报名级别" prop="lateRegistrationLevel">
  261. <el-input v-model="form.lateRegistrationLevel" placeholder="请输入截止报名级别" />
  262. </el-form-item>
  263. <el-form-item label="最大参赛人数" prop="maxPlayers">
  264. <el-input v-model="form.maxPlayers" placeholder="请输入最大参赛人数" />
  265. </el-form-item>
  266. </el-form>
  267. <template #footer>
  268. <div class="dialog-footer">
  269. <el-button :loading="buttonLoading" type="primary" @click="submitForm">确 定</el-button>
  270. <el-button @click="cancel">取 消</el-button>
  271. </div>
  272. </template>
  273. </el-dialog>-->
  274. <el-dialog :title="dialog.title" v-model="dialog.visible" width="600px" append-to-body @close="cancel">
  275. <el-form ref="tournamentsFormRef" :model="form" :rules="rules" label-width="120px">
  276. <!-- <el-form-item label="比赛项目" prop="gameVariant">
  277. <el-select aria-required="true" v-model="form.gameVariant" placeholder="请选择" :disabled="dialog.mode === 'view'">
  278. <el-option v-for="dict in game_variant_type" :key="dict.value" :label="dict.label" :value="dict.value"> </el-option>
  279. </el-select>
  280. </el-form-item>-->
  281. <el-form-item label="比赛配置" prop="configId">
  282. <el-select v-model="form.configId" placeholder="选项" :disabled="dialog.mode === 'view'">
  283. <el-option v-for="item in configOptions" :key="item.id" :label="item.label" :value="item.id" />
  284. </el-select>
  285. </el-form-item>
  286. <el-form-item label="比赛类型" prop="gameType">
  287. <el-select aria-required="true" v-model="form.gameType" placeholder="请选择" :disabled="dialog.mode === 'view'">
  288. <el-option v-for="dict in tournaments_type" :key="dict.value" :label="dict.label" :value="dict.value"> </el-option>
  289. </el-select>
  290. </el-form-item>
  291. <!-- 赛事名称 -->
  292. <el-form-item label="比赛名称" prop="name">
  293. <el-input v-model="form.name" placeholder="请输入比赛名称" :disabled="dialog.mode === 'view'" />
  294. </el-form-item>
  295. <!-- 比赛图标 -->
  296. <el-form-item label="比赛图标" prop="icon">
  297. <div class="upload-container">
  298. <el-upload
  299. class="upload-icon"
  300. action="#"
  301. :on-change="handleIconChange"
  302. :on-remove="handleIconRemove"
  303. :file-list="fileList"
  304. :auto-upload="false"
  305. accept="image/*"
  306. :disabled="dialog.mode === 'view'"
  307. >
  308. <template #trigger>
  309. <el-button type="primary" :disabled="dialog.mode === 'view'">点击选择图标</el-button>
  310. </template>
  311. <!-- 预览图区域 -->
  312. <template #default>
  313. <div class="preview-area" @click="handlePreviewClick">
  314. <!-- 只有当 iconPreviewUrl 或 competitionIcon 存在时才显示 img -->
  315. <img
  316. v-if="iconPreviewUrl || competitionIcon"
  317. :src="iconPreviewUrl || competitionIcon"
  318. alt="预览图"
  319. style="max-width: 100px; max-height: 100px; margin-top: 10px; cursor: pointer"
  320. />
  321. <!-- 可选:无图时显示提示文字 -->
  322. <div v-else style="margin-top: 10px; color: #999"></div>
  323. </div>
  324. </template>
  325. <template #tip>
  326. <div class="el-upload__tip">
  327. <span v-if="fileList.length > 0">当前已选文件:{{ fileList[0].name }}</span>
  328. </div>
  329. </template>
  330. </el-upload>
  331. </div>
  332. </el-form-item>
  333. <el-form-item label="是否冠名" prop="isSponsor">
  334. <el-select v-model="form.isSponsor" placeholder="请选择" :disabled="dialog.mode === 'view'">
  335. <el-option label="是" :value="1"></el-option>
  336. <el-option label="否" :value="0"></el-option>
  337. </el-select>
  338. </el-form-item>
  339. <!-- 比赛图标 -->
  340. <el-form-item label="比赛背景" prop="icon" v-if="form.isSponsor === 1">
  341. <div class="upload-container">
  342. <el-upload
  343. class="upload-icon"
  344. action="#"
  345. :on-change="handleIconChange2"
  346. :on-remove="handleIconRemove2"
  347. :file-list="fileList2"
  348. :auto-upload="false"
  349. accept="image/*"
  350. :disabled="dialog.mode === 'view'"
  351. >
  352. <template #trigger>
  353. <el-button type="primary" :disabled="dialog.mode === 'view'">点击选择背景</el-button>
  354. </template>
  355. <!-- 预览图区域 -->
  356. <template #default>
  357. <div class="preview-area" @click="handlePreviewClick2">
  358. <!-- 只有当 iconPreviewUrl 或 competitionIcon 存在时才显示 img -->
  359. <img
  360. v-if="iconPreviewUrl2 || competitionBg"
  361. :src="iconPreviewUrl2 || competitionBg"
  362. alt="预览图"
  363. style="max-width: 100px; max-height: 100px; margin-top: 10px; cursor: pointer"
  364. />
  365. <!-- 可选:无图时显示提示文字 -->
  366. <div v-else style="margin-top: 10px; color: #999"></div>
  367. </div>
  368. </template>
  369. <template #tip>
  370. <div class="el-upload__tip">
  371. <span v-if="fileList2.length > 0">当前已选文件:{{ fileList2[0].name }}</span>
  372. </div>
  373. </template>
  374. </el-upload>
  375. </div>
  376. </el-form-item>
  377. <el-form-item label="比赛标签" prop="tagId">
  378. <div style="display: flex; align-items: center">
  379. <el-select
  380. v-model="form.tagId"
  381. placeholder="选项"
  382. style="width: 200px"
  383. :disabled="dialog.mode === 'view'"
  384. multiple
  385. collapse-tags
  386. collapse-tags-tooltip
  387. :multiple-limit="1"
  388. >
  389. <el-option v-for="item in itemOptionsTagList" :key="item.id" :label="item.label" :value="item.id" />
  390. </el-select>
  391. <el-button type="primary" :disabled="dialog.mode === 'view'" @click="handleGoToTag"> 标签管理 </el-button>
  392. </div>
  393. </el-form-item>
  394. <el-form-item label="比赛类目" prop="categoryId">
  395. <div style="display: flex; align-items: center">
  396. <el-select
  397. v-model="form.categoryId"
  398. placeholder="选项"
  399. style="width: 200px"
  400. :disabled="dialog.mode === 'view'"
  401. multiple
  402. :multiple-limit="1"
  403. collapse-tags
  404. collapse-tags-tooltip
  405. >
  406. <el-option v-for="item in itemOptionsCategoryList" :key="item.id" :label="item.label" :value="item.id" />
  407. </el-select>
  408. <el-button type="primary" :disabled="dialog.mode === 'view'" @click="handleGoToCategory"> 类目管理 </el-button>
  409. </div>
  410. </el-form-item>
  411. <!-- 开始时间 -->
  412. <el-form-item label="开始时间" prop="startTime">
  413. <el-date-picker
  414. clearable
  415. v-model="form.startTime"
  416. type="datetime"
  417. value-format="YYYY-MM-DD HH:mm"
  418. placeholder="请选择开始时间"
  419. :disabled="dialog.mode === 'view'"
  420. />
  421. </el-form-item>
  422. <!-- <el-form-item label="开赛最低人数" prop="minPlayers">
  423. <el-input v-model="form.minPlayers" placeholder="请输入开赛最低人数" :disabled="dialog.mode === 'view'" />
  424. </el-form-item>-->
  425. <!-- 盲注表 -->
  426. <el-form-item label="盲注等级" prop="blindStructureId">
  427. <div style="display: flex; align-items: center">
  428. <el-select
  429. v-model="form.blindStructureId"
  430. placeholder="选项"
  431. style="width: 200px"
  432. @change="handleBlindStructureChange"
  433. :disabled="dialog.mode === 'view'"
  434. >
  435. <el-option v-for="item in itemOptionsStructures" :key="item.id" :label="item.label" :value="item.id" />
  436. </el-select>
  437. <el-button type="primary" :disabled="dialog.mode === 'view'" @click="handleGoToStructures"> 上传新盲注 </el-button>
  438. <el-button type="primary" @click="handleViewLevels" :disabled="dialog.mode === 'view'">预览</el-button>
  439. </div>
  440. </el-form-item>
  441. <el-form-item label="目标锦标赛" prop="targetTournamentId">
  442. <el-select
  443. v-model="form.targetTournamentId"
  444. placeholder="请选择比赛"
  445. :disabled="dialog.mode === 'view'"
  446. @visible-change="handleSelectVisibleChange"
  447. remote
  448. :remote-method="remoteMethod"
  449. :loading="selectLoading"
  450. clearable
  451. >
  452. <el-option v-for="item in selectOptions" :key="item.id" :label="item.name" :value="item.id" />
  453. </el-select>
  454. </el-form-item>
  455. <el-form-item label="晋级条件类型" prop="qualifierType">
  456. <el-select
  457. aria-required="true"
  458. v-model="form.qualifierType"
  459. placeholder="请选择"
  460. :disabled="dialog.mode === 'view'"
  461. @change="handleQualifierTypeChange"
  462. >
  463. <el-option v-for="dict in qualifier_type" :key="dict.value" :label="dict.label" :value="dict.value"> </el-option>
  464. </el-select>
  465. </el-form-item>
  466. <el-form-item label="晋级条件值" prop="qualifierValue" v-if="form.qualifierType === '2'">
  467. <el-input v-model="form.qualifierValue" placeholder="请输入晋级条件值" :disabled="dialog.mode === 'view'" />
  468. </el-form-item>
  469. <el-form-item label="报名时长" prop="signTime">
  470. <el-select v-model="form.signTime" placeholder="请选择" :disabled="dialog.mode === 'view'">
  471. <el-option v-for="dict in tournaments_time" :key="dict.value" :label="dict.label" :value="dict.value"></el-option>
  472. </el-select>
  473. </el-form-item>
  474. <!-- 报名条件 -->
  475. <!-- <el-form-item label="报名条件" prop="itemsId">
  476. <div style="display: flex; align-items: center; gap: 10px">
  477. <el-select v-model="form.itemsId" placeholder="请选择道具类型" :disabled="dialog.mode === 'view'">
  478. <el-option v-for="item in itemOptions" :key="item.id" :label="item.label" :value="item.id" />
  479. </el-select>
  480. <el-input v-model="form.itemsNum" prop="itemsNum" :min="1" placeholder="数量" :disabled="dialog.mode === 'view'" />
  481. </div>
  482. </el-form-item>-->
  483. <!-- <el-form-item label="报名条件">
  484. <div style="display: flex; align-items: center; gap: 10px">
  485. <el-form-item prop="itemsId" style="margin-bottom: 0">
  486. <el-select v-model="form.itemsId" placeholder="请选择道具类型" :disabled="dialog.mode === 'view'">
  487. <el-option v-for="item in itemOptions" :key="item.id" :label="item.label" :value="item.id" />
  488. </el-select>
  489. </el-form-item>
  490. <el-form-item prop="itemsNum" style="margin-bottom: 0">
  491. <el-input v-model.number="form.itemsNum" placeholder="数量" :disabled="dialog.mode === 'view'" />
  492. </el-form-item>
  493. </div>
  494. </el-form-item>-->
  495. <el-form-item label="晋级级别" prop="qualifierValue" v-if="form.qualifierType === '1'">
  496. <el-select v-model="form.qualifierValue" placeholder="选项" style="width: 200px" :disabled="dialog.mode === 'view'">
  497. <el-option v-for="item in itemOptionsStructuresLevel2" :key="item.id" :label="item.label" :value="item.id" />
  498. </el-select>
  499. </el-form-item>
  500. <el-form-item label="起始级别" prop="startBlindLevel">
  501. <el-select v-model="form.startBlindLevel" placeholder="选项" style="width: 200px" :disabled="dialog.mode === 'view'">
  502. <el-option v-for="item in itemOptionsStructuresLevel3" :key="item.id" :label="item.label" :value="item.id" />
  503. </el-select>
  504. </el-form-item>
  505. <el-form-item label="级别持续时间" prop="levelDuration" v-if="dialog.mode === 'edit'">
  506. <el-input v-model="form.levelDuration" placeholder="请输入级别持续时间" />
  507. </el-form-item>
  508. <!-- <el-form-item label="机器人数" prop="robotCount" v-if="!isProdEnvironment">
  509. <el-input v-model="form.robotCount" placeholder="请输入机器人数" :disabled="dialog.mode === 'view'" />
  510. </el-form-item>-->
  511. <el-form-item label="延迟看牌" prop="delayShow">
  512. <el-select v-model="form.delayShow" placeholder="请选择" :disabled="dialog.mode === 'view'">
  513. <el-option label="开启" :value="1"></el-option>
  514. <el-option label="关闭" :value="0"></el-option>
  515. </el-select>
  516. </el-form-item>
  517. <el-form-item label="延迟卡" prop="isDelay">
  518. <el-select v-model="form.isDelay" placeholder="请选择" :disabled="dialog.mode === 'view'" @change="handleIsDelayChange">
  519. <el-option label="是" :value="1"></el-option>
  520. <el-option label="否" :value="0"></el-option>
  521. </el-select>
  522. </el-form-item>
  523. <!-- <el-form-item label="延迟卡时间" prop="delayCardTime" v-if="form.isDelay === 1">
  524. <el-input v-model="form.delayCardTime" placeholder="请输入延迟卡时间" :disabled="dialog.mode === 'view'" />
  525. </el-form-item>-->
  526. <!-- <el-form-item label="延迟卡数量" prop="delayCardNum" v-if="form.isDelay === 1">
  527. <el-input v-model="form.delayCardNum" placeholder="请输入延迟卡数量" :disabled="dialog.mode === 'view'" />
  528. </el-form-item>-->
  529. <el-form-item label="重复买入" prop="rebuy">
  530. <div style="display: flex; align-items: center; gap: 10px; width: 100%">
  531. <el-select v-model="form.rebuy" placeholder="请选择" :disabled="dialog.mode === 'view'" style="flex: 1">
  532. <el-option label="不可重复" :value="-1"></el-option>
  533. <el-option label="不限制" :value="0"></el-option>
  534. <el-option label="限制次数" :value="1"></el-option>
  535. </el-select>
  536. <el-form-item prop="rebuyLimit" style="margin-bottom: 0; flex: 1" v-if="form.rebuy === 1">
  537. <el-input v-model.number="form.rebuyLimit" placeholder="请输入次数" :disabled="dialog.mode === 'view'" style="width: 100%" />
  538. </el-form-item>
  539. </div>
  540. </el-form-item>
  541. <!-- 报名截止至 -->
  542. <el-form-item label="报名截止至" prop="lateRegistrationLevel">
  543. <el-select v-model="form.lateRegistrationLevel" placeholder="选项" :disabled="dialog.mode === 'view'">
  544. <el-option v-for="item in itemOptionsStructuresLevel" :key="item.id" :label="item.label" :value="item.id" />
  545. </el-select>
  546. </el-form-item>
  547. <el-form-item label="起始记分牌数量" prop="startingChips">
  548. <el-input v-model="form.startingChips" placeholder="请输入起始记分牌数量" :disabled="dialog.mode === 'view'" />
  549. </el-form-item>
  550. <el-form-item label="报名条件" prop="itemsId" v-if="form.gameType !== '3'">
  551. <div style="display: flex; align-items: center; gap: 10px; width: 100%">
  552. <el-select v-model="form.itemsId" placeholder="请选择道具类型" :disabled="dialog.mode === 'view'" style="flex: 1">
  553. <el-option v-for="item in itemOptions" :key="item.id" :label="item.label" :value="item.id" />
  554. </el-select>
  555. <el-form-item prop="itemsNum" style="margin-bottom: 0; flex: 1">
  556. <el-input v-model.number="form.itemsNum" placeholder="数量" :disabled="dialog.mode === 'view'" style="width: 100%" />
  557. </el-form-item>
  558. </div>
  559. </el-form-item>
  560. <!-- 奖励内容 -->
  561. <el-form-item label="比赛奖励">
  562. <div v-for="(reward, index) in formPrize.rewards" :key="index" style="display: flex; align-items: center; margin-bottom: 8px">
  563. <span>第{{ reward.ranking }}名</span>
  564. <!-- 道具选择 -->
  565. <el-select v-model="reward.itemId" placeholder="选项" style="width: 120px; margin-right: 8px" :disabled="dialog.mode === 'view'">
  566. <el-option v-for="item in itemOptions" :key="item.id" :label="item.label" :value="item.id" />
  567. </el-select>
  568. <!-- 数量输入 -->
  569. <el-input v-model="reward.quantity" placeholder="请输入数量" style="width: 100px; margin-right: 8px" :disabled="dialog.mode === 'view'" />
  570. <!-- 操作按钮 -->
  571. <el-button type="primary" @click="addReward" v-if="index === 0 && dialog.mode !== 'view'">+</el-button>
  572. <el-button type="primary" @click="removeReward(index)" v-if="index !== 0 && dialog.mode !== 'view'">-</el-button>
  573. </div>
  574. </el-form-item>
  575. <el-form-item label="赛事备注" prop="remark">
  576. <el-input v-model="form.remark" placeholder="请输入赛事备注" :disabled="dialog.mode === 'view'" type="textarea" :rows="4" />
  577. </el-form-item>
  578. </el-form>
  579. <template #footer>
  580. <div class="dialog-footer">
  581. <el-button :loading="buttonLoading" type="primary" @click="submitForm" v-if="dialog.mode !== 'view'">确 定</el-button>
  582. <el-button @click="cancel">取 消</el-button>
  583. </div>
  584. </template>
  585. </el-dialog>
  586. <!-- 分配模态框 -->
  587. <el-dialog title="分配盲注信息" v-model="assignDialog.visible" width="800px" append-to-body destroy-on-close>
  588. <el-form ref="assignFormRef" :model="assignForm" label-width="100px">
  589. <!-- 可勾选的列表 -->
  590. <el-form-item label="">
  591. <div style="display: flex; flex-direction: column; align-items: center; width: 100%">
  592. <div style="width: 100%; height: 400px; overflow-y: auto; padding: 0">
  593. <el-table border ref="assignTable" :data="assignList" :row-key="'blindStructuresId'" style="width: 100%">
  594. <el-table-column label="操作" align="center" width="55">
  595. <template #default="{ row }">
  596. <el-radio v-model="selectedRadio" :label="row.blindStructuresId">&nbsp;</el-radio>
  597. </template>
  598. </el-table-column>
  599. <!-- <el-table-column label="blindStructuresId" align="center" prop="blindStructuresId" width="100" show-overflow-tooltip />-->
  600. <el-table-column prop="blindStructuresName" label="盲注结构名称" align="center" width="150" show-overflow-tooltip />
  601. <el-table-column prop="blindDescription" label="盲注结构描述" align="center" width="200" show-overflow-tooltip />
  602. <el-table-column prop="allocationStatus" label="绑定情况" align="center" width="100">
  603. <template #default="scope">
  604. {{ scope.row.allocationStatus === 0 ? '未绑定' : '已绑定' }}
  605. </template>
  606. </el-table-column>
  607. <el-table-column prop="tournamentsName" label="赛事名称" align="center" width="150" show-overflow-tooltip />
  608. </el-table>
  609. </div>
  610. <!-- 分页(如果数据量大) -->
  611. <div style="display: flex; justify-content: center; margin-top: 20px; width: 100%">
  612. <pagination
  613. v-if="assignTotal > 0"
  614. :total="assignTotal"
  615. v-model:page="assignQuery.pageNum"
  616. v-model:limit="assignQuery.pageSize"
  617. @pagination="getAssignList"
  618. />
  619. </div>
  620. </div>
  621. </el-form-item>
  622. </el-form>
  623. <template #footer>
  624. <div class="dialog-footer" style="text-align: center">
  625. <el-button :loading="buttonLoading" type="primary" @click="submitAssign">保 存</el-button>
  626. <el-button @click="assignDialog.visible = false">取 消</el-button>
  627. </div>
  628. </template>
  629. </el-dialog>
  630. <el-dialog v-model="levelsDialogVisible" title="盲注等级列表" width="80%">
  631. <!-- 使用 component 动态加载目标组件 -->
  632. <levels-index :blind-structure-id="dialogParams.blindStructureId" :name="dialogParams.name" />
  633. </el-dialog>
  634. <!-- 预览弹窗 -->
  635. <el-dialog v-model="dialogVisible" title="图片预览" width="50%">
  636. <img :src="previewSrc || iconPreviewUrl || competitionIcon" alt="预览图片" style="max-width: 100%; max-height: 80vh" />
  637. </el-dialog>
  638. <el-dialog v-model="dialogVisible2" title="图片预览" width="50%">
  639. <img :src="previewSrc2 || iconPreviewUrl2 || competitionBg" alt="预览图片" style="max-width: 100%; max-height: 80vh" />
  640. </el-dialog>
  641. <el-dialog
  642. :title="`${tournamentInfo.name} ${tournamentInfo.tournamentsBiId}`"
  643. v-model="auditDialog.visible"
  644. width="1000px"
  645. append-to-body
  646. @close="cancelAudit"
  647. >
  648. <!-- 查看对局记录按钮 -->
  649. <!-- <div class="dialog-header-actions" v-if="auditDialog.mode === 'view'">
  650. <el-button type="primary">查看对局记录</el-button>
  651. </div>-->
  652. <!-- 比赛信息展示 -->
  653. <div class="tournament-info" v-if="tournamentInfo.id">
  654. <p>比赛时间:{{ tournamentInfo.startTime }} ~ {{ tournamentInfo.endTime }}</p>
  655. <p>参加人数:{{ tournamentInfo.signNum }}人</p>
  656. <p v-if="auditDialog.mode === 'view'">比赛类型:{{ tournamentInfo.id }}</p>
  657. <p v-if="auditDialog.mode === 'view'">报名条件:{{ tournamentInfo.itemsName }} x {{ tournamentInfo.itemsNum }}</p>
  658. <p v-if="auditDialog.mode === 'view'">
  659. 盲注等级:{{ tournamentInfo.blindStructuresName }}
  660. <el-button type="primary" @click="handleViewLevelsSee(tournamentInfo)">预览</el-button>
  661. </p>
  662. <p v-if="auditDialog.mode === 'view'">报名截止至:{{ tournamentInfo.lateRegistrationLevel }}</p>
  663. <p v-if="tournamentInfo.isComplaints" style="color: red">本场比赛仍有选手存在异议,请处理申述后继续授作。</p>
  664. </div>
  665. <!-- 添加导出按钮 -->
  666. <div style="margin: 10px 0; text-align: right">
  667. <el-button type="warning" plain icon="Download" @click="handleExportAuditData">导出审核数据</el-button>
  668. </div>
  669. <!-- 表格内容 -->
  670. <el-table :data="auditData" border style="width: 100%">
  671. <el-table-column prop="id" label="id" align="center" v-if="false"></el-table-column>
  672. <el-table-column prop="rank" label="排名" align="center"></el-table-column>
  673. <el-table-column prop="playerName" label="用户名" align="center"></el-table-column>
  674. <el-table-column prop="phone" label="手机" align="center"></el-table-column>
  675. <!-- <el-table-column prop="tournamentId" label="tournamentId" align="center"></el-table-column>
  676. <el-table-column prop="playerId" label="playerId" align="center"></el-table-column>-->
  677. <el-table-column label="奖励" align="center">
  678. <template #default="scope">
  679. <div v-for="(prize, index) in scope.row.rewardVoList" :key="index">{{ prize.itemName }} {{ prize.quantity }}</div>
  680. </template>
  681. </el-table-column>
  682. <el-table-column prop="claimedText" label="状态" align="center"></el-table-column>
  683. <el-table-column label="操作" align="center" width="350" class-name="small-padding fixed-width">
  684. <template #default="scope">
  685. <div style="display: flex; justify-content: center; gap: 16px">
  686. <el-tooltip content="查看牌局" placement="top">
  687. <el-button link type="primary" icon="View" @click="handleViewHistory(scope.row)">查看牌局</el-button>
  688. </el-tooltip>
  689. <el-tooltip content="审核" placement="top">
  690. <el-button link type="primary" icon="DocumentChecked" @click="handleAudit(scope.row)">审核</el-button>
  691. </el-tooltip>
  692. <el-tooltip content="移除" placement="top">
  693. <el-button link type="primary" icon="Delete" @click="handleClaimsDelete(scope.row)">移除</el-button>
  694. </el-tooltip>
  695. </div>
  696. </template>
  697. </el-table-column>
  698. </el-table>
  699. <!-- 分页组件 -->
  700. <div class="pagination-container" style="margin-top: 20px; text-align: center">
  701. <el-pagination
  702. v-model:current-page="auditQueryParams.pageNum"
  703. v-model:page-size="auditQueryParams.pageSize"
  704. :total="auditTotal"
  705. layout="prev, pager, next"
  706. @current-change="getAuditData"
  707. />
  708. </div>
  709. <!-- 对话框底部按钮 -->
  710. <template #footer>
  711. <div class="dialog-footer">
  712. <el-button @click="cancelAudit">取 消</el-button>
  713. </div>
  714. </template>
  715. </el-dialog>
  716. <el-dialog title="移除获奖名额" v-model="removeDialog.visible" width="30%">
  717. <span>是否移除用户:{{ removeDialog.playerName }}{{ removeDialog.phone }} 在本次比赛中的获奖名额?</span>
  718. <template #footer>
  719. <span class="dialog-footer">
  720. <el-button @click="cancelRemove">取消</el-button>
  721. <el-button type="danger" @click="confirmRemove">确认移除</el-button>
  722. </span>
  723. </template>
  724. </el-dialog>
  725. </div>
  726. </template>
  727. <script setup name="Tournaments" lang="ts">
  728. import {
  729. listTournaments,
  730. getTournaments,
  731. delTournaments,
  732. deleteCheckTournament,
  733. addTournaments,
  734. updateTournaments,
  735. getSelectTournamentBlindStructuresList,
  736. assignTournamentBlindStructures,
  737. uploadTournament,
  738. closeSendTournament,
  739. recoverTournament,
  740. deleteRelationTournament,
  741. publishToTournament
  742. } from '@/api/system/business/tournaments';
  743. import { selectTagSelList, selectCategorySelList } from '@/api/system/business/tag';
  744. import { selectItemsSelList } from '@/api/system/business/items';
  745. import { selectBlindLevelsById } from '@/api/system/business/levels';
  746. import { selectBlingStructuresInfo } from '@/api/system/business/structures';
  747. import { listClaims, deleteClaim, auditSendReward } from '@/api/system/business/claims';
  748. import { selectTournamentConfigSelList } from '@/api/system/business/tournamentConfig';
  749. import { TournamentsVO, TournamentsQuery, TournamentsForm, TournamentsBindStructuresVO } from '@/api/system/business/tournaments/types';
  750. import { ref } from 'vue';
  751. import LevelsIndex from '@/views/system/business/levels/index.vue';
  752. import { ClaimsVO } from '@/api/system/business/claims/types';
  753. const { proxy } = getCurrentInstance() as ComponentInternalInstance;
  754. const { tournaments_type, tournaments_time, tournaments_status, game_variant_type, qualifier_type } = toRefs<any>(
  755. proxy?.useDict('tournaments_type', 'tournaments_time', 'tournaments_status', 'game_variant_type', 'qualifier_type')
  756. );
  757. const tournamentsList = ref<TournamentsVO[]>([]);
  758. const buttonLoading = ref(false);
  759. const loading = ref(true);
  760. const showSearch = ref(true);
  761. const ids = ref<Array<string | number>>([]);
  762. const single = ref(true);
  763. const multiple = ref(true);
  764. const total = ref(0);
  765. const queryFormRef = ref<ElFormInstance>();
  766. const tournamentsFormRef = ref<ElFormInstance>();
  767. const iconFile = ref<File | null>(null); // 存储选中的图标文件
  768. const dialog = reactive<{
  769. visible: boolean;
  770. title: string;
  771. mode: 'edit' | 'view' | 'add'; // 新增字段
  772. }>({
  773. visible: false,
  774. title: '',
  775. mode: 'edit'
  776. });
  777. const assignDialog = reactive({
  778. visible: false,
  779. title: '分配盲注'
  780. });
  781. const assignForm = reactive({
  782. tournamentId: undefined as number | undefined,
  783. selectedIds: [] as number[]
  784. });
  785. const dialogVisible = ref(false);
  786. const previewSrc = ref('');
  787. const dialogVisible2 = ref(false);
  788. const previewSrc2 = ref('');
  789. function openPreview(src: string) {
  790. previewSrc.value = src;
  791. dialogVisible.value = true;
  792. }
  793. function openPreview2(src: string) {
  794. previewSrc2.value = src;
  795. dialogVisible2.value = true;
  796. }
  797. // 控制 Dialog 是否显示
  798. const levelsDialogVisible = ref(false);
  799. // 传递给子组件的参数
  800. const dialogParams = ref({
  801. blindStructureId: null,
  802. name: null
  803. });
  804. // 下拉选项数据 selectBlingStructuresInfo
  805. const itemOptions = ref<{ id: number; label: string }[]>([]);
  806. // 加载报名条件选项
  807. const loadItemOptions = async () => {
  808. try {
  809. const res = await selectItemsSelList();
  810. if (res.code === 200) {
  811. // 类型断言
  812. const data = res.data as unknown as { id: number; name: string }[];
  813. // 过滤掉 id === 2 的项,并映射为 { id, label }
  814. itemOptions.value = data
  815. .filter((item) => item.id !== 2) // ✅ 过滤 id 为 2 的
  816. .map((item) => ({
  817. id: item.id,
  818. label: item.name
  819. }));
  820. } else {
  821. ElMessage.error('加载失败:' + res.msg);
  822. }
  823. } catch (error) {
  824. console.error('请求出错:', error);
  825. ElMessage.error('请求失败,请检查网络');
  826. }
  827. };
  828. // 下拉选项数据 selectBlingStructuresInfo
  829. const itemOptionsStructures = ref<{ id: number; label: string }[]>([]);
  830. // 加载报名条件选项
  831. const loadItemStructuresOptions = async () => {
  832. try {
  833. const res = await selectBlingStructuresInfo();
  834. if (res.code === 200) {
  835. // 使用 unknown 中间类型进行类型转换
  836. const data = res.data as unknown as { id: number; name: string }[];
  837. const list = [];
  838. for (let i = 0; i < data.length; i++) {
  839. const item = data[i];
  840. list.push({
  841. id: item.id,
  842. label: item.name
  843. });
  844. }
  845. itemOptionsStructures.value = list;
  846. // Automatically select the first option if form is in add mode and no value is set
  847. if (!form.value.blindStructureId && list.length > 0) {
  848. form.value.blindStructureId = list[0].id;
  849. // Trigger the change event to load corresponding levels
  850. handleBlindStructureChange(list[0].id);
  851. }
  852. } else {
  853. alert('加载失败:' + res.msg);
  854. }
  855. } catch (error) {
  856. console.error('请求出错:', error);
  857. }
  858. };
  859. // 下拉选项数据
  860. const itemOptionsStructuresLevel = ref<{ id: number; label: string }[]>([]);
  861. const itemOptionsStructuresLevel2 = ref<{ id: number; label: string }[]>([]);
  862. const itemOptionsStructuresLevel3 = ref<{ id: number; label: string }[]>([]);
  863. // 加载报名条件选项
  864. const handleBlindStructureChange = async (value: number) => {
  865. //data.form.lateRegistrationLevel = null;
  866. try {
  867. const res = await selectBlindLevelsById(value);
  868. if (res.code === 200) {
  869. // 使用 unknown 中间类型进行类型转换
  870. const data2 = res.data as unknown as { id: number; levelNumber: number }[];
  871. const list = [];
  872. for (let i = 0; i < data2.length; i++) {
  873. const item = data2[i];
  874. list.push({
  875. id: item.levelNumber,
  876. label: item.levelNumber
  877. });
  878. }
  879. itemOptionsStructuresLevel.value = list;
  880. itemOptionsStructuresLevel2.value = list;
  881. itemOptionsStructuresLevel3.value = list;
  882. // 判断当前选择的 lateRegistrationLevel 是否在新列表中
  883. const currentLevel = data.form.lateRegistrationLevel;
  884. if (currentLevel && !list.some((item) => item.id === currentLevel)) {
  885. data.form.lateRegistrationLevel = null;
  886. data.form.lateRegistrationLevel = null;
  887. }
  888. } else {
  889. alert('加载失败:' + res.msg);
  890. }
  891. } catch (error) {
  892. console.error('请求出错:', error);
  893. }
  894. };
  895. const formPrize = reactive({
  896. // ...其他表单字段
  897. rewards: [
  898. {
  899. ranking: 1,
  900. itemId: null,
  901. quantity: null
  902. }
  903. ]
  904. });
  905. // 用于保存原始的默认奖励结构
  906. const defaultRewardStructure = {
  907. ranking: 1,
  908. itemId: null,
  909. quantity: null
  910. };
  911. const addReward = () => {
  912. const currentLength = formPrize.rewards.length;
  913. // 判断是否超过 itemOptions 的数量限制
  914. /* if (currentLength < itemOptions.value.length) {*/
  915. formPrize.rewards.push({
  916. ranking: currentLength + 1, // 自动生成排名,从 1 开始
  917. itemId: null,
  918. quantity: null
  919. });
  920. /* } else {
  921. ElMessage.warning(`最多只能添加 ${itemOptions.value.length} 个奖励项`);
  922. }*/
  923. };
  924. const removeReward = (index: number) => {
  925. formPrize.rewards.splice(index, 1);
  926. // 删除后重新设置排名
  927. formPrize.rewards.forEach((r, i) => {
  928. r.ranking = i + 1;
  929. });
  930. };
  931. //预览图标需要
  932. const iconPreviewUrl = ref('');
  933. const competitionIcon = ref('');
  934. const fileList = ref([]);
  935. //预览图标需要
  936. const iconPreviewUrl2 = ref('');
  937. const competitionBg = ref('');
  938. const fileList2 = ref([]);
  939. import { parseTime } from '@/utils/dateUtils';
  940. import { ElSelect } from 'element-plus';
  941. // 单选控制变量(用于绑定 el-radio)
  942. const selectedRadio = ref<number | null>(null);
  943. const assignList = ref<TournamentsBindStructuresVO[]>([]);
  944. const assignSelectedList = ref<TournamentsBindStructuresVO[]>([]);
  945. const assignTotal = ref(0);
  946. const assignQuery = reactive<{
  947. pageNum: number;
  948. pageSize: number;
  949. tournamentId: number | undefined;
  950. }>({
  951. pageNum: 1,
  952. pageSize: 10,
  953. tournamentId: undefined
  954. });
  955. const initFormData: TournamentsForm = {
  956. id: undefined,
  957. name: undefined,
  958. startTime: undefined,
  959. gameType: undefined,
  960. gameVariant: null,
  961. qualifierType: '0',
  962. startingChips: undefined,
  963. levelDuration: undefined,
  964. lateRegistrationLevel: undefined,
  965. maxPlayers: undefined,
  966. status: undefined,
  967. createdAt: undefined,
  968. updatedAt: undefined,
  969. signTime: undefined,
  970. competitionIcon: null,
  971. itemsId: null,
  972. itemsNum: null,
  973. blindStructureId: null,
  974. itemsPrizeList: [],
  975. delayCardNum: 4,
  976. delayCardTime: 15,
  977. isSponsor: 1, // 添加赞助商标识,默认为否
  978. isDelay: 1,
  979. rebuyLimit: null,
  980. tagId: [], // 修改为数组形式以支持多选
  981. categoryId: []
  982. };
  983. const data = reactive<PageData<TournamentsForm, TournamentsQuery>>({
  984. form: { ...initFormData },
  985. queryParams: {
  986. pageNum: 1,
  987. pageSize: 10,
  988. name: undefined,
  989. startTime: undefined,
  990. gameType: undefined,
  991. startingChips: undefined,
  992. levelDuration: undefined,
  993. lateRegistrationLevel: undefined,
  994. maxPlayers: undefined,
  995. status: undefined,
  996. createdAt: undefined,
  997. updatedAt: undefined,
  998. id: undefined,
  999. params: {}
  1000. },
  1001. rules: {
  1002. id: [{ required: true, message: '不能为空', trigger: 'blur' }],
  1003. name: [{ required: true, message: '赛事名称不能为空', trigger: 'blur' }],
  1004. startTime: [{ required: true, message: '比赛开始时间不能为空', trigger: 'blur' }],
  1005. configId: [{ required: true, message: '不能为空', trigger: 'blur' }],
  1006. gameType: [{ required: true, message: '游戏类型不能为空', trigger: 'change' }],
  1007. gameVariant: [{ required: true, message: '玩法类型不能为空', trigger: 'change' }],
  1008. qualifierType: [{ required: true, message: '晋级条件类型不能为空', trigger: 'change' }],
  1009. lateRegistrationLevel: [{ required: true, message: '截止报名级别不能为空', trigger: 'change' }],
  1010. signTime: [{ required: true, message: '报名时间不能为空', trigger: 'change' }],
  1011. itemsId: [
  1012. {
  1013. required: true,
  1014. message: '报名条件不能为空',
  1015. trigger: 'change',
  1016. validator: (rule: any, value: any, callback: any) => {
  1017. // 如果 gameType 为 3,则不进行校验
  1018. if (data.form.gameType === '3') {
  1019. callback();
  1020. return;
  1021. }
  1022. // 其他情况执行默认校验
  1023. if (!value) {
  1024. callback(new Error('报名条件不能为空'));
  1025. } else {
  1026. callback();
  1027. }
  1028. }
  1029. }
  1030. ],
  1031. blindStructureId: [{ required: true, message: '盲注等级不能为空', trigger: 'change' }],
  1032. startingChips: [
  1033. { required: true, message: '起始记分牌数量不能为空', trigger: 'blur' },
  1034. {
  1035. validator: (rule, value, callback) => {
  1036. // 检查是否为字符串类型且包含非数字字符
  1037. if (typeof value === 'string' && !/^\d+$/.test(value)) {
  1038. callback(new Error('起始记分牌数量必须为数字'));
  1039. return;
  1040. }
  1041. const num = Number(value);
  1042. if (isNaN(num) || num <= 0) {
  1043. callback(new Error('起始记分牌数量必须大于0'));
  1044. } else if (!Number.isInteger(num)) {
  1045. callback(new Error('起始记分牌数量必须为整数'));
  1046. } else {
  1047. callback();
  1048. }
  1049. },
  1050. trigger: 'blur'
  1051. }
  1052. ],
  1053. itemsNum: [
  1054. {
  1055. required: true,
  1056. message: '数量不能为空',
  1057. trigger: 'blur',
  1058. validator: (rule: any, value: any, callback: any) => {
  1059. // 如果 gameType 为 3,则不进行校验
  1060. if (data.form.gameType === '3') {
  1061. callback();
  1062. return;
  1063. }
  1064. // 检查是否为字符串类型且包含非数字字符
  1065. if (typeof value === 'string' && !/^\d+$/.test(value)) {
  1066. callback(new Error('数量必须为正整数'));
  1067. return;
  1068. }
  1069. const num = Number(value);
  1070. if (isNaN(num) || num <= 0) {
  1071. callback(new Error('数量必须大于0'));
  1072. } else if (!Number.isInteger(num)) {
  1073. callback(new Error('数量必须为整数'));
  1074. } else {
  1075. callback();
  1076. }
  1077. }
  1078. }
  1079. ],
  1080. delayCardTime: [
  1081. { required: false, message: '延迟卡时间不能为空', trigger: 'blur' },
  1082. {
  1083. validator: (rule: any, value: any, callback: any) => {
  1084. const num = Number(value);
  1085. if (!value) {
  1086. callback(new Error('请输入延迟卡时间'));
  1087. } else if (!/^\d+$/.test(value)) {
  1088. callback(new Error('只能输入正整数'));
  1089. } else if (num < 10 || num > 15) {
  1090. callback(new Error('延迟卡时间必须在10-15秒之间'));
  1091. } else {
  1092. callback();
  1093. }
  1094. },
  1095. trigger: 'blur'
  1096. }
  1097. ],
  1098. delayCardNum: [
  1099. { required: false, message: '延迟卡数量不能为空', trigger: 'blur' },
  1100. {
  1101. validator: (rule: any, value: any, callback: any) => {
  1102. const num = Number(value);
  1103. if (!value) {
  1104. callback(new Error('请输入延迟卡数量'));
  1105. } else if (!/^\d+$/.test(value)) {
  1106. callback(new Error('只能输入正整数'));
  1107. } else if (num < 1 || num > 99) {
  1108. callback(new Error('延迟卡数量必须在1-99之间'));
  1109. } else {
  1110. callback();
  1111. }
  1112. },
  1113. trigger: 'blur'
  1114. }
  1115. ],
  1116. minPlayers: [
  1117. { required: false, message: '最小参赛人数不能为空', trigger: 'blur' },
  1118. {
  1119. validator: (rule: any, value: any, callback: any) => {
  1120. const num = Number(value);
  1121. if (!/^\d+$/.test(value)) {
  1122. callback(new Error('只能输入正整数'));
  1123. } else if (num < 0 || num > 999) {
  1124. callback(new Error('最小参赛人数必须在0-999之间'));
  1125. } else {
  1126. callback();
  1127. }
  1128. },
  1129. trigger: 'blur'
  1130. }
  1131. ]
  1132. }
  1133. });
  1134. const { queryParams, form, rules } = toRefs(data);
  1135. /** 查询【请填写功能名称】列表 */
  1136. const getList = async () => {
  1137. loading.value = true;
  1138. const params = {
  1139. ...queryParams.value,
  1140. sortBy: sortData.value.prop,
  1141. isAsc: sortData.value.order === 'ascending'
  1142. };
  1143. const res = await listTournaments(params);
  1144. tournamentsList.value = res.rows;
  1145. total.value = res.total;
  1146. loading.value = false;
  1147. };
  1148. /** 取消按钮 */
  1149. const cancel = () => {
  1150. reset();
  1151. resetFormPrize();
  1152. fileList.value = [];
  1153. // 清除预览图和临时链接
  1154. iconPreviewUrl.value = '';
  1155. competitionIcon.value = ''; // 如果需要清除后台加载的图标,可以在这里设置为空字符串
  1156. iconPreviewUrl2.value = '';
  1157. competitionBg.value = '';
  1158. dialog.visible = false;
  1159. };
  1160. /** 表单重置 */
  1161. const reset = () => {
  1162. form.value = { ...initFormData };
  1163. tournamentsFormRef.value?.resetFields();
  1164. };
  1165. /** 搜索按钮操作 */
  1166. const handleQuery = () => {
  1167. queryParams.value.pageNum = 1;
  1168. getList();
  1169. };
  1170. /** 重置按钮操作 */
  1171. const resetQuery = () => {
  1172. // 将所有查询参数重置为空
  1173. queryParams.value.id = '';
  1174. queryParams.value.name = '';
  1175. queryParams.value.status = '';
  1176. queryParams.value.startTime = '';
  1177. // 重置页码
  1178. queryParams.value.pageNum = 1;
  1179. // 直接执行搜索
  1180. getList();
  1181. };
  1182. /** 多选框选中数据 */
  1183. const handleSelectionChange = (selection: TournamentsVO[]) => {
  1184. ids.value = selection.map((item) => item.id);
  1185. single.value = selection.length != 1;
  1186. multiple.value = !selection.length;
  1187. };
  1188. /** 新增按钮操作 */
  1189. const handleAdd = () => {
  1190. reset();
  1191. resetFormPrize();
  1192. fileList.value = [];
  1193. // 清除预览图和临时链接
  1194. iconPreviewUrl.value = '';
  1195. competitionIcon.value = ''; // 如果需要清除后台加载的图标,可以在这里设置为空字符串
  1196. iconPreviewUrl2.value = '';
  1197. competitionBg.value = '';
  1198. dialog.visible = true;
  1199. dialog.title = '创建比赛';
  1200. dialog.mode = 'add'; // 设置模式
  1201. // Load structures and automatically select first option
  1202. nextTick(() => {
  1203. loadItemStructuresOptions();
  1204. });
  1205. };
  1206. // 重置函数
  1207. const resetFormPrize = () => {
  1208. // 清空 rewards 并添加默认项
  1209. formPrize.rewards.splice(0); // 清空数组
  1210. formPrize.rewards.push({ ...defaultRewardStructure }); // 添加初始项
  1211. };
  1212. /** 修改按钮操作 */
  1213. const handleUpdate = async (row?: TournamentsVO, mode: 'edit' | 'view' = 'edit') => {
  1214. reset(); // 重置表单
  1215. const _id = row?.id || ids.value[0];
  1216. const res = await getTournaments(_id);
  1217. // 设置表单数据
  1218. Object.assign(form.value, res.data);
  1219. form.value.signTime = String(res.data.signTime);
  1220. form.value.gameVariant = String(res.data.gameVariant);
  1221. form.value.qualifierType = String(res.data.qualifierType);
  1222. form.value.gameType = String(res.data.gameType); // 转为字符串
  1223. competitionIcon.value = res.data.competitionIcon;
  1224. competitionBg.value = res.data.competitionBg;
  1225. // 清空文件列表,避免残留
  1226. fileList.value = [];
  1227. fileList2.value = [];
  1228. // 如果有图片URL,创建虚拟文件对象以便显示
  1229. if (res.data.competitionIcon) {
  1230. fileList.value = [
  1231. {
  1232. name: '已上传图片',
  1233. url: res.data.competitionIcon,
  1234. status: 'success'
  1235. }
  1236. ];
  1237. }
  1238. if (res.data.competitionBg) {
  1239. fileList2.value = [
  1240. {
  1241. name: '已上传背景图',
  1242. url: res.data.competitionBg,
  1243. status: 'success'
  1244. }
  1245. ];
  1246. }
  1247. if (res.data.delayCardTime == 0 && res.data.delayCardNum == 0) {
  1248. form.value.isDelay = 0;
  1249. }
  1250. if (res.data.competitionBg == null) {
  1251. form.value.isSponsor = 0;
  1252. }
  1253. // 处理rebuy和rebuyLimit的显示逻辑
  1254. if (res.data.rebuy > 0) {
  1255. // 如果rebuy值>=0,说明是限制次数的情况
  1256. form.value.rebuy = 1; // 设置为"限制次数"选项
  1257. form.value.rebuyLimit = res.data.rebuy; // 将原值保存到rebuyLimit
  1258. } else {
  1259. // 否则保持原来的值(-1表示不可重复,0表示不限制)
  1260. form.value.rebuy = res.data.rebuy;
  1261. form.value.rebuyLimit = null;
  1262. }
  1263. // 确保tagId是数组格式
  1264. if (!Array.isArray(form.value.tagId)) {
  1265. form.value.tagId = form.value.tagId ? [form.value.tagId] : [];
  1266. }
  1267. // 确保categoryId是数组格式
  1268. if (!Array.isArray(form.value.categoryId)) {
  1269. form.value.categoryId = form.value.categoryId ? [form.value.categoryId] : [];
  1270. }
  1271. // 处理奖励表单数据
  1272. const prizeItems = res.data.itemsPrizeList || [];
  1273. if (prizeItems.length > 0) {
  1274. formPrize.rewards = prizeItems.map((item, index) => ({
  1275. ranking: item.ranking || index + 1,
  1276. itemId: Number(item.itemId),
  1277. quantity: Number(item.quantity)
  1278. }));
  1279. } else {
  1280. formPrize.rewards = [
  1281. {
  1282. ranking: 1,
  1283. itemId: null,
  1284. quantity: null
  1285. }
  1286. ];
  1287. }
  1288. // ✅ 主动触发盲注等级加载
  1289. if (form.value.blindStructureId) {
  1290. await handleBlindStructureChange(form.value.blindStructureId);
  1291. }
  1292. dialog.visible = true;
  1293. dialog.title = mode === 'view' ? '查看比赛' : '编辑比赛';
  1294. dialog.mode = mode; // 设置模式
  1295. nextTick(() => {
  1296. loadItemStructuresOptions();
  1297. });
  1298. };
  1299. /** 提交按钮 */
  1300. const submitForm = () => {
  1301. tournamentsFormRef.value?.validate(async (valid: boolean) => {
  1302. // 校验奖励内容是否至少填写了一项
  1303. if (!formPrize.rewards.some((r) => r.itemId && r.quantity)) {
  1304. ElMessage.warning('请至少填写一项奖励内容');
  1305. return;
  1306. }
  1307. if (!valid) return;
  1308. try {
  1309. buttonLoading.value = true;
  1310. // 当rebuy为1(限制次数)时,将rebuyLimit的值赋给rebuy
  1311. let submitRebuyValue = data.form.rebuy;
  1312. if (data.form.rebuy === 1) {
  1313. submitRebuyValue = data.form.rebuyLimit;
  1314. }
  1315. // 确保tagId始终是数组格式
  1316. let submitTagId = data.form.tagId;
  1317. if (!Array.isArray(submitTagId)) {
  1318. submitTagId = submitTagId ? [submitTagId] : [];
  1319. }
  1320. // 确保categoryId是数组格式
  1321. if (!Array.isArray(form.value.categoryId)) {
  1322. form.value.categoryId = form.value.categoryId ? [form.value.categoryId] : [];
  1323. }
  1324. // 构造最终要提交的数据对象
  1325. const formData: TournamentsForm = {
  1326. ...data.form,
  1327. tagId: submitTagId, // 使用处理后的tagId值
  1328. rebuy: submitRebuyValue, // 使用处理后的rebuy值
  1329. competitionIcon: data.form.competitionIcon, // 确保 iconUrl 存在
  1330. itemsPrizeList: formPrize.rewards.map((reward) => ({
  1331. ranking: Number(reward.ranking),
  1332. itemId: Number(reward.itemId),
  1333. quantity: Number(reward.quantity)
  1334. }))
  1335. };
  1336. // 提交数据(区分新增/编辑)
  1337. let response;
  1338. if (formData.id) {
  1339. response = await updateTournaments(formData);
  1340. } else {
  1341. response = await addTournaments(formData);
  1342. }
  1343. proxy?.$modal.msgSuccess('操作成功');
  1344. dialog.visible = false;
  1345. await getList();
  1346. } catch (error) {
  1347. console.error('提交失败:', error);
  1348. proxy?.$modal.msgError('提交失败,请重试');
  1349. } finally {
  1350. buttonLoading.value = false;
  1351. }
  1352. });
  1353. };
  1354. /** 删除按钮操作 */
  1355. const handleDelete = async (row?: TournamentsVO) => {
  1356. const _ids = row?.id || ids.value;
  1357. await proxy?.$modal.confirm('是否确认删除比赛ID为"' + _ids + '"的数据项?').finally(() => (loading.value = false));
  1358. await deleteCheckTournament(_ids);
  1359. proxy?.$modal.msgSuccess('删除成功');
  1360. await getList();
  1361. };
  1362. /** 恢复比赛按钮操作 */
  1363. const handleRecoverTournament = async (row?: TournamentsVO) => {
  1364. const _ids = row?.id || ids.value;
  1365. await proxy?.$modal.confirm('是否确认恢复比赛ID为"' + _ids + '"的数据项?').finally(() => (loading.value = false));
  1366. await recoverTournament(_ids);
  1367. proxy?.$modal.msgSuccess('恢复成功');
  1368. await getList();
  1369. };
  1370. /** 删除按钮操作 */
  1371. const deleteAllData = async (row?: TournamentsVO) => {
  1372. const _ids = row?.id || ids.value;
  1373. await proxy?.$modal.confirm('是否确认删除比赛ID为"' + _ids + '"的所有关联数据项?').finally(() => (loading.value = false));
  1374. await deleteRelationTournament(_ids);
  1375. proxy?.$modal.msgSuccess('删除成功');
  1376. await getList();
  1377. };
  1378. const publishToTournament2 = async (row?: TournamentsVO) => {
  1379. const _ids = row?.id || ids.value;
  1380. await proxy?.$modal.confirm('是否发布比赛ID为"' + _ids + '"的数据项?').finally(() => (loading.value = false));
  1381. await publishToTournament(_ids);
  1382. proxy?.$modal.msgSuccess('发布成功');
  1383. await getList();
  1384. };
  1385. /** 删除按钮操作 */
  1386. const closeSendTournamentOperate = async (row?: TournamentsVO) => {
  1387. // 1. 获取要关闭的 ID 列表
  1388. const _ids = row ? [row.id] : ids.value; // 始终为 number[]
  1389. if (!_ids || _ids.length === 0) {
  1390. proxy?.$modal.msgWarning('请至少选择一条记录');
  1391. return;
  1392. }
  1393. // 2. 确认操作
  1394. try {
  1395. await proxy?.$modal.confirm(`是否确认关闭比赛ID为 "${_ids.join(', ')}" 的比赛?`);
  1396. // 3. 批量关闭(假设 closeSendTournament 支持传数组)
  1397. await closeSendTournament(_ids); // 推荐:API 支持批量
  1398. // 如果 API 只支持单个,则用循环:
  1399. // for (const id of _ids) {
  1400. // await closeSendTournament(id);
  1401. // }
  1402. proxy?.$modal.msgSuccess('操作成功');
  1403. await getList();
  1404. } catch (error) {
  1405. // 用户取消或请求失败
  1406. console.log('取消操作或操作失败');
  1407. } finally {
  1408. loading.value = false;
  1409. }
  1410. };
  1411. /** 导出按钮操作 */
  1412. const handleExport = () => {
  1413. proxy?.download(
  1414. 'business/tournaments/export',
  1415. {
  1416. ...queryParams.value
  1417. },
  1418. `赛事列表${parseTime(new Date(), '{y}{m}{d}{h}{i}{s}')}.xlsx`
  1419. );
  1420. };
  1421. //导出审核数据
  1422. // 修改导出审核数据函数,传入比赛ID
  1423. const handleExportAuditData = () => {
  1424. proxy?.download(
  1425. 'business/claims/export',
  1426. {
  1427. tournamentId: auditQueryParams.tournamentId
  1428. },
  1429. `领奖审核列表${parseTime(new Date(), '{y}{m}{d}{h}{i}{s}')}.xlsx`
  1430. );
  1431. };
  1432. onMounted(() => {
  1433. // 设置默认开始时间为今天
  1434. const today = new Date();
  1435. const yyyy = today.getFullYear();
  1436. const mm = String(today.getMonth() + 1).padStart(2, '0'); // 月份从0开始,所以+1
  1437. const dd = String(today.getDate()).padStart(2, '0');
  1438. queryParams.value.startTime = `${yyyy}-${mm}-${dd}`;
  1439. console.log('tournaments_type:', tournaments_type.value);
  1440. console.log('tournaments_time:', tournaments_time.value);
  1441. getList();
  1442. loadItemOptions();
  1443. loadItemStructuresOptions();
  1444. // 直接调用 loadSelectData 来加载初始数据
  1445. loadSelectData('');
  1446. loadTagOptions();
  1447. loadCategoryOptions();
  1448. loadConfigOptions();
  1449. });
  1450. async function getAssignList() {
  1451. const res = await getSelectTournamentBlindStructuresList(assignQuery);
  1452. assignList.value = res.rows;
  1453. assignTotal.value = res.total;
  1454. }
  1455. async function openAssignDialog(row: TournamentsVO) {
  1456. assignForm.tournamentId = Number(row.id);
  1457. assignQuery.tournamentId = Number(row.id); // 这里明确转成 number
  1458. // 加载盲注列表
  1459. assignDialog.visible = true;
  1460. // 请求分配盲注结构列表
  1461. getAssignList().then(() => {
  1462. nextTick(() => {
  1463. initSelectedRows(); // 等待 DOM 完全渲染后再初始化选中状态
  1464. });
  1465. });
  1466. }
  1467. function handleAssignSelectionChange(selection: TournamentsBindStructuresVO[]) {
  1468. assignSelectedList.value = selection;
  1469. assignForm.selectedIds = selection.map((item) => item.blindStructuresId);
  1470. }
  1471. async function submitAssign() {
  1472. if (!assignForm.tournamentId || assignForm.selectedIds.length === 0) {
  1473. proxy?.$modal.msgError('请选择至少一个盲注结构');
  1474. return;
  1475. }
  1476. buttonLoading.value = true;
  1477. try {
  1478. await assignTournamentBlindStructures({
  1479. tournamentId: assignForm.tournamentId,
  1480. blindStructureIds: assignForm.selectedIds
  1481. });
  1482. proxy?.$modal.msgSuccess('分配成功');
  1483. assignDialog.visible = false;
  1484. } catch (err) {
  1485. proxy?.$modal.msgError('分配失败');
  1486. } finally {
  1487. buttonLoading.value = false;
  1488. }
  1489. }
  1490. const assignTable = ref(); // 对应 el-table 的 ref
  1491. // 初始化选中行
  1492. function initSelectedRows() {
  1493. const selected = assignList.value.find((item) => item.allocationStatus === 1);
  1494. if (selected) {
  1495. selectedRadio.value = selected.blindStructuresId;
  1496. assignForm.selectedIds = [selected.blindStructuresId];
  1497. }
  1498. }
  1499. watch(
  1500. () => form.value.gameType,
  1501. (newVal) => {
  1502. if (newVal === '3') {
  1503. form.value.itemsId = null;
  1504. form.value.itemsNum = null;
  1505. }
  1506. }
  1507. );
  1508. watch(
  1509. () => assignList.value,
  1510. (newVal) => {
  1511. if (newVal.length > 0) {
  1512. nextTick(() => {
  1513. initSelectedRows();
  1514. });
  1515. }
  1516. },
  1517. { deep: true, immediate: true }
  1518. );
  1519. watch(
  1520. () => selectedRadio.value,
  1521. (newVal) => {
  1522. if (newVal !== null) {
  1523. assignForm.selectedIds = [newVal];
  1524. } else {
  1525. assignForm.selectedIds = [];
  1526. }
  1527. }
  1528. );
  1529. const handleIconChange = async (file) => {
  1530. const index = fileList.value.findIndex((f) => f.uid === file.uid);
  1531. fileList.value = [];
  1532. if (file.raw) {
  1533. iconPreviewUrl.value = URL.createObjectURL(file.raw);
  1534. }
  1535. if (index === -1) {
  1536. // 如果文件不在列表中,则添加进去
  1537. fileList.value.push(file);
  1538. }
  1539. try {
  1540. const rawFile = file.raw;
  1541. const res = await uploadTournament(rawFile);
  1542. if (res.code === 200) {
  1543. // 更新文件状态为成功
  1544. fileList.value[index] = {
  1545. ...file,
  1546. status: 'success',
  1547. response: res.data.url
  1548. };
  1549. data.form.competitionIcon = fileList.value[index].response;
  1550. iconPreviewUrl.value = fileList.value[index].response;
  1551. ElMessage.success('上传成功');
  1552. } else {
  1553. throw new Error(res.msg);
  1554. }
  1555. } catch (error) {
  1556. // 更新文件状态为失败
  1557. fileList.value[index] = {
  1558. ...file,
  1559. status: 'fail',
  1560. error: '上传失败'
  1561. };
  1562. ElMessage.error('上传失败,请重试');
  1563. }
  1564. };
  1565. const handleIconChange2 = async (file) => {
  1566. const index = fileList2.value.findIndex((f) => f.uid === file.uid);
  1567. fileList2.value = [];
  1568. if (file.raw) {
  1569. iconPreviewUrl2.value = URL.createObjectURL(file.raw);
  1570. }
  1571. if (index === -1) {
  1572. // 如果文件不在列表中,则添加进去
  1573. fileList2.value.push(file);
  1574. }
  1575. try {
  1576. const rawFile = file.raw;
  1577. const res = await uploadTournament(rawFile);
  1578. if (res.code === 200) {
  1579. // 更新文件状态为成功
  1580. fileList2.value[index] = {
  1581. ...file,
  1582. status: 'success',
  1583. response: res.data.url
  1584. };
  1585. data.form.competitionBg = fileList2.value[index].response;
  1586. iconPreviewUrl2.value = fileList2.value[index].response;
  1587. ElMessage.success('上传成功');
  1588. } else {
  1589. throw new Error(res.msg);
  1590. }
  1591. } catch (error) {
  1592. // 更新文件状态为失败
  1593. fileList2.value[index] = {
  1594. ...file,
  1595. status: 'fail',
  1596. error: '上传失败'
  1597. };
  1598. ElMessage.error('上传失败,请重试');
  1599. }
  1600. };
  1601. const handleViewLevels = () => {
  1602. const blindStructureId = data.form.blindStructureId;
  1603. if (blindStructureId === null || blindStructureId === undefined) {
  1604. ElMessage.warning('请先选择一个盲注等级');
  1605. return;
  1606. }
  1607. dialogParams.value.blindStructureId = blindStructureId;
  1608. /* dialogParams.value.name = row.name;*/
  1609. levelsDialogVisible.value = true;
  1610. };
  1611. // 点击预览图触发放大
  1612. const handlePreviewClick = () => {
  1613. const currentSrc = iconPreviewUrl.value || competitionIcon.value;
  1614. if (currentSrc) {
  1615. dialogVisible.value = true;
  1616. }
  1617. };
  1618. // 点击预览图触发放大
  1619. const handlePreviewClick2 = () => {
  1620. const currentSrc = iconPreviewUrl2.value || competitionBg.value;
  1621. if (currentSrc) {
  1622. dialogVisible2.value = true;
  1623. }
  1624. };
  1625. // 删除文件处理函数
  1626. const handleIconRemove = (file, updatedFileList) => {
  1627. fileList.value = updatedFileList;
  1628. // 清除预览图和临时链接
  1629. iconPreviewUrl.value = '';
  1630. competitionIcon.value = ''; // 如果需要清除后台加载的图标,可以在这里设置为空字符串
  1631. // 同时清空表单数据中的图标链接
  1632. data.form.competitionIcon = null;
  1633. };
  1634. // 删除文件处理函数
  1635. const handleIconRemove2 = (file, updatedFileList) => {
  1636. fileList2.value = updatedFileList;
  1637. // 清除预览图和临时链接
  1638. iconPreviewUrl2.value = '';
  1639. competitionBg.value = ''; // 如果需要清除后台加载的图标,可以在这里设置为空字符串
  1640. // 同时清空表单数据中的背景图链接
  1641. data.form.competitionBg = null;
  1642. };
  1643. // 排序字段和顺序
  1644. const sortData = ref({
  1645. prop: 'startTime',
  1646. order: 'descending' as 'ascending' | 'descending' | null
  1647. });
  1648. /**
  1649. * 处理排序变化
  1650. */
  1651. const handleSortChange = ({ prop, order }) => {
  1652. if (prop === 'startTime') {
  1653. sortData.value = {
  1654. prop,
  1655. order
  1656. };
  1657. // 触发重新加载数据
  1658. getList();
  1659. }
  1660. };
  1661. const auditDialog = ref({
  1662. visible: false,
  1663. title: '',
  1664. mode: 'view' as 'view' | 'audit'
  1665. });
  1666. const tournamentInfo = ref<Partial<TournamentsVO>>({
  1667. id: '',
  1668. startTime: '',
  1669. endTime: '',
  1670. itemsNum: 0,
  1671. itemsName: '',
  1672. blindStructureId: 0,
  1673. blindStructuresName: '',
  1674. lateRegistrationLevel: 0,
  1675. tournamentsBiId: '',
  1676. signNum: 0,
  1677. isComplaints: false
  1678. });
  1679. const auditData = ref<ClaimsVO[]>([]);
  1680. // 创建新的查询参数对象,不影响原始 queryParams
  1681. const auditQueryParams = {
  1682. pageNum: 1,
  1683. pageSize: 10,
  1684. tournamentId: null as number | null
  1685. };
  1686. const auditTotal = ref(0);
  1687. // 模拟打开对话框并获取数据
  1688. const openAuditDialog = async (row: TournamentsVO | number, mode: 'view' | 'audit' = 'view') => {
  1689. auditDialog.value.visible = true;
  1690. auditDialog.value.mode = mode; // 设置模式
  1691. let tournamentId: number;
  1692. if (typeof row === 'number') {
  1693. tournamentId = row;
  1694. }
  1695. auditQueryParams.tournamentId = tournamentId;
  1696. await getAuditData(1);
  1697. try {
  1698. const res = await getTournaments(tournamentId);
  1699. if (res.code === 200) {
  1700. tournamentInfo.value = res.data;
  1701. }
  1702. } catch (error) {
  1703. console.error('获取数据失败', error);
  1704. }
  1705. };
  1706. const getAuditData = async (pageNum: number) => {
  1707. try {
  1708. auditQueryParams.pageNum = pageNum;
  1709. const res = await listClaims(auditQueryParams);
  1710. auditData.value = res.rows;
  1711. auditTotal.value = res.total;
  1712. } catch (error) {
  1713. console.error('获取审核数据失败', error);
  1714. }
  1715. };
  1716. // 关闭对话框
  1717. const cancelAudit = () => {
  1718. auditDialog.value.visible = false;
  1719. /*tournamentInfo.value = {};
  1720. auditData.value = [];*/
  1721. };
  1722. const removeDialog = ref({
  1723. visible: false,
  1724. id: null,
  1725. tournamentId: null,
  1726. playerName: '',
  1727. phone: ''
  1728. });
  1729. const handleClaimsDelete = (row: ClaimsVO) => {
  1730. removeDialog.value.visible = true;
  1731. removeDialog.value.id = row.id; // 假设 row 中有 id 字段
  1732. removeDialog.value.tournamentId = row.tournamentId; // 假设 row 中有 tournamentId 字段
  1733. removeDialog.value.playerName = row.playerName; // 假设 row 中有 playerName 字段
  1734. removeDialog.value.phone = row.phone;
  1735. };
  1736. const cancelRemove = () => {
  1737. removeDialog.value.visible = false;
  1738. };
  1739. const confirmRemove = async () => {
  1740. try {
  1741. const res = await deleteClaim({
  1742. id: removeDialog.value.id,
  1743. tournamentId: removeDialog.value.tournamentId
  1744. });
  1745. if (res.code === 200) {
  1746. ElMessage.success('移除成功');
  1747. // 重新加载审核数据
  1748. const auditQueryParams = {
  1749. pageNum: 1,
  1750. pageSize: 10,
  1751. tournamentId: tournamentInfo.value.id
  1752. };
  1753. const res2 = await listClaims(auditQueryParams);
  1754. auditData.value = res2.rows;
  1755. getAuditData(1);
  1756. } else {
  1757. ElMessage.error(res.msg);
  1758. }
  1759. } catch (error) {
  1760. ElMessage.error('移除失败,请重试');
  1761. } finally {
  1762. removeDialog.value.visible = false;
  1763. }
  1764. };
  1765. // 审核操作
  1766. const handleAudit = async (row: ClaimsVO) => {
  1767. try {
  1768. await ElMessageBox.confirm('确定要审核该记录吗?', '提示', {
  1769. confirmButtonText: '确定',
  1770. cancelButtonText: '取消',
  1771. type: 'warning'
  1772. });
  1773. // 用户点击了【确定】
  1774. await auditSendReward(row); // 调用接口,假设接口接受 id 参数
  1775. ElMessage.success('审核成功');
  1776. getAuditData(1);
  1777. } catch (error) {
  1778. if (error === 'cancel') {
  1779. ElMessage.info('已取消审核');
  1780. } else {
  1781. ElMessage.error('审核失败');
  1782. }
  1783. }
  1784. };
  1785. const handleViewLevelsSee = (tournamentsVO: TournamentsVO | undefined | null) => {
  1786. if (!tournamentsVO || !tournamentsVO.blindStructureId) {
  1787. proxy?.$modal.msgWarning('该赛事未绑定盲注等级');
  1788. return;
  1789. }
  1790. dialogParams.value.blindStructureId = tournamentsVO.blindStructureId;
  1791. dialogParams.value.name = tournamentsVO.blindStructuresName ?? '';
  1792. levelsDialogVisible.value = true;
  1793. };
  1794. /** 修改按钮操作 */
  1795. const handleCopy = async (row?: TournamentsVO) => {
  1796. reset(); // 重置表单
  1797. const _id = row?.id || ids.value[0];
  1798. const res = await getTournaments(_id);
  1799. // 设置表单数据
  1800. res.data.id = null;
  1801. res.data.status = null;
  1802. Object.assign(form.value, res.data);
  1803. form.value.signTime = String(res.data.signTime);
  1804. form.value.gameVariant = String(res.data.gameVariant);
  1805. form.value.qualifierType = String(res.data.qualifierType);
  1806. form.value.gameType = String(res.data.gameType); // 转为字符串
  1807. form.value.status = 0;
  1808. form.value.id = null;
  1809. form.value.levelDuration = null;
  1810. form.value.delayCardTime = res.data.delayCardTime;
  1811. form.value.delayCardNum = res.data.delayCardNum;
  1812. competitionIcon.value = res.data.competitionIcon;
  1813. // 处理奖励表单数据
  1814. const prizeItems = res.data.itemsPrizeList || [];
  1815. if (prizeItems.length > 0) {
  1816. formPrize.rewards = prizeItems.map((item, index) => ({
  1817. ranking: item.ranking || index + 1,
  1818. itemId: Number(item.itemId),
  1819. quantity: Number(item.quantity)
  1820. }));
  1821. } else {
  1822. formPrize.rewards = [
  1823. {
  1824. ranking: 1,
  1825. itemId: null,
  1826. quantity: null
  1827. }
  1828. ];
  1829. }
  1830. // ✅ 主动触发盲注等级加载
  1831. if (form.value.blindStructureId) {
  1832. await handleBlindStructureChange(form.value.blindStructureId);
  1833. }
  1834. dialog.mode = 'add'; // 设置为新增模式
  1835. dialog.visible = true;
  1836. dialog.title = '创建比赛';
  1837. };
  1838. const handleViewHistory = (row: ClaimsVO) => {
  1839. const tournamentId = row.tournamentId;
  1840. const playerId = row.playerId;
  1841. // 判断 playerId 是否有效(非 null、undefined)
  1842. const playerIdStr = playerId != null ? String(playerId) : '';
  1843. proxy?.$router.push({
  1844. path: '/service/history',
  1845. query: { tournamentId: String(tournamentId), playerId: playerIdStr }
  1846. });
  1847. auditDialog.value.visible = false;
  1848. };
  1849. const handleViewPublicHistory = (row: TournamentsVO) => {
  1850. const tournamentId = row.id;
  1851. const playerId = null;
  1852. // 判断 playerId 是否有效(非 null、undefined)
  1853. const playerIdStr = playerId != null ? String(playerId) : '';
  1854. proxy?.$router.push({
  1855. path: '/service/history',
  1856. query: { tournamentId: String(tournamentId), playerId: playerIdStr }
  1857. });
  1858. };
  1859. // 跳转到盲注表页面,并关闭当前弹窗
  1860. const handleGoToStructures = () => {
  1861. dialog.visible = false; // 关闭弹窗
  1862. // 使用 nextTick 确保关闭动画完成后跳转(可选)
  1863. nextTick(() => {
  1864. proxy?.$router.push('/tournament/structures');
  1865. });
  1866. };
  1867. const handleGoToTag = () => {
  1868. dialog.visible = false; // 关闭弹窗
  1869. // 使用 nextTick 确保关闭动画完成后跳转(可选)
  1870. nextTick(() => {
  1871. proxy?.$router.push('/service/tag');
  1872. });
  1873. };
  1874. const handleGoToCategory = () => {
  1875. dialog.visible = false; // 关闭弹窗
  1876. // 使用 nextTick 确保关闭动画完成后跳转(可选)
  1877. nextTick(() => {
  1878. proxy?.$router.push('/service/catory');
  1879. });
  1880. };
  1881. // 获取奖励提示内容
  1882. const getRewardTooltipContent = (rewards: any[]) => {
  1883. return rewards.map((prize) => `第${prize.ranking}名:${prize.quantity} ${prize.itemsName}`).join('\n');
  1884. };
  1885. // 判断是否为正式环境
  1886. const isProdEnvironment = computed(() => {
  1887. // 根据实际项目中的环境变量判断是否为正式环境
  1888. return import.meta.env.MODE === 'production' || import.meta.env.VITE_APP_ENV === 'prod';
  1889. });
  1890. // 在 setup 中添加
  1891. const selectOptions = ref<TournamentsVO[]>([]);
  1892. const selectLoading = ref(false);
  1893. const selectPageNum = ref(1);
  1894. const selectPageSize = ref(100); // 每页数量
  1895. const selectTotal = ref(0);
  1896. // 远程搜索方法
  1897. const remoteMethod = async (query: string) => {
  1898. if (query !== '') {
  1899. selectPageNum.value = 1;
  1900. await loadSelectData(query);
  1901. } else {
  1902. selectOptions.value = [];
  1903. }
  1904. };
  1905. // 加载数据(支持分页)
  1906. const loadSelectData = async (query: string) => {
  1907. selectLoading.value = true;
  1908. const params = {
  1909. pageNum: selectPageNum.value,
  1910. pageSize: selectPageSize.value,
  1911. status: 0, // 只查 status=0
  1912. name: query, // 搜索关键词
  1913. sortBy: 'name',
  1914. isAsc: true
  1915. };
  1916. try {
  1917. const res = await listTournaments(params);
  1918. const rows = res.rows || [];
  1919. if (selectPageNum.value === 1) {
  1920. selectOptions.value = rows;
  1921. } else {
  1922. selectOptions.value.push(...rows);
  1923. }
  1924. selectTotal.value = res.total;
  1925. } catch (error) {
  1926. console.error('加载比赛列表失败:', error);
  1927. } finally {
  1928. selectLoading.value = false;
  1929. }
  1930. };
  1931. // 监听下拉框展开时触发加载
  1932. const handleSelectVisibleChange = async (visible: boolean) => {
  1933. if (visible && selectOptions.value.length === 0) {
  1934. await remoteMethod('');
  1935. }
  1936. };
  1937. const handleQualifierTypeChange = (value: string) => {
  1938. // Clear the qualifierValue when qualifierType changes
  1939. form.value.qualifierValue = null;
  1940. };
  1941. // 下拉选项数据 selectBlingStructuresInfo
  1942. const itemOptionsTagList = ref<{ id: number; label: string }[]>([]);
  1943. // 加载报名条件选项
  1944. const loadTagOptions = async () => {
  1945. try {
  1946. const res = await selectTagSelList();
  1947. if (res.code === 200) {
  1948. // 使用 unknown 中间类型进行类型转换
  1949. const data = res.data as unknown as { id: number; name: string }[];
  1950. const list = [];
  1951. for (let i = 0; i < data.length; i++) {
  1952. const item = data[i];
  1953. list.push({
  1954. id: item.id,
  1955. label: item.name
  1956. });
  1957. }
  1958. itemOptionsTagList.value = list;
  1959. } else {
  1960. alert('加载失败:' + res.msg);
  1961. }
  1962. } catch (error) {
  1963. console.error('请求出错:', error);
  1964. }
  1965. };
  1966. // 下拉选项数据 selectBlingStructuresInfo
  1967. const itemOptionsCategoryList = ref<{ id: number; label: string }[]>([]);
  1968. // 加载报名条件选项
  1969. const loadCategoryOptions = async () => {
  1970. try {
  1971. const res = await selectCategorySelList();
  1972. if (res.code === 200) {
  1973. // 使用 unknown 中间类型进行类型转换
  1974. const data = res.data as unknown as { id: number; name: string }[];
  1975. const list = [];
  1976. for (let i = 0; i < data.length; i++) {
  1977. const item = data[i];
  1978. list.push({
  1979. id: item.id,
  1980. label: item.name
  1981. });
  1982. }
  1983. itemOptionsCategoryList.value = list;
  1984. } else {
  1985. alert('加载失败:' + res.msg);
  1986. }
  1987. } catch (error) {
  1988. console.error('请求出错:', error);
  1989. }
  1990. };
  1991. const handleIsDelayChange = (value: number) => {
  1992. if (value === 0) {
  1993. // 当选择"否"时,将延迟卡时间和数量设置为0
  1994. form.value.delayCardTime = 0;
  1995. form.value.delayCardNum = 0;
  1996. } else {
  1997. form.value.delayCardTime = 15;
  1998. form.value.delayCardNum = 4;
  1999. }
  2000. };
  2001. const getGameVariantText = (value: string | number | null | undefined): string => {
  2002. const numValue = Number(value);
  2003. switch (numValue) {
  2004. case 0:
  2005. return '德州扑克';
  2006. case 1:
  2007. return '奥马哈';
  2008. case 2:
  2009. return '短牌';
  2010. default:
  2011. return '未知';
  2012. }
  2013. };
  2014. // 下拉选项数据 selectBlingStructuresInfo
  2015. const configOptions = ref<{ id: number; label: string }[]>([]);
  2016. // 加载报名条件选项
  2017. const loadConfigOptions = async () => {
  2018. try {
  2019. const res = await selectTournamentConfigSelList();
  2020. if (res.code === 200) {
  2021. // 类型断言
  2022. const data = res.data as unknown as { id: number; name: string }[];
  2023. // 过滤掉 id === 2 的项,并映射为 { id, label }
  2024. configOptions.value = data.map((item) => ({
  2025. id: item.id,
  2026. label: item.name
  2027. }));
  2028. } else {
  2029. ElMessage.error('加载失败:' + res.msg);
  2030. }
  2031. } catch (error) {
  2032. console.error('请求出错:', error);
  2033. ElMessage.error('请求失败,请检查网络');
  2034. }
  2035. };
  2036. </script>
  2037. <style scoped>
  2038. .more-rewards {
  2039. cursor: pointer;
  2040. color: #409eff;
  2041. margin-left: 5px;
  2042. }
  2043. </style>