]> git.lyx.org Git - lyx.git/blobdiff - src/insets/InsetListings.cpp
de.po
[lyx.git] / src / insets / InsetListings.cpp
index bec33b12cb44ee86ce28d927a1757a21a34cb99d..b50e92623e4489181489cbf73f23737aa85be4f5 100644 (file)
@@ -55,6 +55,7 @@ namespace lyx {
 InsetListings::InsetListings(Buffer * buf, InsetListingsParams const & par)
        : InsetCaptionable(buf,"listing")
 {
+       params_.setMinted(buffer().params().use_minted);
        status_ = par.status();
 }
 
@@ -71,6 +72,15 @@ Inset::DisplayType InsetListings::display() const
 }
 
 
+docstring InsetListings::layoutName() const
+{
+       if (buffer().params().use_minted)
+               return from_ascii("MintedListings");
+       else
+               return from_ascii("Listings");
+}
+
+
 void InsetListings::write(ostream & os) const
 {
        os << "listings" << "\n";
@@ -109,12 +119,72 @@ void InsetListings::read(Lexer & lex)
 }
 
 
+Encoding const * InsetListings::forcedEncoding(Encoding const * inner_enc,
+                                              Encoding const * outer_enc) const
+{
+       // The listings package cannot deal with multi-byte-encoded
+       // glyphs, except for Xe/LuaTeX (with non-TeX fonts) or pLaTeX.
+       // Minted can deal with all encodings.
+       if (buffer().params().use_minted
+               || inner_enc->name() == "utf8-plain"
+               || (buffer().params().encoding().package() == Encoding::japanese
+                       && inner_enc->package() == Encoding::japanese)
+               || inner_enc->hasFixedWidth())
+               return 0;
+
+       // We try if there's a singlebyte encoding for the outer
+       // language; if not, fall back to latin1.
+       // Power-users can set inputenc to utf8-plain to bypass this workaround
+       // and provide alternatives in the user-preamble.
+       return (outer_enc->hasFixedWidth()) ?
+                       outer_enc : encodings.fromLyXName("iso8859-1");
+}
+
+
 void InsetListings::latex(otexstream & os, OutputParams const & runparams) const
 {
        string param_string = params().params();
        // NOTE: I use {} to quote text, which is an experimental feature
        // of the listings package (see page 25 of the manual)
        bool const isInline = params().isInline();
+       bool const use_minted = buffer().params().use_minted;
+       string minted_language;
+       string float_placement;
+       bool const isfloat = params().isFloat();
+       if (use_minted && (isfloat || contains(param_string, "language="))) {
+               // Get float placement and/or language of the code,
+               // then remove the relative options.
+               vector<string> opts =
+                       getVectorFromString(param_string, ",", false);
+               for (size_t i = 0; i < opts.size(); ++i) {
+                       if (prefixIs(opts[i], "float")) {
+                               if (prefixIs(opts[i], "float="))
+                                       float_placement = opts[i].substr(6);
+                               opts.erase(opts.begin() + i--);
+                       }
+                       else if (prefixIs(opts[i], "language=")) {
+                               minted_language = opts[i].substr(9);
+                               opts.erase(opts.begin() + i--);
+                       }
+               }
+               param_string = getStringFromVector(opts, ",");
+       }
+       // Minted needs a language specification
+       if (minted_language.empty()) {
+               // If a language has been set globally, use that,
+               // otherwise use TeX by default
+               string const & blp = buffer().params().listings_params;
+               size_t start = blp.find("language=");
+               if (start != string::npos) {
+                       start += strlen("language=");
+                       size_t len = blp.find(",", start);
+                       if (len != string::npos)
+                               len -= start;
+                       minted_language = blp.substr(start, len);
+               } else
+                       minted_language = "TeX";
+       }
+
        // get the paragraphs. We can not output them directly to given odocstream
        // because we can not yet determine the delimiter character of \lstinline
        docstring code;
@@ -124,35 +194,30 @@ void InsetListings::latex(otexstream & os, OutputParams const & runparams) const
 
        bool encoding_switched = false;
        Encoding const * const save_enc = runparams.encoding;
-       // The listings package cannot deal with multi-byte-encoded
-       // glyphs, except if full-unicode aware backends
-       // such as XeTeX or LuaTeX are used, and with pLaTeX.
-       bool const multibyte_possible = runparams.isFullUnicode()
-           || (buffer().params().encoding().package() == Encoding::japanese
-               && runparams.encoding->package() == Encoding::japanese);
 
-       if (!multibyte_possible && !runparams.encoding->hasFixedWidth()) {
+       Encoding const * const outer_encoding =
+               (runparams.local_font != 0) ?
+                       runparams.local_font->language()->encoding()
+                       : buffer().params().language->encoding();
+       Encoding const * fixedlstenc = forcedEncoding(runparams.encoding, outer_encoding);
+       if (fixedlstenc) {
                // We need to switch to a singlebyte encoding, due to
                // the restrictions of the listings package (see above).
                // This needs to be consistent with
                // LaTeXFeatures::getTClassI18nPreamble().
-               Language const * const outer_language =
-                       (runparams.local_font != 0) ?
-                               runparams.local_font->language()
-                               : buffer().params().language;
-               // We try if there's a singlebyte encoding for the current
-               // language; if not, fall back to latin1.
-               Encoding const * const lstenc =
-                       (outer_language->encoding()->hasFixedWidth()) ?
-                               outer_language->encoding()
-                               : encodings.fromLyXName("iso8859-1");
-               switchEncoding(os.os(), buffer().params(), runparams, *lstenc, true);
-               runparams.encoding = lstenc;
+               // We need to put this into a group in order to prevent encoding leaks
+               // (happens with cprotect).
+               os << "\\bgroup";
+               switchEncoding(os.os(), buffer().params(), runparams, *fixedlstenc, true);
+               runparams.encoding = fixedlstenc;
                encoding_switched = true;
        }
 
+       bool const captionfirst = !isfloat && par->isInset(0)
+                               && par->getInset(0)->lyxCode() == CAPTION_CODE;
+
        while (par != end) {
-               pos_type siz = par->size();
+               pos_type const siz = par->size();
                bool captionline = false;
                for (pos_type i = 0; i < siz; ++i) {
                        if (i == 0 && par->isInset(i) && i + 1 == siz)
@@ -225,13 +290,46 @@ void InsetListings::latex(otexstream & os, OutputParams const & runparams) const
                        }
                }
                docstring const delim(1, delimiters[pos]);
-               os << "\\lstinline";
-               if (!param_string.empty())
-                       os << "[" << from_utf8(param_string) << "]";
-               else if (pos >= delimiters.find('Q'))
-                       // We need to terminate the command before the delimiter
-                       os << " ";
+               if (use_minted) {
+                       os << "\\mintinline";
+                       if (!param_string.empty())
+                               os << "[" << from_utf8(param_string) << "]";
+                       os << "{" << ascii_lowercase(minted_language) << "}";
+               } else {
+                       os << "\\lstinline";
+                       if (!param_string.empty())
+                               os << "[" << from_utf8(param_string) << "]";
+                       else if (pos >= delimiters.find('Q'))
+                               // We need to terminate the command before
+                               // the delimiter
+                               os << " ";
+               }
                os << delim << code << delim;
+       } else if (use_minted) {
+               OutputParams rp = runparams;
+               rp.moving_arg = true;
+               TexString caption = getCaption(rp);
+               if (isfloat) {
+                       os << breakln << "\\begin{listing}";
+                       if (!float_placement.empty())
+                               os << '[' << float_placement << "]";
+               } else if (captionfirst && !caption.str.empty()) {
+                       os << breakln << "\\lyxmintcaption[t]{"
+                          << move(caption) << "}\n";
+               }
+               os << breakln << "\\begin{minted}";
+               if (!param_string.empty())
+                       os << "[" << param_string << "]";
+               os << "{" << ascii_lowercase(minted_language) << "}\n"
+                  << code << breakln << "\\end{minted}\n";
+               if (isfloat) {
+                       if (!caption.str.empty())
+                               os << "\\caption{" << move(caption) << "}\n";
+                       os << "\\end{listing}\n";
+               } else if (!captionfirst && !caption.str.empty()) {
+                       os << breakln << "\\lyxmintcaption[b]{"
+                          << move(caption) << "}";
+               }
        } else {
                OutputParams rp = runparams;
                rp.moving_arg = true;
@@ -255,14 +353,16 @@ void InsetListings::latex(otexstream & os, OutputParams const & runparams) const
 
        if (encoding_switched){
                // Switch back
-               switchEncoding(os.os(), buffer().params(), runparams, *save_enc, true);
+               switchEncoding(os.os(), buffer().params(),
+                              runparams, *save_enc, true, true);
+               os << "\\egroup" << breakln;
                runparams.encoding = save_enc;
        }
 
        if (!uncodable.empty() && !runparams.silent) {
                // issue a warning about omitted characters
                // FIXME: should be passed to the error dialog
-               if (!multibyte_possible && !runparams.encoding->hasFixedWidth())
+               if (fixedlstenc)
                        frontend::Alert::warning(_("Uncodable characters in listings inset"),
                                bformat(_("The following characters in one of the program listings are\n"
                                          "not representable in the current encoding and have been omitted:\n%1$s.\n"
@@ -369,6 +469,7 @@ bool InsetListings::getStatus(Cursor & cur, FuncRequest const & cmd,
                                return true;
                        }
                }
+               // fall through
                default:
                        return InsetCaptionable::getStatus(cur, cmd, status);
        }
@@ -387,11 +488,18 @@ docstring const InsetListings::buttonLabel(BufferView const & bv) const
 
 void InsetListings::validate(LaTeXFeatures & features) const
 {
-       features.require("listings");
        features.useInsetLayout(getLayout());
        string param_string = params().params();
-       if (param_string.find("\\color") != string::npos)
-               features.require("color");
+       if (buffer().params().use_minted) {
+               features.require("minted");
+               OutputParams rp = features.runparams();
+               if (!params().isFloat() && !getCaption(rp).str.empty())
+                       features.require("lyxmintcaption");
+       } else {
+               features.require("listings");
+               if (contains(param_string, "\\color"))
+                       features.require("color");
+       }
        InsetCaptionable::validate(features);
 }
 
@@ -420,7 +528,8 @@ TexString InsetListings::getCaption(OutputParams const & runparams) const
        // the caption may contain \label{} but the listings
        // package prefer caption={}, label={}
        TexString cap = os.release();
-       if (!contains(cap.str, from_ascii("\\label{")))
+       if (buffer().params().use_minted
+           || !contains(cap.str, from_ascii("\\label{")))
                return cap;
        // convert from
        //     blah1\label{blah2} blah3
@@ -432,10 +541,17 @@ TexString InsetListings::getCaption(OutputParams const & runparams) const
        // NOTE that } is not allowed in blah2.
        regex const reg("(.*)\\\\label\\{(.*?)\\}(.*)");
        string const new_cap("$1$3},label={$2");
+       // Remove potential \protect'ion of \label.
+       docstring capstr = subst(cap.str, from_ascii("\\protect\\label"),
+                                from_ascii("\\label"));
        // TexString validity: the substitution preserves the number of newlines.
        // Moreover we assume that $2 does not contain newlines, so that the texrow
        // information remains accurate.
-       cap.str = from_utf8(regex_replace(to_utf8(cap.str), reg, new_cap));
+       // Replace '\n' with an improbable character from Private Use Area-A
+       // and then return to '\n' after the regex replacement.
+       capstr = subst(capstr, char_type('\n'), 0xffffd);
+       cap.str = subst(from_utf8(regex_replace(to_utf8(capstr), reg, new_cap)),
+                       0xffffd, char_type('\n'));
        return cap;
 }