1
0
Fork 0
mirror of https://github.com/ruby-opencv/ruby-opencv synced 2023-03-27 23:22:12 -04:00
ruby-opencv/ext/opencv/point.cpp
2016-04-03 05:19:48 +09:00

167 lines
4.1 KiB
C++

#include "ruby.h"
#include "opencv2/core.hpp"
#include "point.hpp"
namespace rubyopencv {
namespace Point {
void free_point(void* ptr);
size_t memsize_point(const void* ptr);
VALUE rb_klass = Qnil;
rb_data_type_t opencv_point_type = {
"Point",
{ 0, free_point, memsize_point, 0 },
0,
0,
0
};
size_t memsize_point(const void* ptr) {
return sizeof(cv::Point);
}
cv::Point* obj2point(VALUE obj) {
cv::Point* ptr = NULL;
TypedData_Get_Struct(obj, cv::Point, &opencv_point_type, ptr);
return ptr;
}
VALUE point2obj(cv::Point* ptr) {
return TypedData_Wrap_Struct(rb_klass, &opencv_point_type, (void*)ptr);
}
cv::Point conpatible_obj2point(VALUE obj) {
if (rb_respond_to(obj, rb_intern("x")) && rb_respond_to(obj, rb_intern("y"))) {
return cv::Point(NUM2INT(rb_funcall(obj, rb_intern("x"), 0)),
NUM2INT(rb_funcall(obj, rb_intern("y"), 0)));
}
rb_raise(rb_eArgError, "x y");
}
void free_point(void* ptr) {
xfree(ptr);
}
VALUE rb_allocate(VALUE klass) {
cv::Point* ptr = NULL;
return TypedData_Make_Struct(klass, cv::Point, &opencv_point_type, ptr);
}
/*
* call-seq:
* new -> OpenCV::Point.new(0, 0)
* new(obj) -> OpenCV::Point.new(obj.x.to_i, obj.y.to_i)
* new(x, y)
*
* Create new 2D-coordinate, (x, y). It is dropped below the decimal point.
*
* new() is same as new(0, 0)
*
* new(obj) is same as new(obj.x.to_i, obj.y.to_i)
*/
VALUE rb_initialize(int argc, VALUE *argv, VALUE self) {
VALUE v1, v2;
rb_scan_args(argc, argv, "02", &v1, &v2);
cv::Point* selfptr = obj2point(self);
switch (argc) {
case 0:
selfptr->x = 0;
selfptr->y = 0;
break;
case 1: {
// cv::Point point = VALUE_TO_CVPOINT(argv[0]);
// selfptr->x = point.x;
// selfptr->y = point.y;
break;
}
case 2:
selfptr->x = NUM2INT(v1);
selfptr->y = NUM2INT(v2);
break;
default:
rb_raise(rb_eArgError, "wrong number of arguments (%d for 0..2)", argc);
break;
}
return self;
}
/*
* Return parameter on x-axis.
*/
VALUE rb_x(VALUE self) {
return INT2NUM(obj2point(self)->x);
}
/*
* call-seq:
* x = val
*/
VALUE rb_set_x(VALUE self, VALUE x) {
obj2point(self)->x = NUM2INT(x);
return self;
}
/*
* Return parameter on y-axis.
*/
VALUE rb_y(VALUE self) {
return INT2NUM(obj2point(self)->y);
}
/*
* call-seq:
* y = val
*/
VALUE rb_set_y(VALUE self, VALUE y) {
obj2point(self)->y = NUM2INT(y);
return self;
}
/*
* call-seq:
* to_s -> string
*
* Return x and y by String.
*/
VALUE rb_to_s(VALUE self) {
const int i = 4;
VALUE str[i];
str[0] = rb_str_new2("<%s:(%d,%d)>");
str[1] = rb_str_new2(rb_class2name(CLASS_OF(self)));
str[2] = rb_x(self);
str[3] = rb_y(self);
return rb_f_sprintf(i, str);
}
/*
* call-seq:
* to_a -> [x, y]
*
* Return x and y by Array.
*/
VALUE rb_to_a(VALUE self) {
cv::Point* selfptr = obj2point(self);
return rb_ary_new3(2, INT2NUM(selfptr->x), INT2NUM(selfptr->y));
}
void init() {
VALUE opencv = rb_define_module("OpenCV");
rb_klass = rb_define_class_under(opencv, "Point", rb_cData);
rb_define_alloc_func(rb_klass, rb_allocate);
// rb_define_singleton_method(rb_klass, "compatible?", RUBY_METHOD_FUNC(rb_compatible_q), 1);
rb_define_method(rb_klass, "initialize", RUBY_METHOD_FUNC(rb_initialize), -1);
rb_define_method(rb_klass, "x", RUBY_METHOD_FUNC(rb_x), 0);
rb_define_method(rb_klass, "x=", RUBY_METHOD_FUNC(rb_set_x), 1);
rb_define_method(rb_klass, "y", RUBY_METHOD_FUNC(rb_y), 0);
rb_define_method(rb_klass, "y=", RUBY_METHOD_FUNC(rb_set_y), 1);
rb_define_method(rb_klass, "to_s", RUBY_METHOD_FUNC(rb_to_s), 0);
rb_define_method(rb_klass, "to_a", RUBY_METHOD_FUNC(rb_to_a), 0);
// rb_define_alias(rb_klass, "to_a", "to_ary");
}
}
}