{
  "instance_id": "sympy__sympy-23191",
  "repo": "sympy/sympy",
  "created_at": "2022-03-01T17:22:06Z",
  "problem_statement": "display bug while using pretty_print with sympy.vector object in the terminal\nThe following code jumbles some of the outputs in the terminal, essentially by inserting the unit vector in the middle -\r\n```python\r\nfrom sympy import *\r\nfrom sympy.vector import CoordSys3D, Del\r\n\r\ninit_printing()\r\n\r\ndelop = Del()\r\nCC_ = CoordSys3D(\"C\")\r\nx,    y,    z    = CC_.x, CC_.y, CC_.z\r\nxhat, yhat, zhat = CC_.i, CC_.j, CC_.k\r\n\r\nt = symbols(\"t\")\r\nten = symbols(\"10\", positive=True)\r\neps, mu = 4*pi*ten**(-11), ten**(-5)\r\n\r\nBx = 2 * ten**(-4) * cos(ten**5 * t) * sin(ten**(-3) * y)\r\nvecB = Bx * xhat\r\nvecE = (1/eps) * Integral(delop.cross(vecB/mu).doit(), t)\r\n\r\npprint(vecB)\r\nprint()\r\npprint(vecE)\r\nprint()\r\npprint(vecE.doit())\r\n```\r\n\r\nOutput:\r\n```python\r\n\u239b     \u239by_C\u239e    \u239b  5  \u239e\u239e    \r\n\u239c2\u22c5sin\u239c\u2500\u2500\u2500\u239f i_C\u22c5cos\u239d10 \u22c5t\u23a0\u239f\r\n\u239c     \u239c  3\u239f           \u239f    \r\n\u239c     \u239d10 \u23a0           \u239f    \r\n\u239c\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u239f    \r\n\u239c           4         \u239f    \r\n\u239d         10          \u23a0    \r\n\r\n\u239b     \u2320                           \u239e    \r\n\u239c     \u23ae       \u239by_C\u239e    \u239b  5  \u239e    \u239f k_C\r\n\u239c     \u23ae -2\u22c5cos\u239c\u2500\u2500\u2500\u239f\u22c5cos\u239d10 \u22c5t\u23a0    \u239f    \r\n\u239c     \u23ae       \u239c  3\u239f               \u239f    \r\n\u239c  11 \u23ae       \u239d10 \u23a0               \u239f    \r\n\u239c10  \u22c5\u23ae \u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500 dt\u239f    \r\n\u239c     \u23ae             2             \u239f    \r\n\u239c     \u23ae           10              \u239f    \r\n\u239c     \u2321                           \u239f    \r\n\u239c\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u239f    \r\n\u239d               4\u22c5\u03c0               \u23a0    \r\n\r\n\u239b   4    \u239b  5  \u239e    \u239by_C\u239e \u239e    \r\n\u239c-10 \u22c5sin\u239d10 \u22c5t\u23a0\u22c5cos\u239c\u2500\u2500\u2500\u239f k_C \u239f\r\n\u239c                   \u239c  3\u239f \u239f    \r\n\u239c                   \u239d10 \u23a0 \u239f    \r\n\u239c\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u2500\u239f    \r\n\u239d           2\u22c5\u03c0           \u23a0    ```\n",
  "patch": "diff --git a/sympy/printing/pretty/pretty.py b/sympy/printing/pretty/pretty.py\n--- a/sympy/printing/pretty/pretty.py\n+++ b/sympy/printing/pretty/pretty.py\n@@ -1144,22 +1144,24 @@ def _print_BasisDependent(self, expr):\n             if '\\n' in partstr:\n                 tempstr = partstr\n                 tempstr = tempstr.replace(vectstrs[i], '')\n-                if '\\N{right parenthesis extension}' in tempstr:   # If scalar is a fraction\n+                if '\\N{RIGHT PARENTHESIS EXTENSION}' in tempstr:   # If scalar is a fraction\n                     for paren in range(len(tempstr)):\n                         flag[i] = 1\n-                        if tempstr[paren] == '\\N{right parenthesis extension}':\n-                            tempstr = tempstr[:paren] + '\\N{right parenthesis extension}'\\\n+                        if tempstr[paren] == '\\N{RIGHT PARENTHESIS EXTENSION}' and tempstr[paren + 1] == '\\n':\n+                            # We want to place the vector string after all the right parentheses, because\n+                            # otherwise, the vector will be in the middle of the string\n+                            tempstr = tempstr[:paren] + '\\N{RIGHT PARENTHESIS EXTENSION}'\\\n                                          + ' '  + vectstrs[i] + tempstr[paren + 1:]\n                             break\n                 elif '\\N{RIGHT PARENTHESIS LOWER HOOK}' in tempstr:\n-                    flag[i] = 1\n-                    tempstr = tempstr.replace('\\N{RIGHT PARENTHESIS LOWER HOOK}',\n-                                        '\\N{RIGHT PARENTHESIS LOWER HOOK}'\n-                                        + ' ' + vectstrs[i])\n-                else:\n-                    tempstr = tempstr.replace('\\N{RIGHT PARENTHESIS UPPER HOOK}',\n-                                        '\\N{RIGHT PARENTHESIS UPPER HOOK}'\n-                                        + ' ' + vectstrs[i])\n+                    # We want to place the vector string after all the right parentheses, because\n+                    # otherwise, the vector will be in the middle of the string. For this reason,\n+                    # we insert the vector string at the rightmost index.\n+                    index = tempstr.rfind('\\N{RIGHT PARENTHESIS LOWER HOOK}')\n+                    if index != -1: # then this character was found in this string\n+                        flag[i] = 1\n+                        tempstr = tempstr[:index] + '\\N{RIGHT PARENTHESIS LOWER HOOK}'\\\n+                                     + ' '  + vectstrs[i] + tempstr[index + 1:]\n                 o1[i] = tempstr\n \n         o1 = [x.split('\\n') for x in o1]\n",
  "similar_bug_items": [
    {
      "pr_number": 11947,
      "pr_title": "aligned middle of matrix with baseline",
      "pr_body": "**Previous Output**: \r\n\r\n```\r\n>>> Z = Matrix(5,5, lambda i,j: i+j)\r\n>>> pprint(MatMul(2,Z))\r\n2\u22c5\u23a10  1  2  3  4\u23a4\r\n  \u23a2             \u23a5\r\n  \u23a21  2  3  4  5\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a22  3  4  5  6\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a23  4  5  6  7\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a34  5  6  7  8\u23a6\r\n```\r\n\r\n**Output after this PR**:\r\n```\r\n>>> Z = Matrix(5,5, lambda i,j: i+j)\r\n>>> pprint(MatMul(2,Z))\r\n  \u23a10  1  2  3  4\u23a4\r\n  \u23a2             \u23a5\r\n  \u23a21  2  3  4  5\u23a5\r\n  \u23a2             \u23a5\r\n2.\u23a22  3  4  5  6\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a23  4  5  6  7\u23a5\r\n  \u23a2             \u23a5\r\n  \u23a34  5  6  7  8\u23a6\r\n```\r\nFixes #10472 . ",
      "issue_id": 10472,
      "issue_title": "pprint should align the middle of the matrix to the baseline?",
      "issue_body": "Consider the current behaviour, where operators align with the top of the matrix:\n\n```\n>>> Z = Matrix(5,5, lambda i,j: i+j)\n>>> pprint(MatMul(2,Z))\n2\u22c5\u23a10  1  2  3  4\u23a4\n  \u23a2             \u23a5\n  \u23a21  2  3  4  5\u23a5\n  \u23a2             \u23a5\n  \u23a22  3  4  5  6\u23a5\n  \u23a2             \u23a5\n  \u23a23  4  5  6  7\u23a5\n  \u23a2             \u23a5\n  \u23a34  5  6  7  8\u23a6\n>>> pprint(MatAdd(Z,Z))\n\u23a10  1  2  3  4\u23a4 + \u23a10  1  2  3  4\u23a4\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a21  2  3  4  5\u23a5   \u23a21  2  3  4  5\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a22  3  4  5  6\u23a5   \u23a22  3  4  5  6\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a23  4  5  6  7\u23a5   \u23a23  4  5  6  7\u23a5\n\u23a2             \u23a5   \u23a2             \u23a5\n\u23a34  5  6  7  8\u23a6   \u23a34  5  6  7  8\u23a6\n```\n\nThis looks poor, but I figured it might be the desired behaviour...\n\nHere's the example that makes me think its wrong:\n\n```\n>>> pprint(sin((1+1/x)/(1+1/y)) + det(MatMul(2,Z)))\n   \u239b    1\u239e                       \n   \u239c1 + \u2500\u239f                       \n   \u239c    x\u239f                       \nsin\u239c\u2500\u2500\u2500\u2500\u2500\u239f + 32\u22c5\u2502\u23a10  1  2  3  4\u23a4\u2502\n   \u239c    1\u239f      \u2502\u23a2             \u23a5\u2502\n   \u239c1 + \u2500\u239f      \u2502\u23a21  2  3  4  5\u23a5\u2502\n   \u239d    y\u23a0      \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a22  3  4  5  6\u23a5\u2502\n                \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a23  4  5  6  7\u23a5\u2502\n                \u2502\u23a2             \u23a5\u2502\n                \u2502\u23a34  5  6  7  8\u23a6\u2502\n```\n\nHere's how I think it should work, using the proposed #10423:\n\n```\npprint(sin((1+1/x)/y) + Trace(Z))\n               \u239b\u23a10  1  2  3  4\u23a4\u239e\n   \u239b    1\u239e     \u239c\u23a2             \u23a5\u239f\n   \u239c1 + \u2500\u239f     \u239c\u23a21  2  3  4  5\u23a5\u239f\n   \u239c    x\u239f     \u239c\u23a2             \u23a5\u239f\nsin\u239c\u2500\u2500\u2500\u2500\u2500\u239f + tr\u239c\u23a22  3  4  5  6\u23a5\u239f\n   \u239d  y  \u23a0     \u239c\u23a2             \u23a5\u239f\n               \u239c\u23a23  4  5  6  7\u23a5\u239f\n               \u239c\u23a2             \u23a5\u239f\n               \u239d\u23a34  5  6  7  8\u23a6\u23a0\n```\n\nwhere note the `sin` alignment is by-design, and (relevant to this bug) the `+` and `tr` operator line up with `sin`.\n\nAt least, det, MatAdd, MatMul, MatPow would need fixed, maybe others I haven't thought of.\n",
      "issue_closed_at": "2016-12-22T06:52:11Z",
      "base_commit": "9a724a42c033c1aae97064947a0f44ec3b922d73",
      "changes": [
        {
          "file": "sympy/matrices/expressions/matadd.py",
          "type": "function",
          "name": "merge_explicit",
          "class_name": null,
          "code": "def merge_explicit(matadd):\n    \"\"\" Merge explicit MatrixBase arguments\n\n    >>> from sympy import MatrixSymbol, eye, Matrix, MatAdd, pprint\n    >>> from sympy.matrices.expressions.matadd import merge_explicit\n    >>> A = MatrixSymbol('A', 2, 2)\n    >>> B = eye(2)\n    >>> C = Matrix([[1, 2], [3, 4]])\n    >>> X = MatAdd(A, B, C)\n    >>> pprint(X)\n    A + [1  0] + [1  2]\n        [    ]   [    ]\n        [0  1]   [3  4]\n    >>> pprint(merge_explicit(X))\n    A + [2  2]\n        [    ]\n        [3  5]\n    \"\"\"\n    groups = sift(matadd.args, lambda arg: isinstance(arg, MatrixBase))\n    if len(groups[True]) > 1:\n        return MatAdd(*(groups[False] + [reduce(add, groups[True])]))\n    else:\n        return matadd"
        },
        {
          "file": "sympy/matrices/expressions/matmul.py",
          "type": "function",
          "name": "merge_explicit",
          "class_name": null,
          "code": "def merge_explicit(matmul):\n    \"\"\" Merge explicit MatrixBase arguments\n\n    >>> from sympy import MatrixSymbol, eye, Matrix, MatMul, pprint\n    >>> from sympy.matrices.expressions.matmul import merge_explicit\n    >>> A = MatrixSymbol('A', 2, 2)\n    >>> B = Matrix([[1, 1], [1, 1]])\n    >>> C = Matrix([[1, 2], [3, 4]])\n    >>> X = MatMul(A, B, C)\n    >>> pprint(X)\n    A*[1  1]*[1  2]\n      [    ] [    ]\n      [1  1] [3  4]\n    >>> pprint(merge_explicit(X))\n    A*[4  6]\n      [    ]\n      [4  6]\n\n    >>> X = MatMul(B, A, C)\n    >>> pprint(X)\n    [1  1]*A*[1  2]\n    [    ]   [    ]\n    [1  1]   [3  4]\n    >>> pprint(merge_explicit(X))\n    [1  1]*A*[1  2]\n    [    ]   [    ]\n    [1  1]   [3  4]\n    \"\"\"\n    if not any(isinstance(arg, MatrixBase) for arg in matmul.args):\n        return matmul\n    newargs = []\n    last = matmul.args[0]\n    for arg in matmul.args[1:]:\n        if isinstance(arg, (MatrixBase, Number)) and isinstance(last, (MatrixBase, Number)):\n            last = last * arg\n        else:\n            newargs.append(last)\n            last = arg\n    newargs.append(last)\n\n    return MatMul(*newargs)"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_matrix_contents",
          "class_name": "PrettyPrinter",
          "code": "def _print_matrix_contents(self, e):\n        \"\"\"\n        This method factors out what is essentially grid printing.\n        \"\"\"\n        M = e   # matrix\n        Ms = {}  # i,j -> pretty(M[i,j])\n        for i in range(M.rows):\n            for j in range(M.cols):\n                Ms[i, j] = self._print(M[i, j])\n\n        # h- and v- spacers\n        hsep = 2\n        vsep = 1\n\n        # max width for columns\n        maxw = [-1] * M.cols\n\n        for j in range(M.cols):\n            maxw[j] = max([Ms[i, j].width() for i in range(M.rows)] or [0])\n\n        # drawing result\n        D = None\n\n        for i in range(M.rows):\n\n            D_row = None\n            for j in range(M.cols):\n                s = Ms[i, j]\n\n                # reshape s to maxw\n                # XXX this should be generalized, and go to stringPict.reshape ?\n                assert s.width() <= maxw[j]\n\n                # hcenter it, +0.5 to the right                        2\n                # ( it's better to align formula starts for say 0 and r )\n                # XXX this is not good in all cases -- maybe introduce vbaseline?\n                wdelta = maxw[j] - s.width()\n                wleft = wdelta // 2\n                wright = wdelta - wleft\n\n                s = prettyForm(*s.right(' '*wright))\n                s = prettyForm(*s.left(' '*wleft))\n\n                # we don't need vcenter cells -- this is automatically done in\n                # a pretty way because when their baselines are taking into\n                # account in .right()\n\n                if D_row is None:\n                    D_row = s   # first box in a row\n                    continue\n\n                D_row = prettyForm(*D_row.right(' '*hsep))  # h-spacer\n                D_row = prettyForm(*D_row.right(s))\n\n            if D is None:\n                D = D_row       # first row in a picture\n                continue\n\n            # v-spacer\n            for _ in range(vsep):\n                D = prettyForm(*D.below(' '))\n\n            D = prettyForm(*D.below(D_row))\n\n        if D is None:\n            D = prettyForm('')  # Empty Matrix\n\n        return D"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_MatrixElement",
          "class_name": "PrettyPrinter",
          "code": "def _print_MatrixElement(self, expr):\n        from sympy.matrices import MatrixSymbol\n        from sympy import Symbol\n        if (isinstance(expr.parent, MatrixSymbol)\n                and expr.i.is_number and expr.j.is_number):\n            return self._print(\n                    Symbol(expr.parent.name + '_%d%d'%(expr.i, expr.j)))\n        else:\n            prettyFunc = self._print(expr.parent)\n            prettyIndices = self._print_seq((expr.i, expr.j), delimiter=', '\n                    ).parens(left='[', right=']')[0]\n            pform = prettyForm(binding=prettyForm.FUNC,\n                    *stringPict.next(prettyFunc, prettyIndices))\n\n            # store pform parts so it can be reassembled e.g. when powered\n            pform.prettyFunc = prettyFunc\n            pform.prettyArgs = prettyIndices\n\n            return pform"
        }
      ]
    },
    {
      "pr_number": 8627,
      "pr_title": "Bug Fix #8626",
      "pr_body": "Let me know if a test case is needed\n",
      "issue_id": 8626,
      "issue_title": "bug in physics/mechanics/lagrange.py",
      "issue_body": "When providing a `forcelist` with more than one force, not all terms are computed,\nthis comes from the fact that the iterator `flist = zip(*_f_list_parser(self.forcelist, N))` is consumed in the first iteration of the loop.\n\nI propose a following PR that fixes the bug, let me know if I need to also provide a test case.\n\nBest,\nGuillaume\n",
      "issue_closed_at": "2015-01-06T16:57:16Z",
      "base_commit": "65f7c8c2c9c1927eaa8520c4ce06864f93a20ad1",
      "changes": [
        {
          "file": "sympy/physics/mechanics/lagrange.py",
          "type": "function",
          "name": "form_lagranges_equations",
          "class_name": "LagrangesMethod",
          "code": "def form_lagranges_equations(self):\n        \"\"\"Method to form Lagrange's equations of motion.\n\n        Returns a vector of equations of motion using Lagrange's equations of\n        the second kind.\n        \"\"\"\n\n        qds = self._qdots\n        qdd_zero = dict((i, 0) for i in self._qdoubledots)\n        n = len(self.q)\n\n        # Internally we represent the EOM as four terms:\n        # EOM = term1 - term2 - term3 - term4 = 0\n\n        # First term\n        self._term1 = self._L.jacobian(qds)\n        self._term1 = self._term1.diff(dynamicsymbols._t).T\n\n        # Second term\n        self._term2 = self._L.jacobian(self.q).T\n\n        # Third term\n        if self.coneqs:\n            coneqs = self.coneqs\n            m = len(coneqs)\n            # Creating the multipliers\n            self.lam_vec = Matrix(dynamicsymbols('lam1:' + str(m + 1)))\n            self.lam_coeffs = -coneqs.jacobian(qds)\n            self._term3 = self.lam_coeffs.T * self.lam_vec\n            # Extracting the coeffecients of the qdds from the diff coneqs\n            diffconeqs = coneqs.diff(dynamicsymbols._t)\n            self._m_cd = diffconeqs.jacobian(self._qdoubledots)\n            # The remaining terms i.e. the 'forcing' terms in diff coneqs\n            self._f_cd = -diffconeqs.subs(qdd_zero)\n        else:\n            self._term3 = zeros(n, 1)\n\n        # Fourth term\n        if self.forcelist:\n            N = self.inertial\n            self._term4 = zeros(n, 1)\n            flist = zip(*_f_list_parser(self.forcelist, N))\n            for i, qd in enumerate(qds):\n                for obj, force in self.forcelist:\n                    self._term4[i] = sum(v.diff(qd, N) & f for (v, f) in flist)\n        else:\n            self._term4 = zeros(n, 1)\n\n        # Form the dynamic mass and forcing matrices\n        without_lam = self._term1 - self._term2 - self._term4\n        self._m_d = without_lam.jacobian(self._qdoubledots)\n        self._f_d = -without_lam.subs(qdd_zero)\n\n        # Form the EOM\n        self.eom = without_lam - self._term3\n        return self.eom"
        }
      ]
    },
    {
      "pr_number": 20639,
      "pr_title": "Pretty printing bug fix",
      "pr_body": "<!-- Your title above should be a short description of what\r\nwas changed. Do not include the issue number in the title. -->\r\n\r\n#### References to other Issues or PRs\r\n<!-- If this pull request fixes an issue, write \"Fixes #NNNN\" in that exact\r\nformat, e.g. \"Fixes #1234\" (see\r\nhttps://tinyurl.com/auto-closing for more information). Also, please\r\nwrite a comment on that issue linking back to this pull request once it is\r\nopen. -->\r\nFixes #17616 \r\nCloses https://github.com/sympy/sympy/pull/17620\r\nCloses https://github.com/sympy/sympy/pull/17628\r\n\r\n#### Brief description of what is fixed or changed\r\nEarlier, \r\n```\r\n>>> pprint(pi**(1/exp(1)))\r\n-1___\r\n\u2572\u2571 \u03c0 \r\n```\r\nNow,\r\n```\r\n>>> pprint(pi**(1/exp(1)))\r\n \u239b -1\u239e\r\n \u239d\u212f  \u23a0\r\n\u03c0     \r\n```\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n* printing\r\n    * irrational powers are no longer printed with square root sign, they are printed as fractional powers\r\n<!-- END RELEASE NOTES -->",
      "issue_id": 17616,
      "issue_title": "inaccurate rendering of pi**(1/E)",
      "issue_body": "This claims to be version 1.5.dev; I just merged from the project master, so I hope this is current.  I didn't notice this bug among others in printing.pretty.\r\n\r\n```\r\nIn [52]: pi**(1/E)                                                               \r\nOut[52]: \r\n-1___\r\n\u2572\u2571 \u03c0 \r\n\r\n```\r\nLaTeX and str not fooled:\r\n```\r\nIn [53]: print(latex(pi**(1/E)))                                                 \r\n\\pi^{e^{-1}}\r\n\r\nIn [54]: str(pi**(1/E))                                                          \r\nOut[54]: 'pi**exp(-1)'\r\n```\r\n",
      "issue_closed_at": "2021-01-10T06:33:26Z",
      "base_commit": "eb926a1d0c1158bf43f01eaf673dc84416b5ebb1",
      "changes": [
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_Mul",
          "class_name": "PrettyPrinter",
          "code": "def _print_Mul(self, product):\n        from sympy.physics.units import Quantity\n\n        # Check for unevaluated Mul. In this case we need to make sure the\n        # identities are visible, multiple Rational factors are not combined\n        # etc so we display in a straight-forward form that fully preserves all\n        # args and their order.\n        args = product.args\n        if args[0] is S.One or any(isinstance(arg, Number) for arg in args[1:]):\n            strargs = list(map(self._print, args))\n            # XXX: This is a hack to work around the fact that\n            # prettyForm.__mul__ absorbs a leading -1 in the args. Probably it\n            # would be better to fix this in prettyForm.__mul__ instead.\n            negone = strargs[0] == '-1'\n            if negone:\n                strargs[0] = prettyForm('1', 0, 0)\n            obj = prettyForm.__mul__(*strargs)\n            if negone:\n                obj = prettyForm('-' + obj.s, obj.baseline, obj.binding)\n            return obj\n\n        a = []  # items in the numerator\n        b = []  # items that are in the denominator (if any)\n\n        if self.order not in ('old', 'none'):\n            args = product.as_ordered_factors()\n        else:\n            args = list(product.args)\n\n        # If quantities are present append them at the back\n        args = sorted(args, key=lambda x: isinstance(x, Quantity) or\n                     (isinstance(x, Pow) and isinstance(x.base, Quantity)))\n\n        # Gather terms for numerator/denominator\n        for item in args:\n            if item.is_commutative and item.is_Pow and item.exp.is_Rational and item.exp.is_negative:\n                if item.exp != -1:\n                    b.append(Pow(item.base, -item.exp, evaluate=False))\n                else:\n                    b.append(Pow(item.base, -item.exp))\n            elif item.is_Rational and item is not S.Infinity:\n                if item.p != 1:\n                    a.append( Rational(item.p) )\n                if item.q != 1:\n                    b.append( Rational(item.q) )\n            else:\n                a.append(item)\n\n        from sympy import Integral, Piecewise, Product, Sum\n\n        # Convert to pretty forms. Add parens to Add instances if there\n        # is more than one term in the numer/denom\n        for i in range(0, len(a)):\n            if (a[i].is_Add and len(a) > 1) or (i != len(a) - 1 and\n                    isinstance(a[i], (Integral, Piecewise, Product, Sum))):\n                a[i] = prettyForm(*self._print(a[i]).parens())\n            elif a[i].is_Relational:\n                a[i] = prettyForm(*self._print(a[i]).parens())\n            else:\n                a[i] = self._print(a[i])\n\n        for i in range(0, len(b)):\n            if (b[i].is_Add and len(b) > 1) or (i != len(b) - 1 and\n                    isinstance(b[i], (Integral, Piecewise, Product, Sum))):\n                b[i] = prettyForm(*self._print(b[i]).parens())\n            else:\n                b[i] = self._print(b[i])\n\n        # Construct a pretty form\n        if len(b) == 0:\n            return prettyForm.__mul__(*a)\n        else:\n            if len(a) == 0:\n                a.append( self._print(S.One) )\n            return prettyForm.__mul__(*a)/prettyForm.__mul__(*b)"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_nth_root",
          "class_name": "PrettyPrinter",
          "code": "def _print_nth_root(self, base, expt):\n        bpretty = self._print(base)\n\n        # In very simple cases, use a single-char root sign\n        if (self._settings['use_unicode_sqrt_char'] and self._use_unicode\n            and expt is S.Half and bpretty.height() == 1\n            and (bpretty.width() == 1\n                 or (base.is_Integer and base.is_nonnegative))):\n            return prettyForm(*bpretty.left('\\N{SQUARE ROOT}'))\n\n        # Construct root sign, start with the \\/ shape\n        _zZ = xobj('/', 1)\n        rootsign = xobj('\\\\', 1) + _zZ\n        # Make exponent number to put above it\n        if isinstance(expt, Rational):\n            exp = str(expt.q)\n            if exp == '2':\n                exp = ''\n        else:\n            exp = str(expt.args[0])\n        exp = exp.ljust(2)\n        if len(exp) > 2:\n            rootsign = ' '*(len(exp) - 2) + rootsign\n        # Stack the exponent\n        rootsign = stringPict(exp + '\\n' + rootsign)\n        rootsign.baseline = 0\n        # Diagonal: length is one less than height of base\n        linelength = bpretty.height() - 1\n        diagonal = stringPict('\\n'.join(\n            ' '*(linelength - i - 1) + _zZ + ' '*i\n            for i in range(linelength)\n        ))\n        # Put baseline just below lowest line: next to exp\n        diagonal.baseline = linelength - 1\n        # Make the root symbol\n        rootsign = prettyForm(*rootsign.right(diagonal))\n        # Det the baseline to match contents to fix the height\n        # but if the height of bpretty is one, the rootsign must be one higher\n        rootsign.baseline = max(1, bpretty.baseline)\n        #build result\n        s = prettyForm(hobj('_', 2 + bpretty.width()))\n        s = prettyForm(*bpretty.above(s))\n        s = prettyForm(*s.left(rootsign))\n        return s"
        },
        {
          "file": "sympy/printing/pretty/pretty.py",
          "type": "function",
          "name": "_print_Pow",
          "class_name": "PrettyPrinter",
          "code": "def _print_Pow(self, power):\n        from sympy.simplify.simplify import fraction\n        b, e = power.as_base_exp()\n        if power.is_commutative:\n            if e is S.NegativeOne:\n                return prettyForm(\"1\")/self._print(b)\n            n, d = fraction(e)\n            if n is S.One and d.is_Atom and not e.is_Integer and self._settings['root_notation']:\n                return self._print_nth_root(b, e)\n            if e.is_Rational and e < 0:\n                return prettyForm(\"1\")/self._print(Pow(b, -e, evaluate=False))\n\n        if b.is_Relational:\n            return prettyForm(*self._print(b).parens()).__pow__(self._print(e))\n\n        return self._print(b)**self._print(e)"
        }
      ]
    },
    {
      "pr_number": 15849,
      "pr_title": "Fixed vector derivatives printing ",
      "pr_body": "<!-- Your title above should be a short description of what\r\nwas changed. Do not include the issue number in the title. -->\r\n\r\n#### References to other Issues or PRs\r\n<!-- If this pull request fixes an issue, write \"Fixes #NNNN\" in that exact\r\nformat, e.g. \"Fixes #1234\". See\r\nhttps://github.com/blog/1506-closing-issues-via-pull-requests . Please also\r\nwrite a comment on that issue linking back to this pull request once it is\r\nopen. -->\r\nFixes #10173\r\n\r\n#### Brief description of what is fixed or changed\r\nFixed the fourth-order printing issue.\r\n\r\nThis has been attempted in #10283 and #15022 but no unit tests were added. It wasn't clear which one of these I should base it on, but as seen, the current PR deals with higher order derivatives as well. And fixes some unused/redundant imports.\r\n\r\n#### Other comments\r\nI choose to keep two separate `if`-clauses to make the comments stand alone, while the newly added `if dot_i >= 5:` could easily have been added with the previous `if`.\r\n\r\nThe result looks different in different renders, see https://github.com/sympy/sympy/issues/14425#issuecomment-457834386 Hence, it may look a bit \"ugly\" in certain circumstances.\r\n\r\n#### Release Notes\r\n\r\n<!-- Write the release notes for this release below. See\r\nhttps://github.com/sympy/sympy/wiki/Writing-Release-Notes for more information\r\non how to write release notes. The bot will check your release notes\r\nautomatically to see if they are formatted correctly. -->\r\n\r\n<!-- BEGIN RELEASE NOTES -->\r\n* physics.vector\r\n   * printing of arbitrary order vector derivatives is fixed\r\n<!-- END RELEASE NOTES -->\r\n",
      "issue_id": 10173,
      "issue_title": "mechanics_printing module does not display 4th derivatives correctly ",
      "issue_body": "When I run the following code snippet in an IPython notebook, the 1st, 2nd, and 3rd derivatives display correctly, but the 4th derivative does not. The 4th derivative simply displays as p, rather than as \\ddddot{p}.\n\n```\n%pylab inline\n\nimport sympy.physics.mechanics.functions\n\nsympy.physics.mechanics.functions.mechanics_printing(use_latex=\"mathjax\", latex_mode=\"equation\")\n\nt = sympy.Symbol(\"t\")\np = sympy.Symbol(\"p\")(t)\n\np, p.diff(t), p.diff(t).diff(t), p.diff(t).diff(t).diff(t), p.diff(t).diff(t).diff(t).diff(t)\n```\n\nAt first glance, this behavior seems to be caused by the `_print_Derivative` function in sympy/sympy/physics/vector/printing.py not handling 4th derivatives. I'll try hacking this function locally to support 4th derivatives and report back if this works around the issue.\n\nIdeally, mechanics_printing should gracefully fall back to Leibniz notation in the case of high-order derivatives, rather than printing them incorrectly.\n",
      "issue_closed_at": "2019-01-29T15:53:05Z",
      "base_commit": "1e999fc5159b830a9872b86675bc5f3692a9c1be",
      "changes": [
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        },
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        },
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        },
        {
          "file": "sympy/physics/vector/printing.py",
          "type": "function",
          "name": "_print_Derivative",
          "class_name": "VectorPrettyPrinter",
          "code": "def _print_Derivative(self, deriv):\n        from sympy.physics.vector.functions import dynamicsymbols\n        # XXX use U('PARTIAL DIFFERENTIAL') here ?\n        t = dynamicsymbols._t\n        dot_i = 0\n        can_break = True\n        syms = list(reversed(deriv.variables))\n        x = None\n\n        while len(syms) > 0:\n            if syms[-1] == t:\n                syms.pop()\n                dot_i += 1\n            else:\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        if not (isinstance(type(deriv.expr), UndefinedFunction)\n                and (deriv.expr.args == (t,))):\n                return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n        else:\n            pform = self._print_Function(deriv.expr)\n        # the following condition would happen with some sort of non-standard\n        # dynamic symbol I guess, so we'll just print the SymPy way\n        if len(pform.picture) > 1:\n            return super(VectorPrettyPrinter, self)._print_Derivative(deriv)\n\n        dots = {0 : u\"\",\n                1 : u\"\\N{COMBINING DOT ABOVE}\",\n                2 : u\"\\N{COMBINING DIAERESIS}\",\n                3 : u\"\\N{COMBINING THREE DOTS ABOVE}\",\n                4 : u\"\\N{COMBINING FOUR DOTS ABOVE}\"}\n\n        d = pform.__dict__\n        pic = d['picture'][0]\n        uni = d['unicode']\n        lp = len(pic) // 2 + 1\n        lu = len(uni) // 2 + 1\n        pic_split = [pic[:lp], pic[lp:]]\n        uni_split = [uni[:lu], uni[lu:]]\n\n        d['picture'] = [pic_split[0] + dots[dot_i] + pic_split[1]]\n        d['unicode'] =  uni_split[0] + dots[dot_i] + uni_split[1]\n\n        return pform"
        }
      ]
    },
    {
      "pr_number": 13437,
      "pr_title": "Fix bell(n) for n=oo, and negative/non-integer 'n'",
      "pr_body": "This PR uses commits of #9198 (maintaining ownership to original author). And tries to address the requested changes on PR #9198 .\r\nFixes #9184",
      "issue_id": 9184,
      "issue_title": "bell(n).limit(n, oo) should be oo rather than bell(oo)",
      "issue_body": "`bell(n).limit(n,oo)` should take the value infinity, but the current output is `bell(oo)`. As the Bell numbers represent the number of partitions of a set, it seems natural that `bell(oo)` should be able to be evaluated rather than be returned unevaluated. This issue is also in line with the recent fixes to the corresponding limit for the Fibonacci numbers and Lucas numbers.\n\n```\nfrom sympy import *\nn = symbols('n')\nbell(n).limit(n,oo)\n\nOutput:\nbell(oo)\n```\n\nI'm new to Sympy, so I'd appreciate the opportunity to fix this bug myself if that's alright.\n",
      "issue_closed_at": "2017-10-13T04:35:13Z",
      "base_commit": "674afc619d7f5c519b6a5393a8b0532a131e57e0",
      "changes": [
        {
          "file": "sympy/functions/combinatorial/numbers.py",
          "type": "function",
          "name": "_bell_incomplete_poly",
          "class_name": "bell",
          "code": "def _bell_incomplete_poly(n, k, symbols):\n        r\"\"\"\n        The second kind of Bell polynomials (incomplete Bell polynomials).\n\n        Calculated by recurrence formula:\n\n        .. math:: B_{n,k}(x_1, x_2, \\dotsc, x_{n-k+1}) =\n                \\sum_{m=1}^{n-k+1}\n                \\x_m \\binom{n-1}{m-1} B_{n-m,k-1}(x_1, x_2, \\dotsc, x_{n-m-k})\n\n        where\n            B_{0,0} = 1;\n            B_{n,0} = 0; for n>=1\n            B_{0,k} = 0; for k>=1\n\n        \"\"\"\n        if (n == 0) and (k == 0):\n            return S.One\n        elif (n == 0) or (k == 0):\n            return S.Zero\n        s = S.Zero\n        a = S.One\n        for m in range(1, n - k + 2):\n            s += a * bell._bell_incomplete_poly(\n                n - m, k - 1, symbols) * symbols[m - 1]\n            a = a * (n - m) / m\n        return expand_mul(s)"
        }
      ]
    }
  ]
}