realm-js/src/js_list.hpp

235 lines
8.1 KiB
C++
Raw Normal View History

2016-02-18 11:59:34 -08:00
////////////////////////////////////////////////////////////////////////////
//
// Copyright 2016 Realm Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
////////////////////////////////////////////////////////////////////////////
2015-08-13 09:12:48 -07:00
2015-12-01 14:52:38 -08:00
#pragma once
2016-04-12 14:42:05 -07:00
#include "js_collection.hpp"
#include "js_realm_object.hpp"
2016-04-12 14:42:05 -07:00
#include "js_results.hpp"
#include "js_types.hpp"
2015-12-01 14:52:38 -08:00
#include "js_util.hpp"
2016-03-28 13:21:36 -07:00
2015-12-01 14:52:38 -08:00
#include "shared_realm.hpp"
#include "list.hpp"
2016-03-28 13:21:36 -07:00
#include "parser.hpp"
#include "query_builder.hpp"
namespace realm {
namespace js {
2016-03-28 13:21:36 -07:00
template<typename T>
class List {
2016-04-18 12:15:00 -07:00
using ContextType = typename T::Context;
using ObjectType = typename T::Object;
using ValueType = typename T::Value;
using Object = js::Object<T>;
using Value = js::Value<T>;
using ReturnValue = js::ReturnValue<T>;
public:
2016-04-18 12:15:00 -07:00
static ObjectType create_instance(ContextType, realm::List &);
// properties
2016-04-18 12:15:00 -07:00
static void get_length(ContextType, ObjectType, ReturnValue &);
static void get_index(ContextType, ObjectType, uint32_t, ReturnValue &);
static bool set_index(ContextType, ObjectType, uint32_t, ValueType);
// methods
2016-04-18 12:15:00 -07:00
static void push(ContextType, ObjectType, size_t, const ValueType[], ReturnValue &);
static void pop(ContextType, ObjectType, size_t, const ValueType[], ReturnValue &);
static void unshift(ContextType, ObjectType, size_t, const ValueType[], ReturnValue &);
static void shift(ContextType, ObjectType, size_t, const ValueType[], ReturnValue &);
static void splice(ContextType, ObjectType, size_t, const ValueType[], ReturnValue &);
static void snapshot(ContextType, ObjectType, size_t, const ValueType[], ReturnValue &);
static void filtered(ContextType, ObjectType, size_t, const ValueType[], ReturnValue &);
static void sorted(ContextType, ObjectType, size_t, const ValueType[], ReturnValue &);
};
2016-03-30 14:18:44 -07:00
template<typename T>
2016-04-16 16:46:16 -07:00
struct ListClass : ClassDefinition<T, realm::List, CollectionClass<T>> {
using List = js::List<T>;
std::string const name = "List";
MethodMap<T> const methods = {
{"push", wrap<List::push>},
{"pop", wrap<List::pop>},
{"unshift", wrap<List::unshift>},
{"shift", wrap<List::shift>},
{"splice", wrap<List::splice>},
{"snapshot", wrap<List::snapshot>},
{"filtered", wrap<List::filtered>},
{"sorted", wrap<List::sorted>},
};
PropertyMap<T> const properties = {
{"length", {wrap<List::get_length>, nullptr}},
};
IndexPropertyType<T> const index_accessor = {wrap<List::get_index>, wrap<List::set_index>};
};
template<typename T>
2016-04-18 12:15:00 -07:00
typename T::Object List<T>::create_instance(ContextType ctx, realm::List &list) {
return create_object<T, ListClass<T>>(ctx, new realm::List(list));
2016-03-30 14:18:44 -07:00
}
2016-03-30 14:18:44 -07:00
template<typename T>
2016-04-18 12:15:00 -07:00
void List<T>::get_length(ContextType ctx, ObjectType object, ReturnValue &return_value) {
auto list = get_internal<T, ListClass<T>>(object);
return_value.set((uint32_t)list->size());
2016-03-30 14:18:44 -07:00
}
template<typename T>
2016-04-18 12:15:00 -07:00
void List<T>::get_index(ContextType ctx, ObjectType object, uint32_t index, ReturnValue &return_value) {
auto list = get_internal<T, ListClass<T>>(object);
auto realm_object = realm::Object(list->get_realm(), list->get_object_schema(), list->get(index));
2016-04-14 11:19:01 -07:00
return_value.set(RealmObject<T>::create_instance(ctx, realm_object));
}
template<typename T>
2016-04-18 12:15:00 -07:00
bool List<T>::set_index(ContextType ctx, ObjectType object, uint32_t index, ValueType value) {
auto list = get_internal<T, ListClass<T>>(object);
2016-03-30 14:18:44 -07:00
list->set(ctx, value, index);
return true;
2016-03-30 14:18:44 -07:00
}
template<typename T>
2016-04-18 12:15:00 -07:00
void List<T>::push(ContextType ctx, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) {
validate_argument_count_at_least(argc, 1);
auto list = get_internal<T, ListClass<T>>(this_object);
for (size_t i = 0; i < argc; i++) {
list->add(ctx, arguments[i]);
2016-03-28 13:21:36 -07:00
}
return_value.set((uint32_t)list->size());
2016-03-28 13:21:36 -07:00
}
template<typename T>
2016-04-18 12:15:00 -07:00
void List<T>::pop(ContextType ctx, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) {
validate_argument_count(argc, 0);
auto list = get_internal<T, ListClass<T>>(this_object);
size_t size = list->size();
if (size == 0) {
list->verify_in_transaction();
return_value.set_undefined();
2016-03-28 13:21:36 -07:00
}
else {
size_t index = size - 1;
auto realm_object = realm::Object(list->get_realm(), list->get_object_schema(), list->get(index));
2016-04-14 11:19:01 -07:00
return_value.set(RealmObject<T>::create_instance(ctx, realm_object));
list->remove(index);
2016-03-28 13:21:36 -07:00
}
}
template<typename T>
2016-04-18 12:15:00 -07:00
void List<T>::unshift(ContextType ctx, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) {
validate_argument_count_at_least(argc, 1);
auto list = get_internal<T, ListClass<T>>(this_object);
for (size_t i = 0; i < argc; i++) {
list->insert(ctx, arguments[i], i);
2016-03-28 13:21:36 -07:00
}
return_value.set((uint32_t)list->size());
2016-03-28 13:21:36 -07:00
}
template<typename T>
2016-04-18 12:15:00 -07:00
void List<T>::shift(ContextType ctx, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) {
validate_argument_count(argc, 0);
auto list = get_internal<T, ListClass<T>>(this_object);
if (list->size() == 0) {
list->verify_in_transaction();
return_value.set_undefined();
2016-03-28 13:21:36 -07:00
}
else {
auto realm_object = realm::Object(list->get_realm(), list->get_object_schema(), list->get(0));
2016-04-14 11:19:01 -07:00
return_value.set(RealmObject<T>::create_instance(ctx, realm_object));
list->remove(0);
2016-03-28 13:21:36 -07:00
}
}
template<typename T>
2016-04-18 12:15:00 -07:00
void List<T>::splice(ContextType ctx, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) {
validate_argument_count_at_least(argc, 1);
auto list = get_internal<T, ListClass<T>>(this_object);
size_t size = list->size();
2016-04-12 14:42:05 -07:00
long index = std::min<long>(Value::to_number(ctx, arguments[0]), size);
if (index < 0) {
index = std::max<long>(size + index, 0);
}
size_t remove;
if (argc < 2) {
remove = size - index;
}
else {
2016-04-12 14:42:05 -07:00
remove = std::max<long>(Value::to_number(ctx, arguments[1]), 0);
remove = std::min<long>(remove, size - index);
}
2016-04-18 12:15:00 -07:00
std::vector<ValueType> removed_objects;
removed_objects.reserve(remove);
for (size_t i = 0; i < remove; i++) {
auto realm_object = realm::Object(list->get_realm(), list->get_object_schema(), list->get(index));
2016-04-14 11:19:01 -07:00
removed_objects.push_back(RealmObject<T>::create_instance(ctx, realm_object));
list->remove(index);
2016-03-28 13:21:36 -07:00
}
for (size_t i = 2; i < argc; i++) {
list->insert(ctx, arguments[i], index + i - 2);
2016-03-28 13:21:36 -07:00
}
return_value.set(Object::create_array(ctx, removed_objects));
}
2016-03-28 13:21:36 -07:00
template<typename T>
2016-04-18 12:15:00 -07:00
void List<T>::snapshot(ContextType ctx, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) {
validate_argument_count(argc, 0);
auto list = get_internal<T, ListClass<T>>(this_object);
2016-04-14 11:19:01 -07:00
return_value.set(Results<T>::create_instance(ctx, *list, false));
2016-03-28 13:21:36 -07:00
}
template<typename T>
2016-04-18 12:15:00 -07:00
void List<T>::filtered(ContextType ctx, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) {
validate_argument_count_at_least(argc, 1);
auto list = get_internal<T, ListClass<T>>(this_object);
2016-04-12 14:42:05 -07:00
return_value.set(Results<T>::create_filtered(ctx, *list, argc, arguments));
2016-03-28 13:21:36 -07:00
}
2015-08-13 09:12:48 -07:00
template<typename T>
2016-04-18 12:15:00 -07:00
void List<T>::sorted(ContextType ctx, ObjectType this_object, size_t argc, const ValueType arguments[], ReturnValue &return_value) {
validate_argument_count(argc, 1, 2);
auto list = get_internal<T, ListClass<T>>(this_object);
2016-04-12 14:42:05 -07:00
return_value.set(Results<T>::create_sorted(ctx, *list, argc, arguments));
}
} // js
} // realm