updated for version 7.3.966
Problem: There is ":py3do" but no ":pydo". Solution: Add the ":pydo" command. (Lilydjwg)
This commit is contained in:
		| @ -57,6 +57,22 @@ Example: > | |||||||
| Note: Python is very sensitive to the indenting.  Make sure the "class" line | Note: Python is very sensitive to the indenting.  Make sure the "class" line | ||||||
| and "EOF" do not have any indent. | and "EOF" do not have any indent. | ||||||
|  |  | ||||||
|  | 							*:pydo* | ||||||
|  | :[range]pydo {body}	Execute Python function "def _vim_pydo(line, linenr): | ||||||
|  | 			{body}" for each line in the [range], with the | ||||||
|  | 			function arguments being set to the text of each line | ||||||
|  | 			in turn, without a trailing <EOL>, and the current | ||||||
|  | 			line number. The function should return a string or | ||||||
|  | 			None. If a string is returned, it becomes the text of | ||||||
|  | 			the line in the current turn. The default for [range] | ||||||
|  | 			is the whole file: "1,$". | ||||||
|  | 			{not in Vi} | ||||||
|  |  | ||||||
|  | Examples: | ||||||
|  | > | ||||||
|  | 	:pydo return "%s\t%d" % (line[::-1], len(line)) | ||||||
|  | 	:pydo if line: return "%4d: %s" % (linenr, line) | ||||||
|  | < | ||||||
| 							*:pyfile* *:pyf* | 							*:pyfile* *:pyf* | ||||||
| :[range]pyf[ile] {file} | :[range]pyf[ile] {file} | ||||||
| 			Execute the Python script in {file}.  The whole | 			Execute the Python script in {file}.  The whole | ||||||
| @ -485,27 +501,14 @@ sure edit "gvim.exe" and search for "python\d*.dll\c". | |||||||
| 8. Python 3						*python3* | 8. Python 3						*python3* | ||||||
|  |  | ||||||
| 							*:py3* *:python3* | 							*:py3* *:python3* | ||||||
| The |:py3| and |:python3| commands work similar to |:python|.  A simple check | The `:py3` and `:python3` commands work similar to `:python`.  A simple check | ||||||
| if the `:py3` command is working: > | if the `:py3` command is working: > | ||||||
| 	:py3 print("Hello") | 	:py3 print("Hello") | ||||||
| <							*:py3file* | <							*:py3file* | ||||||
| The |:py3file| command works similar to |:pyfile|. | The `:py3file` command works similar to `:pyfile`. | ||||||
|  |  | ||||||
| 							*:py3do* *E863* | 							*:py3do* *E863* | ||||||
| :[range]py3do {body}	Execute Python function "def _vim_pydo(line, linenr): | The `:py3do` command works similar to `:pydo`. | ||||||
| 			{body}" for each line in the [range], with the |  | ||||||
| 			function arguments being set to the text of each line |  | ||||||
| 			in turn, without a trailing <EOL>, and the current |  | ||||||
| 			line number. The function should return a string or |  | ||||||
| 			None. If a string is returned, it becomes the text of |  | ||||||
| 			the line in the current turn. The default for [range] |  | ||||||
| 			is the whole file: "1,$". |  | ||||||
| 			{not in Vi} |  | ||||||
|  |  | ||||||
| Examples: |  | ||||||
| > |  | ||||||
| 	:py3do return "%s\t%d" % (line[::-1], len(line)) |  | ||||||
| 	:py3do if line: return "%4d: %s" % (linenr, line) |  | ||||||
|  |  | ||||||
| Vim can be built in four ways (:version output): | Vim can be built in four ways (:version output): | ||||||
| 1. No Python support	    (-python, -python3) | 1. No Python support	    (-python, -python3) | ||||||
|  | |||||||
| @ -739,6 +739,8 @@ EX(CMD_pwd,		"pwd",		ex_pwd, | |||||||
| 			TRLBAR|CMDWIN), | 			TRLBAR|CMDWIN), | ||||||
| EX(CMD_python,		"python",	ex_python, | EX(CMD_python,		"python",	ex_python, | ||||||
| 			RANGE|EXTRA|NEEDARG|CMDWIN), | 			RANGE|EXTRA|NEEDARG|CMDWIN), | ||||||
|  | EX(CMD_pydo,		"pydo",		ex_pydo, | ||||||
|  | 			RANGE|DFLALL|EXTRA|NEEDARG|CMDWIN), | ||||||
| EX(CMD_pyfile,		"pyfile",	ex_pyfile, | EX(CMD_pyfile,		"pyfile",	ex_pyfile, | ||||||
| 			RANGE|FILE1|NEEDARG|CMDWIN), | 			RANGE|FILE1|NEEDARG|CMDWIN), | ||||||
| EX(CMD_py3,		"py3",		ex_py3, | EX(CMD_py3,		"py3",		ex_py3, | ||||||
|  | |||||||
| @ -268,6 +268,7 @@ static void	ex_popup __ARGS((exarg_T *eap)); | |||||||
| #endif | #endif | ||||||
| #ifndef FEAT_PYTHON | #ifndef FEAT_PYTHON | ||||||
| # define ex_python		ex_script_ni | # define ex_python		ex_script_ni | ||||||
|  | # define ex_pydo		ex_ni | ||||||
| # define ex_pyfile		ex_ni | # define ex_pyfile		ex_ni | ||||||
| #endif | #endif | ||||||
| #ifndef FEAT_PYTHON3 | #ifndef FEAT_PYTHON3 | ||||||
|  | |||||||
| @ -22,6 +22,7 @@ typedef int Py_ssize_t;  /* Python 2.4 and earlier don't have this type. */ | |||||||
| #else | #else | ||||||
| # define ENC_OPT "latin1" | # define ENC_OPT "latin1" | ||||||
| #endif | #endif | ||||||
|  | #define DOPY_FUNC "_vim_pydo" | ||||||
|  |  | ||||||
| #define PyErr_SetVim(str) PyErr_SetString(VimError, str) | #define PyErr_SetVim(str) PyErr_SetString(VimError, str) | ||||||
|  |  | ||||||
|  | |||||||
| @ -198,6 +198,9 @@ struct PyMethodDef { Py_ssize_t a; }; | |||||||
| # define PyModule_GetDict dll_PyModule_GetDict | # define PyModule_GetDict dll_PyModule_GetDict | ||||||
| # define PyRun_SimpleString dll_PyRun_SimpleString | # define PyRun_SimpleString dll_PyRun_SimpleString | ||||||
| # define PyRun_String dll_PyRun_String | # define PyRun_String dll_PyRun_String | ||||||
|  | # define PyObject_GetAttrString dll_PyObject_GetAttrString | ||||||
|  | # define PyObject_SetAttrString dll_PyObject_SetAttrString | ||||||
|  | # define PyObject_CallFunctionObjArgs dll_PyObject_CallFunctionObjArgs | ||||||
| # define PyString_AsString dll_PyString_AsString | # define PyString_AsString dll_PyString_AsString | ||||||
| # define PyString_AsStringAndSize dll_PyString_AsStringAndSize | # define PyString_AsStringAndSize dll_PyString_AsStringAndSize | ||||||
| # define PyString_FromString dll_PyString_FromString | # define PyString_FromString dll_PyString_FromString | ||||||
| @ -303,6 +306,9 @@ static PyObject* (*dll_PyIter_Next)(PyObject *); | |||||||
| static PyObject*(*dll_PyModule_GetDict)(PyObject *); | static PyObject*(*dll_PyModule_GetDict)(PyObject *); | ||||||
| static int(*dll_PyRun_SimpleString)(char *); | static int(*dll_PyRun_SimpleString)(char *); | ||||||
| static PyObject *(*dll_PyRun_String)(char *, int, PyObject *, PyObject *); | static PyObject *(*dll_PyRun_String)(char *, int, PyObject *, PyObject *); | ||||||
|  | static PyObject* (*dll_PyObject_GetAttrString)(PyObject *, const char *); | ||||||
|  | static PyObject* (*dll_PyObject_SetAttrString)(PyObject *, const char *, PyObject *); | ||||||
|  | static PyObject* (*dll_PyObject_CallFunctionObjArgs)(PyObject *, ...); | ||||||
| static char*(*dll_PyString_AsString)(PyObject *); | static char*(*dll_PyString_AsString)(PyObject *); | ||||||
| static int(*dll_PyString_AsStringAndSize)(PyObject *, char **, int *); | static int(*dll_PyString_AsStringAndSize)(PyObject *, char **, int *); | ||||||
| static PyObject*(*dll_PyString_FromString)(const char *); | static PyObject*(*dll_PyString_FromString)(const char *); | ||||||
| @ -440,6 +446,9 @@ static struct | |||||||
|     {"PyModule_GetDict", (PYTHON_PROC*)&dll_PyModule_GetDict}, |     {"PyModule_GetDict", (PYTHON_PROC*)&dll_PyModule_GetDict}, | ||||||
|     {"PyRun_SimpleString", (PYTHON_PROC*)&dll_PyRun_SimpleString}, |     {"PyRun_SimpleString", (PYTHON_PROC*)&dll_PyRun_SimpleString}, | ||||||
|     {"PyRun_String", (PYTHON_PROC*)&dll_PyRun_String}, |     {"PyRun_String", (PYTHON_PROC*)&dll_PyRun_String}, | ||||||
|  |     {"PyObject_GetAttrString", (PYTHON_PROC*)&dll_PyObject_GetAttrString}, | ||||||
|  |     {"PyObject_SetAttrString", (PYTHON_PROC*)&dll_PyObject_SetAttrString}, | ||||||
|  |     {"PyObject_CallFunctionObjArgs", (PYTHON_PROC*)&dll_PyObject_CallFunctionObjArgs}, | ||||||
|     {"PyString_AsString", (PYTHON_PROC*)&dll_PyString_AsString}, |     {"PyString_AsString", (PYTHON_PROC*)&dll_PyString_AsString}, | ||||||
|     {"PyString_AsStringAndSize", (PYTHON_PROC*)&dll_PyString_AsStringAndSize}, |     {"PyString_AsStringAndSize", (PYTHON_PROC*)&dll_PyString_AsStringAndSize}, | ||||||
|     {"PyString_FromString", (PYTHON_PROC*)&dll_PyString_FromString}, |     {"PyString_FromString", (PYTHON_PROC*)&dll_PyString_FromString}, | ||||||
| @ -995,6 +1004,93 @@ ex_pyfile(exarg_T *eap) | |||||||
|     DoPythonCommand(eap, buffer, NULL); |     DoPythonCommand(eap, buffer, NULL); | ||||||
| } | } | ||||||
|  |  | ||||||
|  |     void | ||||||
|  | ex_pydo(exarg_T *eap) | ||||||
|  | { | ||||||
|  |     linenr_T		i; | ||||||
|  |     const char		*code_hdr = "def " DOPY_FUNC "(line, linenr):\n "; | ||||||
|  |     const char		*s = (const char *) eap->arg; | ||||||
|  |     size_t		len; | ||||||
|  |     char		*code; | ||||||
|  |     int			status; | ||||||
|  |     PyObject		*pyfunc, *pymain; | ||||||
|  |     PyGILState_STATE	pygilstate; | ||||||
|  |  | ||||||
|  |     if (Python_Init()) | ||||||
|  |         return; | ||||||
|  |  | ||||||
|  |     if (u_save(eap->line1 - 1, eap->line2 + 1) != OK) | ||||||
|  |     { | ||||||
|  | 	EMSG(_("cannot save undo information")); | ||||||
|  | 	return; | ||||||
|  |     } | ||||||
|  |     len = strlen(code_hdr) + strlen(s); | ||||||
|  |     code = malloc(len + 1); | ||||||
|  |     STRCPY(code, code_hdr); | ||||||
|  |     STRNCAT(code, s, len + 1); | ||||||
|  |     pygilstate = PyGILState_Ensure(); | ||||||
|  |     status = PyRun_SimpleString(code); | ||||||
|  |     vim_free(code); | ||||||
|  |     if (status) | ||||||
|  |     { | ||||||
|  | 	EMSG(_("failed to run the code")); | ||||||
|  | 	return; | ||||||
|  |     } | ||||||
|  |     status = 0; /* good */ | ||||||
|  |     pymain = PyImport_AddModule("__main__"); | ||||||
|  |     pyfunc = PyObject_GetAttrString(pymain, DOPY_FUNC); | ||||||
|  |     PyGILState_Release(pygilstate); | ||||||
|  |  | ||||||
|  |     for (i = eap->line1; i <= eap->line2; i++) | ||||||
|  |     { | ||||||
|  | 	const char *line; | ||||||
|  | 	PyObject *pyline, *pylinenr, *pyret; | ||||||
|  |  | ||||||
|  | 	line = (char *)ml_get(i); | ||||||
|  | 	pygilstate = PyGILState_Ensure(); | ||||||
|  | 	pyline = PyString_FromStringAndSize(line, strlen(line)); | ||||||
|  | 	pylinenr = PyLong_FromLong(i); | ||||||
|  | 	pyret = PyObject_CallFunctionObjArgs(pyfunc, pyline, pylinenr, NULL); | ||||||
|  | 	Py_DECREF(pyline); | ||||||
|  | 	Py_DECREF(pylinenr); | ||||||
|  | 	if (!pyret) | ||||||
|  | 	{ | ||||||
|  | 	    PyErr_PrintEx(0); | ||||||
|  | 	    PythonIO_Flush(); | ||||||
|  | 	    status = 1; | ||||||
|  | 	    goto out; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	if (pyret && pyret != Py_None) | ||||||
|  | 	{ | ||||||
|  | 	    if (!PyString_Check(pyret)) | ||||||
|  | 	    { | ||||||
|  | 		EMSG(_("E863: return value must be an instance of str")); | ||||||
|  | 		Py_XDECREF(pyret); | ||||||
|  | 		status = 1; | ||||||
|  | 		goto out; | ||||||
|  | 	    } | ||||||
|  | 	    ml_replace(i, (char_u *) PyString_AsString(pyret), 1); | ||||||
|  | 	    changed(); | ||||||
|  | #ifdef SYNTAX_HL | ||||||
|  | 	    syn_changed(i); /* recompute syntax hl. for this line */ | ||||||
|  | #endif | ||||||
|  | 	} | ||||||
|  | 	Py_XDECREF(pyret); | ||||||
|  | 	PythonIO_Flush(); | ||||||
|  | 	PyGILState_Release(pygilstate); | ||||||
|  |     } | ||||||
|  |     pygilstate = PyGILState_Ensure(); | ||||||
|  | out: | ||||||
|  |     Py_DECREF(pyfunc); | ||||||
|  |     PyObject_SetAttrString(pymain, DOPY_FUNC, NULL); | ||||||
|  |     PyGILState_Release(pygilstate); | ||||||
|  |     if (status) | ||||||
|  | 	return; | ||||||
|  |     check_cursor(); | ||||||
|  |     update_curbuf(NOT_VALID); | ||||||
|  | } | ||||||
|  |  | ||||||
| /****************************************************** | /****************************************************** | ||||||
|  * 2. Python output stream: writes output via [e]msg(). |  * 2. Python output stream: writes output via [e]msg(). | ||||||
|  */ |  */ | ||||||
|  | |||||||
| @ -76,7 +76,6 @@ static void init_structs(void); | |||||||
| #else | #else | ||||||
| # define CODEC_ERROR_HANDLER NULL | # define CODEC_ERROR_HANDLER NULL | ||||||
| #endif | #endif | ||||||
| #define DOPY_FUNC "_vim_pydo" |  | ||||||
|  |  | ||||||
| /* Python 3 does not support CObjects, always use Capsules */ | /* Python 3 does not support CObjects, always use Capsules */ | ||||||
| #define PY_USE_CAPSULE | #define PY_USE_CAPSULE | ||||||
|  | |||||||
| @ -3,6 +3,7 @@ int python_enabled __ARGS((int verbose)); | |||||||
| void python_end __ARGS((void)); | void python_end __ARGS((void)); | ||||||
| int python_loaded __ARGS((void)); | int python_loaded __ARGS((void)); | ||||||
| void ex_python __ARGS((exarg_T *eap)); | void ex_python __ARGS((exarg_T *eap)); | ||||||
|  | void ex_pydo __ARGS((exarg_T *eap)); | ||||||
| void ex_pyfile __ARGS((exarg_T *eap)); | void ex_pyfile __ARGS((exarg_T *eap)); | ||||||
| void python_buffer_free __ARGS((buf_T *buf)); | void python_buffer_free __ARGS((buf_T *buf)); | ||||||
| void python_window_free __ARGS((win_T *win)); | void python_window_free __ARGS((win_T *win)); | ||||||
|  | |||||||
| @ -728,6 +728,8 @@ static char *(features[]) = | |||||||
|  |  | ||||||
| static int included_patches[] = | static int included_patches[] = | ||||||
| {   /* Add new patch number below this line */ | {   /* Add new patch number below this line */ | ||||||
|  | /**/ | ||||||
|  |     966, | ||||||
| /**/ | /**/ | ||||||
|     965, |     965, | ||||||
| /**/ | /**/ | ||||||
|  | |||||||
		Reference in New Issue
	
	Block a user