How to extract the Photo/Video component of a MVIMG?

前端 未结 4 823
遥遥无期
遥遥无期 2021-02-13 23:17

The Google Pixel 2 and probably other phones since have the capability to cover \"Motion Photos\". These are saved as MVIMG and comparatively big.

I’m looking for a way

相关标签:
4条回答
  • 2021-02-13 23:56

    The above suggestion using grep -F --byte-offset ... and dd does not work for me on macOS High Sierra as /usr/bin/grep outputs a wrong offset — I guess it yields the offset of the "line" which contains the word ftypmp4, i.e. the position of the previous LF character plus one. I might guess wrong, but anyway, this is my solution:

    for i in MVIMG*.jpg; do \
        perl -0777 -ne 's/^.*(....ftypmp4.*)$/$1/s && print' "$i" >"${i%.jpg}.mp4"; \
    done
    

    This uses the ability of perl to slurp in a whole file at once and treat it as one big string. If no ftypmp4 with at least four leading bytes is present, an empty file is created, if multiple are present, the last one is extracted.

    Similarly, to remove the video from all the files:

    for i in MVIMG*.jpg; do \
        perl -0777 -pi -e 's/^(.*?)....ftypmp4.*$/$1/s' "$i"; \
    done
    

    This uses the in-place editing feature of perl. Everything after the first occurrence of ftypmp4 with its four leading bytes is cut off. If there are no occurrences, the file is rewritten with its contents unchanged.

    (One might or might not need to set PERLIO=raw in the environment and/or unset the locale related variables to avoid UTF-8 interpretation, which could fail for binary files which happen to include byte sequences that violate the UTF-8 composition rules. In my tests with various MVIMG files no such problems occurred though.)

    0 讨论(0)
  • 2021-02-14 00:07

    I did find https://github.com/cliveontoast/GoMoPho which scans for the mp4 header and then dumps the video.

    We can do the same, scanning for ftypmp4 from the MP4 header (actual file starts 4 bytes earlier):

    Thus to extract videos:

    for i in MVIMG*.jpg; do \
      ofs=$(grep -F --byte-offset --only-matching --text ftypmp4 "$i"); \
      ofs=${ofs%:*}; \
      [[ $ofs ]] && dd "if=$i" "of=${i%.jpg}.mp4" bs=$((ofs-4)) skip=1; \
    done
    

    And to remove videos:

    for i in MVIMG*.jpg; do \
      ofs=$(grep -F --byte-offset --only-matching --text ftypmp4 "$i"); \
      ofs=${ofs%:*}; \
      [[ $ofs ]] && truncate -s $((ofs-4)) "$i"; \
    done
    
    0 讨论(0)
  • 2021-02-14 00:11

    The non-perl shell scripts at the top of the post worked on my Linux system. I merged them into a single shell script that preserves the input file (like MVIMG_20191216_153039.jpg) and creates two output files (like IMG_20191216_153039.jpg and IMG_20191216_153039.mp4):

    #!/bin/bash
    # extract-mvimg: Extract .mp4 video and .jpg still image from a Pixel phone
    # camera "motion video" file with a name like MVIMG_20191216_153039.jpg
    # to make files like IMG_20191216_153039.jpg and IMG_20191216_153039.mp4
    #
    # Usage: extract-mvimg MVIMG*.jpg [MVIMG*.jpg...]
    
    for srcfile
    do
      case "$srcfile" in
      MVIMG_*_*.jpg) ;;
      *)
        echo "extract-mvimg: skipping '$srcfile': not an MVIMG*.jpg file?" 2>&1
        continue
        ;;
      esac
    
      # Get base filename: strip leading MV and trailing .jpg
      # Example: MVIMG_20191216_153039.jpg becomes IMG_20191216_153039
      basefile=${srcfile#MV}
      basefile=${basefile%.jpg}
    
      # Get byte offset. Example output: 2983617:ftypmp4
      offset=$(grep -F --byte-offset --only-matching --text ftypmp4 "$srcfile")
      # Strip trailing text. Example output: 2983617
      offset=${offset%:*}
    
      # If $offset isn't an empty string, create .mp4 file and
      # truncate a copy of input file to make .jpg file.
      if [[ $offset ]]
      then
        dd status=none "if=$srcfile" "of=${basefile}.mp4" bs=$((offset-4)) skip=1
        cp -ip "$srcfile" "${basefile}.jpg" || exit 1
        truncate -s $((offset-4)) "${basefile}.jpg"
      else
        echo "extract-mvimg: can't find ftypmp4 in $srcfile; skipping..." 2>&1
      fi
    done
    

    The status=none suppresses the "1+1 records in" and "1+1 records out" status output from dd. If your dd doesn't understand, you can remove that.

    0 讨论(0)
  • 2021-02-14 00:15

    The EXIF tag is useful, but the offset is with the respect to the end of the file. The mp4 file is embedded at:

    [file_size-micro_video_offset, file_size)
    

    For example:

    $ exiftool -xmp:all MVIMG_123.jpg
    XMP Toolkit                     : Adobe XMP Core 5.1.0-jc003
    Micro Video                     : 1
    Micro Video Version             : 1
    Micro Video Offset              : 2107172
    Micro Video Presentation Timestamp Us: 966280
    $ python -c 'import os; print os.path.getsize("MVIMG_123.jpg") - 2107172'
    3322791
    $ dd if=MVIMG_123.jpg of=video.mp4 bs=3322791 skip=1
    $ file video.mp4 
    video.mp4: ISO Media, MP4 v2 [ISO 14496-14]
    
    0 讨论(0)
提交回复
热议问题