From 18d506911ea0a3f65b3714d6b3675ee6d1a9b183 Mon Sep 17 00:00:00 2001 From: Max Smolens Date: Wed, 25 Feb 2015 22:59:28 -0500 Subject: [PATCH] Enable setting primitive data type fields on Java classes --- jnius/jnius_export_class.pxi | 55 +++++++++++++++++++++++++++++++++ tests/org/jnius/BasicsTest.java | 50 ++++++++++++++++++++++++++++-- tests/test_basics.py | 20 ++++++++++++ 3 files changed, 123 insertions(+), 2 deletions(-) diff --git a/jnius/jnius_export_class.pxi b/jnius/jnius_export_class.pxi index 7638ab5..63100b0 100644 --- a/jnius/jnius_export_class.pxi +++ b/jnius/jnius_export_class.pxi @@ -321,6 +321,61 @@ cdef class JavaField(object): j_self = (obj).j_self.obj return self.read_field(j_self) + def __set__(self, obj, value): + cdef jobject j_self + + self.ensure_field() + if obj is None: + # set not implemented for static fields + raise NotImplementedError() + + j_self = (obj).j_self.obj + self.write_field(j_self, value) + + cdef write_field(self, jobject j_self, value): + cdef jboolean j_boolean + cdef jbyte j_byte + cdef jchar j_char + cdef jshort j_short + cdef jint j_int + cdef jlong j_long + cdef jfloat j_float + cdef jdouble j_double + cdef JNIEnv *j_env = get_jnienv() + + # type of the java field + r = self.definition[0] + + # set the java field; implemented only for primitive types + if r == 'Z': + j_boolean = value + j_env[0].SetBooleanField(j_env, j_self, self.j_field, j_boolean) + elif r == 'B': + j_byte = value + j_env[0].SetByteField(j_env, j_self, self.j_field, j_byte) + elif r == 'C': + j_char = value + j_env[0].SetCharField(j_env, j_self, self.j_field, j_char) + elif r == 'S': + j_short = value + j_env[0].SetShortField(j_env, j_self, self.j_field, j_short) + elif r == 'I': + j_int = value + j_env[0].SetIntField(j_env, j_self, self.j_field, j_int) + elif r == 'J': + j_long = value + j_env[0].SetLongField(j_env, j_self, self.j_field, j_long) + elif r == 'F': + j_float = value + j_env[0].SetFloatField(j_env, j_self, self.j_field, j_float) + elif r == 'D': + j_double = value + j_env[0].SetDoubleField(j_env, j_self, self.j_field, j_double) + else: + raise Exception('Invalid field definition') + + check_exception(j_env) + cdef read_field(self, jobject j_self): cdef jboolean j_boolean cdef jbyte j_byte diff --git a/tests/org/jnius/BasicsTest.java b/tests/org/jnius/BasicsTest.java index 6a4e47d..cdfa4e8 100644 --- a/tests/org/jnius/BasicsTest.java +++ b/tests/org/jnius/BasicsTest.java @@ -43,6 +43,19 @@ public class BasicsTest { public double fieldD = 1.23456789; public String fieldString = new String("helloworld"); + public boolean fieldSetZ; + public byte fieldSetB; + public char fieldSetC; + public short fieldSetS; + public int fieldSetI; + public long fieldSetJ; + public float fieldSetF; + public double fieldSetD; + public String fieldSetString; + + // Floating-point comparison epsilon + private final static double EPSILON = 1E-6; + public BasicsTest() {} public BasicsTest(byte fieldBVal) { fieldB = fieldBVal; @@ -97,9 +110,10 @@ public class BasicsTest { public boolean methodParamsZBCSIJFD(boolean x1, byte x2, char x3, short x4, int x5, long x6, float x7, double x8) { - // ADD float / double, but dunno how to do with approx return (x1 == true && x2 == 127 && x3 == 'k' && x4 == 32767 && - x5 == 2147483467 && x6 == 2147483467); + x5 == 2147483467 && x6 == 2147483467 && + (Math.abs(x7 - 1.23456789f) < EPSILON) && + (Math.abs(x8 - 1.23456789) < EPSILON)); } public boolean methodParamsString(String s) { @@ -149,4 +163,36 @@ public class BasicsTest { x[1] = 1; x[2] = -127; } + + public boolean testFieldSetZ() { + return (fieldSetZ == true); + } + + public boolean testFieldSetB() { + return (fieldSetB == 127); + } + + public boolean testFieldSetC() { + return (fieldSetC == 'k'); + } + + public boolean testFieldSetS() { + return (fieldSetS == 32767); + } + + public boolean testFieldSetI() { + return (fieldSetI == 2147483467); + } + + public boolean testFieldSetJ() { + return (fieldSetJ == 2147483467); + } + + public boolean testFieldSetF() { + return (Math.abs(fieldSetF - 1.23456789f) < EPSILON); + } + + public boolean testFieldSetD() { + return (Math.abs(fieldSetD - 1.23456789) < EPSILON); + } } diff --git a/tests/test_basics.py b/tests/test_basics.py index 3964561..a28b620 100644 --- a/tests/test_basics.py +++ b/tests/test_basics.py @@ -55,6 +55,26 @@ class BasicsTest(unittest.TestCase): self.assertEquals(test.fieldB, 127) self.assertEquals(test2.fieldB, 10) + def test_instance_set_fields(self): + test = autoclass('org.jnius.BasicsTest')() + test.fieldSetZ = True + test.fieldSetB = 127 + test.fieldSetC = ord('k') + test.fieldSetS = 32767 + test.fieldSetI = 2147483467 + test.fieldSetJ = 2147483467 + test.fieldSetF = 1.23456789 + test.fieldSetD = 1.23456789 + + self.assertTrue(test.testFieldSetZ()) + self.assertTrue(test.testFieldSetB()) + self.assertTrue(test.testFieldSetC()) + self.assertTrue(test.testFieldSetS()) + self.assertTrue(test.testFieldSetI()) + self.assertTrue(test.testFieldSetJ()) + self.assertTrue(test.testFieldSetF()) + self.assertTrue(test.testFieldSetD()) + def test_instances_methods_array(self): test = autoclass('org.jnius.BasicsTest')() self.assertEquals(test.methodArrayZ(), [True] * 3)