summaryrefslogtreecommitdiffhomepage
path: root/src/range.c
diff options
context:
space:
mode:
authorYukihiro "Matz" Matsumoto <[email protected]>2014-05-03 21:13:03 +0900
committerYukihiro "Matz" Matsumoto <[email protected]>2014-05-03 21:13:03 +0900
commitc3aa436e2c9de93a21750c92886756c5f1b56ab5 (patch)
tree31b5b64ea76aa5a82242a9d27da72671831c0ed2 /src/range.c
parentc2bd0d33eadfd42db80a908781f860079f0fd46b (diff)
parent44dc05f12a06e329119b6bf5606e4836b653c48f (diff)
downloadmruby-c3aa436e2c9de93a21750c92886756c5f1b56ab5.tar.gz
mruby-c3aa436e2c9de93a21750c92886756c5f1b56ab5.zip
Merge branch 'values_at' of https://github.com/take-cheeze/mruby into take-cheeze-values_at
Diffstat (limited to 'src/range.c')
-rw-r--r--src/range.c28
1 files changed, 28 insertions, 0 deletions
diff --git a/src/range.c b/src/range.c
index 4ab6708e1..627b572d4 100644
--- a/src/range.c
+++ b/src/range.c
@@ -8,6 +8,7 @@
#include "mruby/class.h"
#include "mruby/range.h"
#include "mruby/string.h"
+#include "mruby/array.h"
#define RANGE_CLASS (mrb_class_get(mrb, "Range"))
@@ -371,6 +372,33 @@ range_initialize_copy(mrb_state *mrb, mrb_value copy)
return copy;
}
+mrb_value
+mrb_get_values_at(mrb_state *mrb, mrb_value obj, mrb_int olen, mrb_int argc, const mrb_value *argv, mrb_value (*func)(mrb_state*, mrb_value, mrb_int))
+{
+ mrb_int i, j, beg, len;
+ mrb_value result;
+ result = mrb_ary_new(mrb);
+
+ for (i = 0; i < argc; ++i) {
+ if (mrb_fixnum_p(argv[i])) {
+ mrb_ary_push(mrb, result, func(mrb, obj, mrb_fixnum(argv[i])));
+ } else if (mrb_range_beg_len(mrb, argv[i], &beg, &len, olen)) {
+ mrb_int const end = RARRAY_LEN(obj) < beg + len ? RARRAY_LEN(obj) : beg + len;
+ for (j = beg; j < end; ++j) {
+ mrb_ary_push(mrb, result, func(mrb, obj, j));
+ }
+
+ for (; j < beg + len; ++j) {
+ mrb_ary_push(mrb, result, mrb_nil_value());
+ }
+ } else {
+ mrb_raisef(mrb, E_ARGUMENT_ERROR, "invalid values selector: %S", argv[i]);
+ }
+ }
+
+ return result;
+}
+
void
mrb_init_range(mrb_state *mrb)
{