{
  "Selected_candidate": {
    "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⋅⎡0  1  2  3  4⎤\r\n  ⎢             ⎥\r\n  ⎢1  2  3  4  5⎥\r\n  ⎢             ⎥\r\n  ⎢2  3  4  5  6⎥\r\n  ⎢             ⎥\r\n  ⎢3  4  5  6  7⎥\r\n  ⎢             ⎥\r\n  ⎣4  5  6  7  8⎦\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  ⎡0  1  2  3  4⎤\r\n  ⎢             ⎥\r\n  ⎢1  2  3  4  5⎥\r\n  ⎢             ⎥\r\n2.⎢2  3  4  5  6⎥\r\n  ⎢             ⎥\r\n  ⎢3  4  5  6  7⎥\r\n  ⎢             ⎥\r\n  ⎣4  5  6  7  8⎦\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⋅⎡0  1  2  3  4⎤\n  ⎢             ⎥\n  ⎢1  2  3  4  5⎥\n  ⎢             ⎥\n  ⎢2  3  4  5  6⎥\n  ⎢             ⎥\n  ⎢3  4  5  6  7⎥\n  ⎢             ⎥\n  ⎣4  5  6  7  8⎦\n>>> pprint(MatAdd(Z,Z))\n⎡0  1  2  3  4⎤ + ⎡0  1  2  3  4⎤\n⎢             ⎥   ⎢             ⎥\n⎢1  2  3  4  5⎥   ⎢1  2  3  4  5⎥\n⎢             ⎥   ⎢             ⎥\n⎢2  3  4  5  6⎥   ⎢2  3  4  5  6⎥\n⎢             ⎥   ⎢             ⎥\n⎢3  4  5  6  7⎥   ⎢3  4  5  6  7⎥\n⎢             ⎥   ⎢             ⎥\n⎣4  5  6  7  8⎦   ⎣4  5  6  7  8⎦\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   ⎛    1⎞                       \n   ⎜1 + ─⎟                       \n   ⎜    x⎟                       \nsin⎜─────⎟ + 32⋅│⎡0  1  2  3  4⎤│\n   ⎜    1⎟      │⎢             ⎥│\n   ⎜1 + ─⎟      │⎢1  2  3  4  5⎥│\n   ⎝    y⎠      │⎢             ⎥│\n                │⎢2  3  4  5  6⎥│\n                │⎢             ⎥│\n                │⎢3  4  5  6  7⎥│\n                │⎢             ⎥│\n                │⎣4  5  6  7  8⎦│\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               ⎛⎡0  1  2  3  4⎤⎞\n   ⎛    1⎞     ⎜⎢             ⎥⎟\n   ⎜1 + ─⎟     ⎜⎢1  2  3  4  5⎥⎟\n   ⎜    x⎟     ⎜⎢             ⎥⎟\nsin⎜─────⎟ + tr⎜⎢2  3  4  5  6⎥⎟\n   ⎝  y  ⎠     ⎜⎢             ⎥⎟\n               ⎜⎢3  4  5  6  7⎥⎟\n               ⎜⎢             ⎥⎟\n               ⎝⎣4  5  6  7  8⎦⎠\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"
      }
    ]
  },
  "Justification": "Candidate A is the most helpful because it directly relates to the pretty printing functionality, which is the core aspect of the CURRENT bug involving the display issues of `sympy.vector` objects. Both bugs deal with rendering outputs correctly in the terminal, and the structural similarity in terms of the printing mechanism can provide valuable insights into improving the faulty pretty-printing of vector outputs observed in the CURRENT bug report. Additionally, since they both involve `sympy.printing.pretty`, there is a strong module similarity which enhances its relevance for locating and fixing the issue.",
  "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⎛     ⎛y_C⎞    ⎛  5  ⎞⎞    \r\n⎜2⋅sin⎜───⎟ i_C⋅cos⎝10 ⋅t⎠⎟\r\n⎜     ⎜  3⎟           ⎟    \r\n⎜     ⎝10 ⎠           ⎟    \r\n⎜─────────────────────⎟    \r\n⎜           4         ⎟    \r\n⎝         10          ⎠    \r\n\r\n⎛     ⌠                           ⎞    \r\n⎜     ⎮       ⎛y_C⎞    ⎛  5  ⎞    ⎟ k_C\r\n⎜     ⎮ -2⋅cos⎜───⎟⋅cos⎝10 ⋅t⎠    ⎟    \r\n⎜     ⎮       ⎜  3⎟               ⎟    \r\n⎜  11 ⎮       ⎝10 ⎠               ⎟    \r\n⎜10  ⋅⎮ ─────────────────────── dt⎟    \r\n⎜     ⎮             2             ⎟    \r\n⎜     ⎮           10              ⎟    \r\n⎜     ⌡                           ⎟    \r\n⎜─────────────────────────────────⎟    \r\n⎝               4⋅π               ⎠    \r\n\r\n⎛   4    ⎛  5  ⎞    ⎛y_C⎞ ⎞    \r\n⎜-10 ⋅sin⎝10 ⋅t⎠⋅cos⎜───⎟ k_C ⎟\r\n⎜                   ⎜  3⎟ ⎟    \r\n⎜                   ⎝10 ⎠ ⎟    \r\n⎜─────────────────────────⎟    \r\n⎝           2⋅π           ⎠    ```\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"
}