| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249 |
- import type {SelectOption} from 'naive-ui'
- import {
- NAlert,
- NButton,
- NCascader,
- NCheckbox,
- NForm,
- NFormItemGi,
- NGi,
- NGrid,
- NInput,
- NInputNumber,
- NModal,
- NRadio,
- NRadioGroup,
- NSelect,
- NSpace,
- useDialog,
- useMessage,
- NCheckboxGroup, NCol
- } from 'naive-ui'
- import {defineComponent, onMounted, PropType, reactive, ref} from 'vue'
- import {musicSheetDetail, musicSheetSave, musicSheetUpdate} from '../../api'
- import UploadFile from '@/components/upload-file'
- import styles from './index.module.less'
- import deepClone from '@/utils/deep.clone'
- import axios from 'axios'
- import {musicSheetSourceType, musicSheetType} from "@/utils/constant";
- import {getSelectDataFromObj} from "@/utils/objectUtil";
- import {musicalInstrumentPage} from "@views/system-manage/subject-manage/api";
- import {subjectPage} from "@views/system-manage/api";
- import MusicSheetOwnerDialog from "@views/music-library/music-sheet/modal/musicSheetOwnerDialog";
- /**
- * 获取指定元素下一个Note元素
- * @param ele 指定元素
- * @param selectors 选择器
- */
- const getNextNote = (ele: any, selectors: any) => {
- let index = 0
- const parentEle = ele.closest(selectors)
- let pointer = parentEle
- const measure = parentEle?.closest('measure')
- let siblingNote = null
- // 查找到相邻的第一个note元素
- while (!siblingNote && index < (measure?.childNodes.length || 50)) {
- index++
- if (pointer?.nextElementSibling?.tagName === 'note') {
- siblingNote = pointer?.nextElementSibling
- }
- pointer = pointer?.nextElementSibling
- }
- return siblingNote
- }
- export const onlyVisible = (xml: any, partIndex: any) => {
- if (!xml) return ''
- const xmlParse = new DOMParser().parseFromString(xml, 'text/xml')
- const partList =
- xmlParse.getElementsByTagName('part-list')?.[0]?.getElementsByTagName('score-part') || []
- const parts = xmlParse.getElementsByTagName('part')
- const visiblePartInfo = partList[partIndex]
- if (visiblePartInfo) {
- const id = visiblePartInfo.getAttribute('id')
- Array.from(parts).forEach((part) => {
- if (part && part.getAttribute('id') !== id) {
- part.parentNode?.removeChild(part)
- // 不等于第一行才添加避免重复添加
- }
- // 最后一个小节的结束线元素不在最后 调整
- if (part && part.getAttribute('id') === id) {
- const barlines = part.getElementsByTagName('barline')
- const lastParent = barlines[barlines.length - 1]?.parentElement
- if (lastParent?.lastElementChild?.tagName !== 'barline') {
- const children: any[] = (lastParent?.children as any) || []
- for (let el of children) {
- if (el.tagName === 'barline') {
- // 将结束线元素放到最后
- lastParent?.appendChild(el)
- break
- }
- }
- }
- }
- })
- Array.from(partList).forEach((part) => {
- if (part && part.getAttribute('id') !== id) {
- part.parentNode?.removeChild(part)
- }
- })
- // 处理装饰音问题
- const notes = xmlParse.getElementsByTagName('note')
- const getNextvNoteDuration = (i: any) => {
- let nextNote = notes[i + 1]
- // 可能存在多个装饰音问题,取下一个非装饰音时值
- for (let index = i; index < notes.length; index++) {
- const note = notes[index]
- if (!note.getElementsByTagName('grace')?.length) {
- nextNote = note
- break
- }
- }
- return nextNote?.getElementsByTagName('duration')[0]
- }
- Array.from(notes).forEach((note, i) => {
- const graces = note.getElementsByTagName('grace')
- if (graces && graces.length) {
- note.appendChild(getNextvNoteDuration(i)?.cloneNode(true))
- }
- })
- }
- return new XMLSerializer().serializeToString(xmlParse)
- }
- const speedInfo = {
- 'rall.': 1.333333333,
- 'poco rit.': 1.333333333,
- 'rit.': 1.333333333,
- 'molto rit.': 1.333333333,
- 'molto rall': 1.333333333,
- molto: 1.333333333,
- lentando: 1.333333333,
- allargando: 1.333333333,
- morendo: 1.333333333,
- 'accel.': 0.8,
- calando: 2,
- 'poco accel.': 0.8,
- 'gradually slowing': 1.333333333,
- slowing: 1.333333333,
- slow: 1.333333333,
- slowly: 1.333333333,
- faster: 1.333333333
- }
- /**
- * 按照xml进行减慢速度的计算
- * @param xml 始终按照第一分谱进行减慢速度的计算
- */
- export function getGradualLengthByXml(xml: string) {
- const firstPartXml = onlyVisible(xml, 0)
- const xmlParse = new DOMParser().parseFromString(firstPartXml, 'text/xml')
- const measures = Array.from(xmlParse.querySelectorAll('measure'))
- const notes = Array.from(xmlParse.querySelectorAll('note'))
- const words = Array.from(xmlParse.querySelectorAll('words'))
- const metronomes = Array.from(xmlParse.querySelectorAll('metronome'))
- const eles = []
- for (const ele of [...words, ...metronomes]) {
- const note = getNextNote(ele, 'direction')
- // console.log(ele, note)
- if (note) {
- const measure = note?.closest('measure')
- const measureNotes = Array.from(measure.querySelectorAll('note'))
- const noteInMeasureIndex = Array.from(measure.childNodes)
- .filter((item: any) => item.nodeName === 'note')
- .findIndex((item) => item === note)
- let allDuration = 0
- let leftDuration = 0
- for (let i = 0; i < measureNotes.length; i++) {
- const n: any = measureNotes[i]
- const duration = +(n.querySelector('duration')?.textContent || '0')
- allDuration += duration
- if (i < noteInMeasureIndex) {
- leftDuration = allDuration
- }
- }
- eles.push({
- ele,
- index: notes.indexOf(note),
- noteInMeasureIndex,
- textContent: ele.textContent,
- measureIndex: measures.indexOf(measure), //,measure?.getAttribute('number')
- type: ele.tagName,
- allDuration,
- leftDuration
- })
- }
- }
- // 结尾处手动插入一个音符节点
- eles.push({
- ele: notes[notes.length - 1],
- index: notes.length,
- noteInMeasureIndex: 0,
- textContent: '',
- type: 'metronome',
- allDuration: 1,
- leftDuration: 1,
- measureIndex: measures.length
- })
- const gradualNotes: any[] = []
- eles.sort((a, b) => a.index - b.index)
- const keys = Object.keys(speedInfo).map((w) => w.toLocaleLowerCase())
- let isLastNoteAndNotClosed = false
- for (const ele of eles) {
- const textContent: any = ele.textContent?.toLocaleLowerCase().trim()
- if (ele === eles[eles.length - 1]) {
- if (gradualNotes[gradualNotes.length - 1]?.length === 1) {
- isLastNoteAndNotClosed = true
- }
- }
- const isKeyWork = keys.find((k) => {
- const ks = k.split(' ')
- return textContent && ks.includes(textContent)
- })
- if (
- ele.type === 'metronome' ||
- (ele.type === 'words' && (textContent.startsWith('a tempo') || isKeyWork)) ||
- isLastNoteAndNotClosed
- ) {
- const indexOf = gradualNotes.findIndex((item) => item.length === 1)
- if (indexOf > -1 && ele.index > gradualNotes[indexOf]?.[0].start) {
- gradualNotes[indexOf][1] = {
- start: ele.index,
- measureIndex: ele.measureIndex,
- noteInMeasureIndex: ele.noteInMeasureIndex,
- allDuration: ele.allDuration,
- leftDuration: ele.leftDuration,
- type: textContent
- }
- }
- }
- if (ele.type === 'words' && isKeyWork) {
- gradualNotes.push([
- {
- start: ele.index,
- measureIndex: ele.measureIndex,
- noteInMeasureIndex: ele.noteInMeasureIndex,
- allDuration: ele.allDuration,
- leftDuration: ele.leftDuration,
- type: textContent
- }
- ])
- }
- }
- return gradualNotes
- }
- export default defineComponent({
- name: 'music-operation',
- props: {
- type: {
- type: String,
- default: 'add'
- },
- data: {
- type: Object as PropType<any>,
- default: () => {
- }
- },
- tagList: {
- type: Array as PropType<Array<SelectOption>>,
- default: () => []
- },
- subjectList: {
- type: Array as PropType<Array<SelectOption>>,
- default: () => []
- },
- musicSheetCategories: {
- type: Array as PropType<Array<SelectOption>>,
- default: () => []
- }
- },
- emits: ['close', 'getList'],
- setup(props, {slots, attrs, emit}) {
- const forms = reactive({
- graduals: {} as any, // 渐变速度
- playMode: 'MP3', // 播放类型
- xmlFileUrl: null, // XML
- midiUrl: null, // mid
- name: null, // 曲目名称
- // musicTag: [] as any, // 曲目标签
- composer: null, // 音乐人
- playSpeed: null as any, // 曲谱速度
- // showFingering: null as any, // 是否显示指法
- // canEvaluate: null as any, // 是否评测
- // notation: null as any, // 能否转和简谱
- // auditVersion: null as any, // 审核版本
- // sortNumber: null, // 排序
- musicCover: null, // 曲谱封面
- remark: null, // 曲谱描述
- musicSheetSoundList: [] as any, // 原音
- // musicSheetCategoriesId: null,
- status: false,
- musicSheetType: 'SINGLE', // 曲目类型
- sourceType: undefined, //来源类型/作者属性(PLATFORM: 平台; ORG: 机构; PERSON: 个人)
- // userId: null, // 所属人
- appAuditFlag: 0, // 是否审核版本
- midiFileUrl: null, // 伴奏文件 MIDI文件(保留字段)
- subjectIds: '', // 可用声部
- subjectIdList: [] as any, // 可用声部
- musicalInstrumentIdList: [] as any, //可用乐器
- musicCategoryId: null, //曲目分类
- musicSheetAccompanimentList: [] as any, //曲目伴奏
- audioType: 'HOMEMODE', // 伴奏类型
- isPlayBeat: true, // 是否播放节拍器
- isUseSystemBeat: true, // 是否使用系统节拍器(0:否;1:是)
- repeatedBeats: false, // 是否重复节拍时长
- evaluationStandard: 'FREQUENCY', // 评分标准 节奏 AMPLITUDE 音准 FREQUENCY 分贝 DECIBELS
- multiTracksSelection: [] as any, // 声轨
- musicSheetExtend: {} as any,//所属人信息
- })
- const state = reactive({
- loading: false,
- previewMode: false,//是否是预览模式
- tagList: [...props.tagList] as any, // 标签列表
- xmlFirstSpeed: null as any, // 第一个音轨速度
- partListNames: [] as any, // 所有音轨声部列表
- musicSheetCategories: [...props.musicSheetCategories] as any,
- musicSheetAccompanimentUrls: '' as any,
- musicSheetAccompanimentUrlList: [] as any,
- instrumentData: [],
- instrumentList: [],
- subjectList: [],
- showMusicSheetOwnerDialog: false, //所属人弹框
- // musicSheetOwnerData: {}, //所属人信息
- multiTracks: null,
- })
- const gradualData = reactive({
- list: [] as any[],
- gradualRefs: [] as any[]
- })
- const btnLoading = ref(false)
- const formsRef = ref()
- const message = useMessage()
- const dialog = useDialog()
- // 提交记录
- const onSubmit = async () => {
- formsRef.value.validate(async (error: any) => {
- console.log(error, 'error')
- if (error) {
- message.error(error[0]?.[0]?.message)
- return
- }
- try {
- //extConfigJson: {"repeatedBeats":0,"gradualTimes":{"75":"02:38:60","77":"02:43:39"}}
- const obj = {
- ...forms,
- musicTag: '-1',
- multiTracksSelection: forms.multiTracksSelection.join(','),
- musicSheetSoundList: forms.musicSheetSoundList.filter((next: any) => {
- return !!next.audioFileUrl;
- }),
- musicalInstrumentIds: forms.musicalInstrumentIdList.join(','),
- extConfigJson: JSON.stringify({gradualTimes: forms.graduals})
- }
- if (forms.audioType == 'MIDI') {
- obj.musicSheetSoundList = []
- }
- btnLoading.value = true
- if (props.type === 'add') {
- await musicSheetSave(obj)
- message.success('添加成功')
- } else if (props.type === 'edit') {
- await musicSheetUpdate({...obj, id: props.data.id})
- message.success('修改成功')
- }
- emit('getList')
- emit('close')
- } catch (e) {
- console.log(e)
- }
- setTimeout(() => {
- btnLoading.value = false
- }, 100)
- })
- }
- // 上传XML,初始化音轨 音轨速度 乐器、声部
- const readFileInputEventAsArrayBuffer = (file: any) => {
- const xmlRead = new FileReader()
- xmlRead.onload = (res) => {
- try {
- gradualData.list = getGradualLengthByXml(res?.target?.result as any).filter(
- (item: any) => item.length === 2
- )
- } catch (error) {
- }
- state.partListNames = getPartListNames(res?.target?.result as any) as any
- // 这里是如果没有当前音轨就重新写
- for (let j = 0; j < state.partListNames.length; j++) {
- if (!forms.musicSheetSoundList[j]) {
- forms.musicSheetSoundList.push({audioFileUrl: null, track: null})
- }
- forms.musicSheetSoundList[j].track = state.partListNames[j].value
- }
- // 循环添加所在音轨的原音
- for (let index = forms.musicSheetSoundList.length; index < state.partListNames.length; index++) {
- const part = state.partListNames[index].value
- const sysData = {
- ...forms.musicSheetSoundList[0],
- track: part
- }
- if (!sysData.speed) {
- sysData.speed = state.xmlFirstSpeed
- }
- createSys(sysData)
- }
- if (forms.musicSheetSoundList.length == 0) {
- forms.musicSheetSoundList.push({audioFileUrl: '', track: ''})
- }
- }
- xmlRead.readAsText(file)
- }
- // 获取xml中所有轨道 乐器
- const getPartListNames = (xml: any) => {
- if (!xml) return []
- const xmlParse = new DOMParser().parseFromString(xml, 'text/xml')
- const partList =
- xmlParse.getElementsByTagName('part-list')?.[0]?.getElementsByTagName('score-part') || []
- const partListNames = Array.from(partList).map((item) => {
- const part = item.getElementsByTagName('part-name')?.[0].textContent || ''
- return {
- value: part,
- label: part
- }
- })
- if (partListNames.length > 0) {
- forms.musicSheetSoundList = forms.musicSheetSoundList.slice(0, partListNames.length)
- }
- state.xmlFirstSpeed = xmlParse.getElementsByTagName('per-minute')?.[0]?.textContent || ''
- if (!forms.playSpeed) {
- forms.playSpeed = Number.parseInt(state.xmlFirstSpeed)
- }
- // 乐器
- const instrumentCodeList: any = [];
- const instrumentEle = xmlParse.getElementsByTagName('virtual-instrument');
- for (let index = 0; index < instrumentEle.length; index++) {
- const note = instrumentEle[index]
- const instrumentCode = note.getElementsByTagName('virtual-name')?.[0].textContent || '';
- if (instrumentCode && !instrumentCodeList.includes(instrumentCode)) {
- instrumentCodeList.push(instrumentCode);
- }
- }
- const codeIdMap = new Map<string, string>();
- state.instrumentData.forEach((data: any) => {
- codeIdMap.set(data.code, data.id + '');
- })
- forms.musicalInstrumentIdList = [];
- instrumentCodeList.forEach((code: string) => {
- if (codeIdMap.has(code)) {
- forms.musicalInstrumentIdList.push(codeIdMap.get(code));
- }
- })
- // 声部
- if (forms.musicalInstrumentIdList.length > 0) {
- showBackSubject(forms.musicalInstrumentIdList);
- }
- return partListNames
- }
- // 判断选择的音轨是否在选中
- const initPartsListStatus = (track: string): any => {
- const _names = state.partListNames.filter((n: any) => n.value?.toLocaleUpperCase?.() != 'COMMON')
- const partListNames = deepClone(_names) || []
- partListNames.forEach((item: any) => {
- const index = forms.musicSheetSoundList.findIndex((ground: any) => item.value == ground.track)
- if (index > -1 && track != item.value) {
- item.disabled = true
- } else {
- item.disabled = false
- }
- })
- return partListNames || []
- }
- // 获取乐器信息
- const initInstrumentList = async () => {
- if (state.instrumentList && state.instrumentList.length > 0) {
- return
- }
- try {
- const {data} = await musicalInstrumentPage({page: 1, rows: 999})
- const tempList = data.rows || []
- state.instrumentData = tempList
- tempList.forEach((item: any) => {
- item.label = item.name
- item.value = item.id + ''
- })
- state.instrumentList = tempList
- } catch {
- }
- }
- // 反显声部
- const showBackSubject = async (musicalInstrumentIdList: []) => {
- try {
- const {data} = await subjectPage({page: 1, rows: 999, musicalInstrumentIdList: musicalInstrumentIdList})
- const tempList = data.rows || []
- tempList.forEach((item: any) => {
- forms.subjectIdList.push(item.id + '')
- })
- } catch {
- }
- }
- // 添加原音
- const createSys = (initData?: any) => {
- forms.musicSheetSoundList.push({
- audioFileUrl: null, // 原音
- track: null, // 轨道
- ...initData
- })
- }
- // 删除原音
- const removeSys = (index: number) => {
- dialog.warning({
- title: '警告',
- content: `是否确认删除此原音?`,
- positiveText: '确定',
- negativeText: '取消',
- onPositiveClick: async () => {
- forms.musicSheetSoundList.splice(index, 1)
- }
- })
- }
- const checkMultiTracks = (value: string) => {
- if (!value) {
- return;
- }
- if (value === 'all') {
- forms.multiTracksSelection = []
- state.partListNames.forEach((next: any) => {
- forms.multiTracksSelection.push(next.value)
- })
- } else if (value === 'invert') {
- state.partListNames.forEach((next: any) => {
- const indexOf = forms.multiTracksSelection.indexOf(next.value);
- if (indexOf > -1) {
- forms.multiTracksSelection.splice(indexOf, 1)
- } else {
- forms.multiTracksSelection.push(next.value)
- }
- })
- } else if (value === 'allUncheck') {
- forms.multiTracksSelection = []
- }
- }
- onMounted(async () => {
- initInstrumentList();
- if (props.type === 'edit' || props.type === 'preview') {
- const detail = props.data
- try {
- state.loading = true
- if (props.type === 'preview') {
- state.previewMode = true
- }
- const {data} = await musicSheetDetail({id: detail.id})
- forms.audioType = data.audioType
- forms.playMode = data.playMode
- forms.xmlFileUrl = data.xmlFileUrl
- forms.midiUrl = data.midiUrl
- forms.name = data.name
- // forms.musicTag = data.musicTag?.split(',')
- forms.composer = data.composer
- forms.playSpeed = data.playSpeed
- // forms.showFingering = Number(data.showFingering)
- // forms.canEvaluate = Number(data.canEvaluate)
- // forms.notation = Number(data.notation)
- // forms.auditVersion = Number(data.auditVersion)
- // forms.sortNumber = data.sortNumber
- forms.musicCover = data.musicCover
- forms.remark = data.remark
- forms.status = data.status
- forms.musicCategoryId = data.musicCategoryId
- forms.musicSheetType = data.musicSheetType || "SINGLE"
- forms.musicSheetAccompanimentList = data.musicSheetAccompanimentList
- data.musicSheetAccompanimentList?.forEach((next: any) => {
- state.musicSheetAccompanimentUrlList.push(next.audioFileUrl);
- })
- forms.evaluationStandard = data.evaluationStandard
- forms.musicalInstrumentIdList = data.musicalInstrumentIds.split(',') || []
- forms.subjectIds = data.subjectIds
- forms.subjectIdList = data.subjectIds?.split(',') || []
- forms.sourceType = data.sourceType
- forms.musicSheetExtend = data.musicSheetExtend
- // 获取渐变 和 是否多声部
- try {
- const extConfigJson = data.extConfigJson ? JSON.parse(data.extConfigJson) : {}
- forms.graduals = extConfigJson.gradualTimes || {}
- } catch (error) {
- }
- axios.get(data.xmlFileUrl).then((res: any) => {
- if (res?.data) {
- gradualData.list = getGradualLengthByXml(res?.data as any).filter(
- (item: any) => item.length === 2
- )
- state.partListNames = getPartListNames(res?.data as any) as any
- // 初始化音轨和原音
- forms.musicSheetSoundList = data.musicSheetSoundList || []
- forms.musicSheetSoundList.forEach((next: any) => {
- forms.multiTracksSelection.push(next.track)
- })
- state.partListNames.forEach((item: any) => {
- const tracks = forms.musicSheetSoundList.map((item: any) => item.track);
- if (tracks.indexOf(item.label) <= -1) {
- forms.musicSheetSoundList.push({audioFileUrl: '', track: item.label})
- }
- })
- }
- })
- } catch (error) {
- console.log(error)
- }
- state.loading = false
- }
- })
- return () => (
- <div style="background: #fff; padding-top: 12px">
- <NForm
- class={styles.formContainer}
- model={forms}
- ref={formsRef}
- label-placement="left"
- label-width="130"
- disabled={state.previewMode}
- >
- <NAlert showIcon={false} style={{marginBottom: "12px"}}>曲目信息</NAlert>
- <NGrid cols={2}>
- <NFormItemGi
- label="曲目名称"
- path="name"
- rule={[
- {
- required: true,
- message: '请输入曲目名称'
- }
- ]}
- >
- <NInput
- v-model:value={forms.name}
- placeholder="请输入曲目名称"
- maxlength={120}
- showCount
- />
- </NFormItemGi>
- <NFormItemGi
- label="音乐人"
- path="composer"
- rule={[
- {
- required: true,
- message: '请输入音乐人'
- }
- ]}
- >
- <NInput v-model:value={forms.composer}
- placeholder="请输入音乐人名称"
- showCount
- maxlength={14}
- />
- </NFormItemGi>
- </NGrid>
- <NGrid cols={2}>
- <NFormItemGi label="曲目描述" path="remark">
- <NInput
- placeholder="请输入曲目描述"
- type="textarea"
- rows={4}
- showCount
- maxlength={200}
- v-model:value={forms.remark}
- />
- </NFormItemGi>
- <NFormItemGi label="曲目封面" path="musicCover"
- rule={[
- {
- required: true
- }
- ]}>
- <UploadFile
- disabled={state.previewMode}
- accept=".jpg,.jpeg,.png"
- tips="请上传大小1M以内的JPG、PNG图片"
- size={1}
- v-model:fileList={forms.musicCover}
- cropper
- bucketName="cbs"
- options={{
- autoCrop: true, //是否默认生成截图框
- enlarge: 2, // 图片放大倍数
- autoCropWidth: 200, //默框高度
- fixedBox: true, //是否固定截图框大认生成截图框宽度
- autoCropHeight: 200, //默认生成截图小 不允许改变
- previewsCircle: false, //预览图是否是原圆形
- title: '曲目封面'
- }}
- />
- </NFormItemGi>
- </NGrid>
- <NGrid cols={2}>
- <NFormItemGi
- label="曲目类型"
- path="musicSheetType"
- rule={[
- {
- required: true,
- message: '请选择曲目类型'
- }
- ]}
- >
- <NSelect
- placeholder="请选择曲目类型"
- v-model:value={forms.musicSheetType}
- options={getSelectDataFromObj(musicSheetType)}
- />
- </NFormItemGi>
- <NFormItemGi
- label="作者属性"
- path="sourceType"
- rule={[
- {
- required: true,
- message: '请选择作者属性'
- }
- ]}
- >
- <NSelect
- v-model:value={forms.sourceType}
- options={getSelectDataFromObj(musicSheetSourceType)}
- placeholder="请选择作者属性"
- onUpdateValue={() => {
- // 发送变化,清理选择的所属人信息
- forms.musicSheetExtend = {}
- // forms.musicSheetExtend.userId = null
- // forms.musicSheetExtend.userName = null
- // forms.musicSheetExtend.applicationId = null
- // forms.musicSheetExtend.organizationRoleId = null
- }}
- />
- </NFormItemGi>
- </NGrid>
- <NGrid cols={2}>
- <NFormItemGi
- label="所属人"
- path="musicSheetExtend.userId"
- rule={[
- {
- required: true,
- message: '请选择曲目所属人'
- }
- ]}
- >
- <NButton
- disabled={state.previewMode || !forms.sourceType}
- type="primary"
- size="small"
- text
- //v-auth="orchestraSubsidyStandard/update1597887579789053953"
- onClick={() => {
- state.showMusicSheetOwnerDialog = true
- }}
- >
- {forms.musicSheetExtend?.userId ? forms.musicSheetExtend.userName + "(" + forms.musicSheetExtend.userId + ")" : '请选择所属人'}
- </NButton>
- </NFormItemGi>
- <NFormItemGi label="速度" path="playSpeed">
- <NInputNumber
- placeholder="请输入速度"
- v-model:value={forms.playSpeed}
- style="width:100%"
- />
- </NFormItemGi>
- </NGrid>
- <NGrid cols={2}>
- <NFormItemGi label="审核版本" path="appAuditFlag"
- rule={[
- {
- required: true,
- message: '请选择曲目所属人'
- }
- ]}
- >
- <NSelect
- options={
- [
- {
- label: '是',
- value: 1
- },
- {
- label: '否',
- value: 0
- }
- ] as any
- }
- v-model:value={forms.appAuditFlag}
- />
- </NFormItemGi>
- <NFormItemGi label="曲目分类" path="musicCategoryId"
- rule={[
- {
- required: true,
- message: '请选择曲目分类'
- }
- ]}
- >
- <NCascader
- valueField="id"
- labelField="name"
- children-field="musicSheetCategoriesList"
- placeholder="请选择分类"
- v-model:value={forms.musicCategoryId}
- options={state.musicSheetCategories}
- clearable
- />
- </NFormItemGi>
- </NGrid>
- <NGrid cols={2}>
- <NFormItemGi label="是否重复节拍时长" path="repeatedBeats"
- rule={[
- {
- required: true,
- message: '请选择是否重复节拍时长'
- }
- ]}
- >
- <NRadioGroup
- v-model:value={forms.repeatedBeats}
- >
- <NRadio value={true}>是</NRadio>
- <NRadio value={false}>否</NRadio>
- </NRadioGroup>
- </NFormItemGi>
- <NFormItemGi
- label="评分标准"
- path="evaluationStandard"
- rule={[
- {
- required: true
- }
- ]}
- >
- <NRadioGroup
- v-model:value={forms.evaluationStandard}
- >
- <NRadio value={'FREQUENCY'}>标准评测</NRadio>
- <NRadio value={'AMPLITUDE'}>打击乐(振幅)</NRadio>
- <NRadio value={'DECIBELS'}>节奏(分贝)</NRadio>
- </NRadioGroup>
- </NFormItemGi>
- </NGrid>
- <NAlert showIcon={false} style={{marginBottom: "12px"}}>曲目上传</NAlert>
- <NGrid cols={2}>
- <NFormItemGi label="播放模式" path="playMode"
- rule={[
- {
- required: true,
- message: '请选择播放模式'
- }
- ]}
- >
- <NRadioGroup
- v-model:value={forms.playMode}
- onUpdateValue={(value: string | number | boolean) => {
- if (value === 'MP3') {
- forms.playMode = 'MP3'
- } else {
- forms.playMode = 'MIDI'
- }
- }}
- >
- <NRadio value="MP3">MP3</NRadio>
- <NRadio value="MIDI">MIDI</NRadio>
- </NRadioGroup>
- </NFormItemGi>
- {forms.playMode === 'MP3' && (
- <NFormItemGi
- label="伴奏类型"
- path="audioType"
- rule={[
- {
- required: true
- }
- ]}
- >
- <NRadioGroup
- v-model:value={forms.audioType}
- >
- <NRadio value={'HOMEMODE'}>自制伴奏</NRadio>
- <NRadio value={'COMMON'}>普通伴奏</NRadio>
- </NRadioGroup>
- </NFormItemGi>
- )}
- </NGrid>
- <NGrid cols={2}>
- {forms.playMode === 'MP3' && (
- <NFormItemGi
- label="上传伴奏"
- path="musicSheetAccompanimentList"
- rule={[
- {
- required: true,
- message: '请选择上传.mp3'
- }
- ]}
- >
- <UploadFile
- disabled={state.previewMode}
- size={10}
- v-model:imageList={state.musicSheetAccompanimentUrlList}
- tips="仅支持上传.mp3格式文件"
- listType="image"
- accept=".mp3"
- bucketName="cloud-coach"
- text="点击上传伴奏文件"
- max={10}
- onUpload:success={(file) => {
- state.musicSheetAccompanimentUrls = [state.musicSheetAccompanimentUrls, file.url].filter(Boolean).join(',')
- state.musicSheetAccompanimentUrlList = state.musicSheetAccompanimentUrls?.split(',').filter(Boolean)
- forms.musicSheetAccompanimentList = []
- for (let i = 0; i < state.musicSheetAccompanimentUrlList.length; i++) {
- forms.musicSheetAccompanimentList.push({
- audioFileUrl: state.musicSheetAccompanimentUrlList[i],
- track: file.name,
- sortNumber: i + 1
- })
- }
- }}
- onRemove={() => {
- state.musicSheetAccompanimentUrlList = []
- state.musicSheetAccompanimentUrls = ''
- }}
- // onReadFileInputEventAsArrayBuffer={readFileInputEventAsArrayBuffer}
- multiple={true}
- />
- </NFormItemGi>
- )}
- {forms.playMode === 'MIDI' && (
- <NFormItemGi
- label="上传MIDI"
- path="midiFileUrl"
- rule={[
- {
- required: true,
- message: '请选择上传.MIDI格式文件'
- }
- ]}
- >
- <UploadFile
- disabled={state.previewMode}
- size={10}
- v-model:fileList={forms.midiFileUrl}
- tips="仅支持上传.MIDI格式文件"
- listType="image"
- accept=".mp3,.aac"
- bucketName="cloud-coach"
- text="点击上传MIDI文件"
- // onReadFileInputEventAsArrayBuffer={readFileInputEventAsArrayBuffer}
- />
- </NFormItemGi>
- )}
- <NFormItemGi
- label="上传XML"
- path="xmlFileUrl"
- rule={[
- {
- required: true,
- message: '请选择上传XML'
- }
- ]}
- >
- <UploadFile
- disabled={state.previewMode}
- size={10}
- v-model:fileList={forms.xmlFileUrl}
- tips="仅支持上传.xml格式文件"
- listType="image"
- accept=".xml"
- bucketName="cloud-coach"
- text="点击上传XML文件"
- onReadFileInputEventAsArrayBuffer={readFileInputEventAsArrayBuffer}
- onRemove={() => {
- forms.multiTracksSelection = []
- state.partListNames = []
- forms.musicSheetSoundList = []
- }}
- />
- </NFormItemGi>
- </NGrid>
- <NGrid cols={2}>
- <NFormItemGi label="可用声部" path="subjectIdList"
- rule={[
- {
- required: true,
- message: '请选择可用声部'
- }
- ]}
- >
- <NSelect
- v-model:value={forms.subjectIdList}
- options={props.subjectList}
- multiple
- filterable
- clearable
- placeholder="请选择可用声部"
- />
- </NFormItemGi>
- <NFormItemGi label="可用乐器" path="musicalInstrumentIdList"
- rule={[
- {
- required: true,
- message: '请选择可用乐器'
- }
- ]}
- >
- <NSelect
- placeholder="请选择可用乐器"
- options={state.instrumentList}
- v-model:value={forms.musicalInstrumentIdList}
- clearable
- multiple
- maxTagCount={2}
- />
- </NFormItemGi>
- </NGrid>
- {(forms.musicSheetType) && (
- <NGrid cols={1}>
- <NFormItemGi
- label={`${forms.musicSheetType === 'SINGLE' ? '页面渲染声轨' : '用户可切换声轨'}`}
- path="multiTracksSelection"
- rule={[
- {
- required: true,
- message: `请选择${forms.musicSheetType === 'SINGLE' ? '页面渲染声轨' : '用户可切换声轨'}`
- }
- ]}
- >
- <NGrid style="padding-top: 4px;">
- <NGi span={24}>
- <NRadioGroup
- v-model:value={state.multiTracks}
- onUpdateValue={(value) => {
- checkMultiTracks(value)
- }}
- >
- <NRadio value={'all'}>全选</NRadio>
- <NRadio value={'allUncheck'}>重置</NRadio>
- <NRadio value={'invert'}>反选</NRadio>
- </NRadioGroup>
- </NGi>
- <NGi span={24} style={"margin-top:5px"}><NFormItemGi
- label=''
- path="multiTracksSelection"
- rule={[
- {
- required: false,
- }
- ]}
- >
- <NCheckboxGroup
- v-model:value={forms.multiTracksSelection}
- >
- <NGrid yGap={2} cols={4}
- >
- {state.partListNames.map((item: any, index: number) => (
- <NGi>
- <NCheckbox value={item.value} label={item.label} onUpdateChecked={() => {
- console.log("forms.multiTracksSelection", forms.multiTracksSelection)
- console.log("forms.musicSheetSoundList", forms.musicSheetSoundList)
- }}/>
- </NGi>
- ))}
- </NGrid>
- </NCheckboxGroup>
- </NFormItemGi></NGi>
- </NGrid>
- </NFormItemGi>
- </NGrid>
- )
- }
- <NGrid cols={2}>
- <NFormItemGi label="是否播放节拍器" path="isPlayBeat"
- rule={[
- {
- required: true,
- message: '请选择是否播放节拍器'
- }
- ]}
- >
- <NRadioGroup
- v-model:value={forms.isPlayBeat}
- >
- <NRadio value={true}>是</NRadio>
- <NRadio value={false}>否</NRadio>
- </NRadioGroup>
- </NFormItemGi>
- {forms.isPlayBeat && (
- <NFormItemGi label="播放方式" path="audioType"
- rule={[
- {
- required: true,
- message: '请选择播放方式'
- }
- ]}
- >
- <NRadioGroup
- v-model:value={forms.isUseSystemBeat}
- >
- <NRadio value={true}>系统节拍器</NRadio>
- <NRadio value={false}>MP3节拍器</NRadio>
- </NRadioGroup>
- </NFormItemGi>
- )}
- </NGrid>
- {/* 只有播放类型为mp3时才会有原音 */}
- {forms.playMode === 'MP3' && forms.musicSheetSoundList.length > 0 && (
- <>
- {forms.musicSheetSoundList.map((item: any, index: number) => (
- <>
- {item.track?.toLocaleUpperCase?.() != 'COMMON' && forms.multiTracksSelection.indexOf(item.track) > -1 &&
- <NGrid class={styles.audioSection}
- // v-show={forms.multiTracksSelection.indexOf(item.track) > -1}
- >
- <NFormItemGi
- span={12}
- label="原音"
- path={`musicSheetSoundList[${index}].audioFileUrl`}
- rule={[
- {
- // required: forms.multiTracksSelection.indexOf(forms.musicSheetSoundList[index].audioFileUrl) > -1,
- required: true,
- message: `请上传${
- item.track ? item.track + '的' : '第' + (index + 1) + '个'
- }原音`
- }
- ]}
- >
- <UploadFile
- disabled={state.previewMode}
- size={10}
- v-model:fileList={item.audioFileUrl}
- tips="仅支持上传.mp3/.aac格式文件"
- listType="image"
- accept=".mp3,.aac"
- bucketName="cloud-coach"
- />
- </NFormItemGi>
- {state.partListNames.length > 1 && (
- <NFormItemGi
- span={12}
- label="所属轨道"
- path={`musicSheetSoundList[${index}].track`}
- rule={[
- {
- required: true,
- message: '请选择所属轨道'
- }
- ]}
- >
- <NSelect
- placeholder="请选择所属轨道"
- v-model:value={item.track}
- options={initPartsListStatus(item.track)}
- />
- </NFormItemGi>
- )}
- </NGrid>}
- </>
- ))}
- <NButton
- type="primary"
- dashed
- block
- disabled={state.partListNames.length <= forms.musicSheetSoundList.length}
- style={{
- marginBottom: '24px'
- }}
- onClick={createSys}
- >
- 添加原音
- </NButton>
- </>
- )}
- </NForm>
- {props.type !== 'preview' &&
- (
- <NSpace justify="end" style="padding-top:12px">
- <NButton type="default" onClick={() => emit('close')}>
- 取消
- </NButton>
- <NButton
- type="primary"
- onClick={() => onSubmit()}
- loading={btnLoading.value}
- disabled={btnLoading.value}
- >
- 确认
- </NButton>
- </NSpace>
- )}
- <NModal
- v-model:show={state.showMusicSheetOwnerDialog}
- preset="dialog"
- showIcon={false}
- maskClosable={false}
- title="所属人"
- style={{width: '800px'}}
- >
- <MusicSheetOwnerDialog
- sourceType={forms.sourceType}
- onClose={() => {
- state.showMusicSheetOwnerDialog = false
- }}
- onChoseMusicSheetOwnerData={(musicSheetOwnerData) => {
- forms.musicSheetExtend = {
- ...musicSheetOwnerData
- }
- console.log(forms.musicSheetExtend)
- }}
- />
- </NModal>
- </div>
- )
- }
- })
|