{
  "Selected_candidate": {
    "pr_number": 11384,
    "pr_title": "Update formal power series printing",
    "pr_body": "fixes #11102 \nping @asmeurer \n",
    "issue_id": 11102,
    "issue_title": "fps should print as a formal power series",
    "issue_body": "When I first used `fps`, I didn't realize it really was a formal power series as it claims to be, because it prints like a normal series (same as `series`)\n\n```\nIn [21]: fps(sin(x))\nOut[21]:\n     3     5\n    x     x     ⎛ 6⎞\nx - ── + ─── + O⎝x ⎠\n    6    120\n```\n\nBut if you look at the string form, you see\n\n```\nIn [22]: print(fps(sin(x)))\nFormalPowerSeries(sin(x), x, 0, 1, (SeqFormula(Piecewise(((-1/4)**(_k/2 - 1/2)/(RisingFactorial(3/2, _k/2 - 1/2)*factorial(_k/2 - 1/2)), Eq(Mod(_k, 2), 1)), (0, True)), (_k, 2, oo)), SeqFormula(x**_k, (_k, 0, oo)), x))\n```\n\nThat is, it really does represent it as the formula `Sum((-1)**n/factorial(2*n + 1)*x**n, (n, 0, oo))` (albiet, not simplified). It out to print it like this, so you can see that that's what it's working with.\n\nSide question: if you enter something it can't compute, it just returns the function\n\n```\nIn [25]: fps(tan(x))\nOut[25]: tan(x)\n```\n\nIs that intentional? It seems like it ought to raise an exception in that case. \n\n@leosartaj \n",
    "issue_closed_at": "2016-07-14T22:47:58Z",
    "base_commit": "496e776108957d8c049cbef49522cef4c1955e2f",
    "changes": [
      {
        "file": "sympy/printing/latex.py",
        "type": "function",
        "name": "_print_FourierSeries",
        "class_name": "LatexPrinter",
        "code": "def _print_FourierSeries(self, s):\n        return self._print_Add(s.truncate()) + self._print(' + \\ldots')"
      },
      {
        "file": "sympy/printing/pretty/pretty.py",
        "type": "function",
        "name": "_print_FourierSeries",
        "class_name": "PrettyPrinter",
        "code": "def _print_FourierSeries(self, s):\n        if self._use_unicode:\n            dots = u(\"\\N{HORIZONTAL ELLIPSIS}\")\n        else:\n            dots = '...'\n        return self._print_Add(s.truncate()) + self._print(dots)"
      }
    ]
  },
  "Justification": "Candidate D is the most helpful as it pertains to how series are printed, which bears relevance to how monomials are displayed in LaTeX. The current bug involves printing a polynomial correctly in LaTeX format and Candidate D addresses updates to the formal power series printing, which could share insights into the LaTeX representation of mathematical expressions. Both bugs involve discrepancies in string formatting and rendering of mathematical structures, making the lessons learned from this fix particularly useful. Additionally, there is similar involvement of the `latex.py` file, indicating a potential overlap in the underlying printing mechanisms.",
  "instance_id": "sympy__sympy-14317",
  "repo": "sympy/sympy",
  "created_at": "2018-02-24T10:05:10Z",
  "problem_statement": "LaTeX printer does not use the same order of monomials as pretty and str \nWhen printing a Poly, the str and pretty printers use the logical order of monomials, from highest to lowest degrees. But latex printer does not. \r\n```\r\n>>> var('a b c x')\r\n>>> p = Poly([a, 1, b, 2, c, 3], x)\r\n>>> p\r\nPoly(a*x**5 + x**4 + b*x**3 + 2*x**2 + c*x + 3, x, domain='ZZ[a,b,c]')\r\n>>> pretty(p)\r\n\"Poly(a*x**5 + x**4 + b*x**3 + 2*x**2 + c*x + 3, x, domain='ZZ[a,b,c]')\"\r\n>>> latex(p)\r\n'\\\\operatorname{Poly}{\\\\left( a x^{5} + b x^{3} + c x + x^{4} + 2 x^{2} + 3, x, domain=\\\\mathbb{Z}\\\\left[a, b, c\\\\right] \\\\right)}'\r\n```\n",
  "patch": "diff --git a/sympy/printing/latex.py b/sympy/printing/latex.py\n--- a/sympy/printing/latex.py\n+++ b/sympy/printing/latex.py\n@@ -1813,7 +1813,50 @@ def _print_PolynomialRingBase(self, expr):\n \n     def _print_Poly(self, poly):\n         cls = poly.__class__.__name__\n-        expr = self._print(poly.as_expr())\n+        terms = []\n+        for monom, coeff in poly.terms():\n+            s_monom = ''\n+            for i, exp in enumerate(monom):\n+                if exp > 0:\n+                    if exp == 1:\n+                        s_monom += self._print(poly.gens[i])\n+                    else:\n+                        s_monom += self._print(pow(poly.gens[i], exp))\n+\n+            if coeff.is_Add:\n+                if s_monom:\n+                    s_coeff = r\"\\left(%s\\right)\" % self._print(coeff)\n+                else:\n+                    s_coeff = self._print(coeff)\n+            else:\n+                if s_monom:\n+                    if coeff is S.One:\n+                        terms.extend(['+', s_monom])\n+                        continue\n+\n+                    if coeff is S.NegativeOne:\n+                        terms.extend(['-', s_monom])\n+                        continue\n+\n+                s_coeff = self._print(coeff)\n+\n+            if not s_monom:\n+                s_term = s_coeff\n+            else:\n+                s_term = s_coeff + \" \" + s_monom\n+\n+            if s_term.startswith('-'):\n+                terms.extend(['-', s_term[1:]])\n+            else:\n+                terms.extend(['+', s_term])\n+\n+        if terms[0] in ['-', '+']:\n+            modifier = terms.pop(0)\n+\n+            if modifier == '-':\n+                terms[0] = '-' + terms[0]\n+\n+        expr = ' '.join(terms)\n         gens = list(map(self._print, poly.gens))\n         domain = \"domain=%s\" % self._print(poly.get_domain())\n \n"
}