feat(xgplayer-transmuxer): MP4格式支持解析matrix、rotation

This commit is contained in:
liujing.cyan 2024-06-17 20:22:39 +08:00 committed by gemstone
parent 685f04fd3f
commit fb7fbb73e2
2 changed files with 46 additions and 4 deletions

View File

@ -1,5 +1,5 @@
import { AudioCodecType, VideoCodecType } from '../model' import { AudioCodecType, VideoCodecType } from '../model'
import { getAvcCodec, readBig16, readBig24, readBig32, readBig64 } from '../utils' import { getAvcCodec, readBig16, readBig24, readBig32, readBig64, combineToFloat, toDegree } from '../utils'
import { AAC } from '../codec' import { AAC } from '../codec'
export class MP4Parser { export class MP4Parser {
static findBox (data, names, start = 0) { static findBox (data, names, start = 0) {
@ -126,15 +126,26 @@ export class MP4Parser {
return parseBox(box, true, (ret, data) => { return parseBox(box, true, (ret, data) => {
let start = 0 let start = 0
if (ret.version === 1) { if (ret.version === 1) {
ret.createTime = readBig64(data, 0)
ret.modifyTime = readBig64(data, 8)
ret.timescale = readBig32(data, 16) ret.timescale = readBig32(data, 16)
ret.duration = readBig64(data, 20) ret.duration = readBig64(data, 20)
start += 28 start += 28
} else { } else {
ret.createTime = readBig32(data, 0)
ret.modifyTime = readBig32(data, 4)
ret.timescale = readBig32(data, 8) ret.timescale = readBig32(data, 8)
ret.duration = readBig32(data, 12) ret.duration = readBig32(data, 12)
start += 16 start += 16
} }
ret.nextTrackId = readBig32(data, start + 76) ret.rate = combineToFloat(readBig16(data, start), readBig16(data, start + 2))
start += 4
ret.volume = combineToFloat(data[start], data[start + 1])
start += 2
start += 10 // skip
start += 36 // skip matrix, Because the rotate of the mp4 video track is determined by the matrix in tkhd box
start += 24 // skip
ret.nextTrackId = readBig32(data, start)
}) })
} }
@ -157,8 +168,26 @@ export class MP4Parser {
ret.duration = readBig32(data, 16) ret.duration = readBig32(data, 16)
start += 20 start += 20
} }
ret.width = readBig32(data, start + 52) start += 8 // skip
ret.height = readBig32(data, start + 56) ret.layer = readBig16(data, start)
start += 2
ret.alternateGroup = readBig16(data, start)
start += 2
start += 4 // skip
ret.matrix = [] // for remux
for (let i = 0; i < 36; i++) {
ret.matrix.push(data[start + i])
}
const caculatedMatrix = [] // for caculation of rotation
for (let i = 0; i < 9; i++) {
caculatedMatrix.push(combineToFloat(readBig16(data, start + i * 2), readBig16(data, start + i * 2 + 2))) // 16.16 fixed point
caculatedMatrix.push(combineToFloat(readBig16(data, start + i * 2 + 4), readBig16(data, start + i * 2 + 6))) // 16.16 fixed point
caculatedMatrix.push(combineToFloat(data[start + i * 2 + 8] & 0x3, readBig32(data, start + i * 2 + 8) >>> 2)) // 2.30 fixed point
}
start += 36
ret.rotation = toDegree(caculatedMatrix)
ret.width = readBig32(data, start) // 16.16 fixed point, no parsed
ret.height = readBig32(data, start + 4) // 16.16 fixed point, no parsed
}) })
} }
@ -767,6 +796,8 @@ export class MP4Parser {
v.mvhdTimecale = moov.mvhd.timescale v.mvhdTimecale = moov.mvhd.timescale
v.timescale = v.formatTimescale = vTrack.mdia.mdhd.timescale v.timescale = v.formatTimescale = vTrack.mdia.mdhd.timescale
v.duration = vTrack.mdia.mdhd.duration || (v.mvhdDurtion / v.mvhdTimecale * v.timescale) v.duration = vTrack.mdia.mdhd.duration || (v.mvhdDurtion / v.mvhdTimecale * v.timescale)
v.rotation = vTrack.tkhd.rotation
v.matrix = vTrack.tkhd.matrix
const e1 = vTrack.mdia.minf.stbl.stsd.entries[0] const e1 = vTrack.mdia.minf.stbl.stsd.entries[0]
v.width = e1.width v.width = e1.width
v.height = e1.height v.height = e1.height

View File

@ -87,3 +87,14 @@ export function hashVal (str) {
} }
return hash return hash
} }
export function combineToFloat (integer, decimal) {
return Number(integer + '.' + decimal)
}
export function toDegree (matrix) {
if (matrix.length < 5)
return 0
const scaled0 = Math.hypot(matrix[0], matrix[3]), scaled1 = Math.hypot(matrix[1], matrix[4])
return 0 === scaled0 || 0 === scaled1 ? 0 : 180 * Math.atan2(matrix[1] / scaled1, matrix[0] / scaled0) / Math.PI
}