<p>All timecodes and durations are stored as <code>int64_t</code> variables with nanosecond precision.</p>
<h1><spanclass="header-section-number">4</span> Outputting messages to the user</h1>
<p>There are basically four functions that output stuff: <code>mxinfo()</code>, <code>mxverb()</code>, <code>mxwarn()</code> and <code>mxerror()</code>. Each takes either a string or an argument produced by <code>boost::format()</code> (see <ahref="http://www.boost.org/doc/libs/1_47_0/libs/format/doc/format.html">the Boost::Format documentation</a> if you're not familiar with it), and <code>mxverb()</code> also takes a verbosity level argument.</p>
<p><code>mxinfo()</code> is supposed to be used for messages that are always output. Its messages must be translatable by using the <code>_()</code> macro, Example:</p>
<p>The same applies to <code>mxwarn()</code> and <code>mxerror()</code>. However, both prefix their messages with "Warning:" and "Error:" respectively. Also they modify mkvmerge's exit code -- "1" if <code>mxwarn()</code> has been used at least once and "2" if <code>mxerror()</code> is used. <code>mxerror()</code> also instantly exits the program. The GUI shows such warnings and errors in separate areas as to draw the user's attention to them. Therefore <code>mxwarn()</code> should be used for stuff the user really should know, and <code>mxerror()</code> for stuff that prevents further processing altogether.</p>
<p><code>mxverb()</code> is a debugging facility. mkvmerge starts out at verbosity level 1. Therefore <code>mxverb(1, ...)</code> is equivalent to <code>mxinfo(...)</code>. You can raise the verbosity level by adding the "-v" or "--verbose" command line parameters multiple times. Messages output with <code>mxverb()</code> should not be translatable, meaning you shouldn't use the <code>_()</code> macro.</p>
<p>Most classes are nameed <code>*_c</code>, and the corresponding counted pointer implementation is called <code>*_cptr</code>. A counted pointer is a reference counter implementation that automatically deletes its instance once the counter reaches zero.</p>
<p>Base class for input/output. Several derived classes implement access to binary files (<code>mm_file_io_c</code>), text files (<code>mm_text_io_c</code>, including BOM handling), memory buffers (<code>mm_mem_io_c</code>) etc.</p>
<p>A file and track information storage class. It is created by the command line parser with the source file name and all the command line options the user has passed in.</p>
<p>The instance is then passed to the reader which stores a copy in the <code>m_ti</code> variable. The reader also passes that copy to each packetizer it creates which in turn stores its own copy (again in the <code>m_ti</code> variable).</p>
<p>A reader is a class that demuxes a certain file type. It has to parse the file content, create one packetizer for each track, report the progress etc. Each reader class is derived from <code>generic_reader_c</code> (from <code>src/merge/pr_generic.h</code>).</p>
<p>An example for a rather minimal implementation is the MP3 reader in <code>src/input/r_mp3.h</code>.</p>
<h2><spanclass="header-section-number">6.1</span> Constructor & virtual destructor</h2>
<p>The constructor usually only takes a <code>track_info_c</code> argument. A virtual destructor must be present, even if it is empty. Otherwise you'll get linker errors.</p>
<p>The constructor must open the file and parse all its headers so that all track information has been processed. This will most likely be split up into a separate function in the future.</p>
<h2><spanclass="header-section-number">6.2</span><code>read(generic_packetizer_c *ptzr, bool force = false)</code></h2>
<p>Requests that the reader reads data for a specific track. mkvmerge uses a pulling model: the core asks each packetizer to provide data. The packetizers in turn ask the reader they've been created from to read data by calling this function.</p>
<p>If the reader supports arbitrary access to track data (e.g. for AVI and MP4/MOV files) then the reader should only read data for the requested track in order not to consume too much memory. If the file format doesn't allow for direct access to that data then the reader can simply read the next packet regardless which track that packet belongs to. The packetizer will call the reader's <code>read()</code> function as often as necessary until it has enough data.</p>
<p>The reader must return <code>FILE_STATUS_MOREDATA</code> if more data is available and <code>FILE_STATUS_DONE</code> when the end has been reached.</p>
<p>Each data packet is stored in an instance of <code>packet_c</code>. If the source container provides a timecode then that timecode should be set in the packet as well.</p>
<p>Has to create a packetizer for the track with the specific ID. This ID is the same number that the user uses on the command line as well as the number used in the call to <code>id_result_track()</code> (see below).</p>
<p>This function has to verify that the user actually wants this track processed. This is checked with the <code>demuxing_requested()</code> function. Example from the MP3 reader (as the MP3 file format can only contain one track the ID is always 0; see the Matroska reader for more complex examples):</p>
<pre><code>// Check if the audio demuxer with ID 0 is requested. Also make
<p>A lot of packetizers expect their codec private data to be constructed completely by the reader. This often requires that the reader processes at least a few packets. For a rather complex case see the MPEG PS reader's handling of h264 tracks.</p>
<p>File identification. Has to call <code>id_result_container()</code> once for the container type.</p>
<p>For each supported track the function must call <code>id_result_track(...)</code> once.</p>
<p>Both the container and the track identification can contain extended information. For an extensive example see the Matroska reader's identification function in <code>src/input/r_matroska.cpp</code>.</p>
<p>These are general hints for anyone wishing to translate MKVToolNix.</p>
<p>There are basically three things that can be translated in the MKVToolNix package:</p>
<ol>
<li>The programs and their messages themselves. This is the most important part.</li>
<li>The manual pages.</li>
</ol>
<p>All three are translated by the <code>gettext</code> system. This system uses .po-files which map the original English version to the translated version. There are several (OpenSource) applications available that can be used to translate such files, e.g. <ahref="http://www.poedit.net/">poedit</a> if you're using Windows.</p>
<p>Alternatively you can use the <ahref="https://www.transifex.com/">Transifex</a> web service for translating MKVToolNix. Please apply to join the <ahref="https://www.transifex.com/moritz-bunkus/mkvtoolnix/dashboard/">MKVToolNix project</a> and drop me a line via <scripttype="text/javascript">
</script><noscript>email (moritz at bunkus dot org)</noscript>. I'll add you to the project as soon as possible.</p>
<p>Please ask the author for an up-to-date template of the untranslated strings before you start doing anything else. He will attach the untranslated translation file (xyz.po) for the programs themselves (the first point of the three mentioned above). Take a look at it and at the poedit software. If you feel up to the task then simply start translating and send the translated .po file back to the author. The author can notify you whenever phrases change or are added.</p>
<p>If you want to test your translations you can do that as well:</p>
<ol>
<li>Install mkvtoolnix.</li>
<li>Translate the xyz.po file.</li>
<li>Whenever you want to test something compile the xyz.po to a .mo file with your po editor application.</li>
<li>Replace one of the installed translation files (e.g. <code>mkvtoolnix\locale\de\mkvtoolnix.mo</code>) with your compiled file.</li>
<p>Please note that a lot of strings must not be translated, especially command line options and their values. If a value to an option is a fixed string then it must not be translated, but if it is a placeholder e.g. for a file name then you can of course translate the <code>filename</code> part. Example:</p>
<p>Here neither <code>--stereo-mode</code> nor <code>none</code> or <code>both</code> may be translated because mkvmerge expects all those in English. But in</p>
<pre><code>--attach-file filename</code></pre>
<p>you can of course translate <code>filename</code>.</p>
<p>Some other hints:</p>
<ul>
<li><code>TID</code> = track ID, a number</li>
<li><code>FID</code> = file ID, again a number</li>
<p>If you're unsure what the author is trying to say (the author is not a native English speaker) or what you can and what you must not translate just drop him a note.</p>
<p>You can send the author your work in progress whenever you feel like it -- you don't have to finish everything at once.</p>
<p>Once you have shown that you are comfortable with translating you can also get write access to the project's Git repository and push updates there yourself.</p>
<h1><spanclass="header-section-number">8</span> Adding new translations</h1>
<p>This is a TODO list for adding a new translation (a .po file) to MKVToolNix:</p>
<ul>
<li>Add new entry to translation<em>c::initialize</em>available_translations() in src/common/translation.cpp</li>
<li>Copy .po file to po/ sub-directory</li>
<li>Add entry in mkvtoolnix.spec</li>
<li>Add installation and removal entries in installer/mkvtoolnix.nsi</li>
<li>Add new language file to installer/translations/</li>
<li>Either re-run configure or add the new translation in build-config to TRANSLATIONS; afterwards verify the format strings with »rake translations:verify-format-strings«</li>