diff options
| author | take_cheeze <[email protected]> | 2014-05-02 23:20:48 +0900 |
|---|---|---|
| committer | take_cheeze <[email protected]> | 2014-05-02 23:20:48 +0900 |
| commit | 44dc05f12a06e329119b6bf5606e4836b653c48f (patch) | |
| tree | 1dcf10c618c387c61d956e75e5e017495b7ce6d2 /src | |
| parent | d91c9a9ea41349b2455ed89e4fbd46de0374c53b (diff) | |
| download | mruby-44dc05f12a06e329119b6bf5606e4836b653c48f.tar.gz mruby-44dc05f12a06e329119b6bf5606e4836b653c48f.zip | |
Implement Struct#values_at and Array#values_at .
Add API `mrb_get_values_at()` to mruby/range.h .
Diffstat (limited to 'src')
| -rw-r--r-- | src/range.c | 28 |
1 files changed, 28 insertions, 0 deletions
diff --git a/src/range.c b/src/range.c index b59b234ef..340d64ec1 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")) @@ -395,6 +396,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) { |
