throw helpful errors when params supplied are not ctype pointers (#122)
This commit is contained in:
parent
33dce4ecfd
commit
3744686f7b
3 changed files with 77 additions and 19 deletions
|
@ -41,6 +41,8 @@ def ctype_to_python_type(t):
|
||||||
return "int"
|
return "int"
|
||||||
elif t == "double":
|
elif t == "double":
|
||||||
return "float"
|
return "float"
|
||||||
|
elif "char * *" in t:
|
||||||
|
return "list[str]"
|
||||||
elif "char *" in t:
|
elif "char *" in t:
|
||||||
return "str"
|
return "str"
|
||||||
elif "char" in t:
|
elif "char" in t:
|
||||||
|
|
|
@ -47,32 +47,43 @@ def makefunc(a):
|
||||||
for (c_arg, arg) in zip(ffi.typeof(a).args, args):
|
for (c_arg, arg) in zip(ffi.typeof(a).args, args):
|
||||||
# print("arg:",str(arg), "c_arg.kind:", c_arg.kind, "c_arg:", c_arg, "type(arg):",str(type(arg)))
|
# print("arg:",str(arg), "c_arg.kind:", c_arg.kind, "c_arg:", c_arg, "type(arg):",str(type(arg)))
|
||||||
if c_arg.kind == 'pointer':
|
if c_arg.kind == 'pointer':
|
||||||
if type(arg) == str:
|
if type(arg) is str:
|
||||||
arg = arg.encode('utf-8')
|
arg = arg.encode('utf-8')
|
||||||
elif type(arg) is bool:
|
# if c_arg is a 'char *' not a 'const char *' then we ought to raise here because its an out
|
||||||
arg = ffi.new("bool *", arg)
|
# parameter and user should supply a ctype pointer, but cffi cant detect const
|
||||||
elif type(arg) is int:
|
# so we would have to get the info from raylib.json
|
||||||
arg = ffi.new("int *", arg)
|
|
||||||
elif type(arg) is float:
|
|
||||||
arg = ffi.new("float *", arg)
|
|
||||||
elif type(arg) is list and str(c_arg) == "<ctype 'char * *'>":
|
elif type(arg) is list and str(c_arg) == "<ctype 'char * *'>":
|
||||||
arg = [ffi.new("char[]", x.encode('utf-8')) for x in arg]
|
arg = [ffi.new("char[]", x.encode('utf-8')) for x in arg]
|
||||||
elif str(type(arg)) == "<class '_cffi_backend.__CDataOwn'>" and "*" not in str(arg): # CPython
|
elif is_cdata(arg) and "*" not in str(arg):
|
||||||
arg = ffi.addressof(arg)
|
|
||||||
elif str(type(arg)) == "<class '_cffi_backend._CDataBase'>" and "*" not in str(arg): # Pypy
|
|
||||||
arg = ffi.addressof(arg)
|
arg = ffi.addressof(arg)
|
||||||
elif arg is None:
|
elif arg is None:
|
||||||
arg = ffi.NULL
|
arg = ffi.NULL
|
||||||
|
elif not is_cdata(arg):
|
||||||
|
if str(c_arg) == "<ctype '_Bool *'>":
|
||||||
|
raise TypeError(
|
||||||
|
"Argument must be a ctype bool, please create one with: pyray.ffi.new('bool *', True)")
|
||||||
|
elif str(c_arg) == "<ctype 'int *'>":
|
||||||
|
raise TypeError(
|
||||||
|
"Argument must be a ctype int, please create one with: pyray.ffi.new('int *', 1)")
|
||||||
|
elif str(c_arg) == "<ctype 'float *'>":
|
||||||
|
raise TypeError(
|
||||||
|
"Argument must be a ctype float, please create one with: pyray.ffi.new('float *', 1.0)")
|
||||||
modified_args.append(arg)
|
modified_args.append(arg)
|
||||||
result = a(*modified_args)
|
result = a(*modified_args)
|
||||||
if result is None:
|
if result is None:
|
||||||
return
|
return
|
||||||
if str(type(result)) == "<class '_cffi_backend._CDataBase'>" and str(result).startswith("<cdata 'char *'"):
|
elif is_cdata(result) and str(result).startswith("<cdata 'char *'"):
|
||||||
if str(result) == "<cdata 'char *' NULL>":
|
if str(result) == "<cdata 'char *' NULL>":
|
||||||
result = ""
|
return ""
|
||||||
else:
|
else:
|
||||||
result = ffi.string(result).decode('utf-8')
|
return ffi.string(result).decode('utf-8')
|
||||||
return result
|
else:
|
||||||
|
return result
|
||||||
|
|
||||||
|
# apparently pypy and cpython produce different types so check for both
|
||||||
|
def is_cdata(arg):
|
||||||
|
return str(type(arg)) == "<class '_cffi_backend.__CDataOwn'>" or str(
|
||||||
|
type(arg)) == "<class '_cffi_backend._CDataBase'>"
|
||||||
|
|
||||||
return func
|
return func
|
||||||
|
|
||||||
|
@ -99,20 +110,16 @@ def makeStructHelper(struct):
|
||||||
|
|
||||||
|
|
||||||
for name, attr in getmembers(rl):
|
for name, attr in getmembers(rl):
|
||||||
# print(name, attr)
|
#print(name, dir(attr))
|
||||||
uname = inflection.underscore(name).replace('3_d', '_3d').replace('2_d', '_2d')
|
uname = inflection.underscore(name).replace('3_d', '_3d').replace('2_d', '_2d')
|
||||||
if isbuiltin(attr) or str(type(attr)) == "<class '_cffi_backend.__FFIFunctionWrapper'>" or str(
|
if isbuiltin(attr) or str(type(attr)) == "<class '_cffi_backend.__FFIFunctionWrapper'>" or str(
|
||||||
type(attr)) == "<class '_cffi_backend._CDataBase'>":
|
type(attr)) == "<class '_cffi_backend._CDataBase'>":
|
||||||
# print(attr.__call__)
|
# print(attr.__call__)
|
||||||
# print(attr.__doc__)
|
# print(attr.__doc__)
|
||||||
# print(attr.__text_signature__)
|
|
||||||
# print(dir(attr))
|
# print(dir(attr))
|
||||||
# print(dir(attr.__repr__))
|
# print(dir(attr.__repr__))
|
||||||
f = makefunc(attr)
|
f = makefunc(attr)
|
||||||
setattr(current_module, uname, f)
|
setattr(current_module, uname, f)
|
||||||
# def wrap(*args):
|
|
||||||
# print("call to ",attr)
|
|
||||||
# setattr(PyRay, uname, lambda *args: print("call to ",attr))
|
|
||||||
else:
|
else:
|
||||||
setattr(current_module, name, attr)
|
setattr(current_module, name, attr)
|
||||||
|
|
||||||
|
|
49
tests/xtest_raygui2.py
Normal file
49
tests/xtest_raygui2.py
Normal file
|
@ -0,0 +1,49 @@
|
||||||
|
import pyray
|
||||||
|
from pyray import *
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
SCREEN_WIDTH = 640
|
||||||
|
SCREEN_HEIGHT = 480
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
init_window(SCREEN_WIDTH, SCREEN_HEIGHT, "raygui")
|
||||||
|
set_exit_key(KeyboardKey.KEY_ESCAPE)
|
||||||
|
set_target_fps(30)
|
||||||
|
|
||||||
|
_bool_pointer = pyray.ffi.new("bool *", True)
|
||||||
|
_bool = True
|
||||||
|
_float = 5.5
|
||||||
|
_float_pointer = pyray.ffi.new("float *", 5.2)
|
||||||
|
color = RED
|
||||||
|
color_pointer = Color()
|
||||||
|
_int=0
|
||||||
|
_int_pointer = pyray.ffi.new("int *", 0)
|
||||||
|
_string ="cant edit this string"
|
||||||
|
_string_pointer = pyray.ffi.new("char []", b"0123456789")
|
||||||
|
|
||||||
|
while not window_should_close():
|
||||||
|
begin_drawing()
|
||||||
|
clear_background(WHITE)
|
||||||
|
|
||||||
|
gui_check_box((40, 40, 30, 30), "checkbox", _bool_pointer)
|
||||||
|
with pytest.raises(TypeError):
|
||||||
|
gui_check_box(Rectangle(40, 40, 100, 100), "checkbox", _bool)
|
||||||
|
|
||||||
|
gui_color_bar_alpha((40, 80, 100, 20), "bar", _float_pointer)
|
||||||
|
with pytest.raises(TypeError):
|
||||||
|
gui_color_bar_alpha((40, 80, 100, 20), "bar", _float)
|
||||||
|
|
||||||
|
gui_color_picker((40,120, 100, 100), "color", color_pointer)
|
||||||
|
with pytest.raises(TypeError):
|
||||||
|
gui_color_picker((40,120, 100, 100), "color", color)
|
||||||
|
|
||||||
|
gui_text_box((40,230, 100, 30), _string_pointer, 10, True)
|
||||||
|
gui_text_box((200,230, 100, 30), _string, 10, True)
|
||||||
|
|
||||||
|
gui_tab_bar((40,300, 100, 30), ["foo", "boo"], 2, _int_pointer)
|
||||||
|
with pytest.raises(TypeError):
|
||||||
|
gui_tab_bar((40,300, 100, 30), ["foo", "boo"], 2, _int)
|
||||||
|
|
||||||
|
end_drawing()
|
||||||
|
main()
|
Reference in a new issue