//////////////////////////////////////////////////////////////////////////// // // 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. // //////////////////////////////////////////////////////////////////////////// #pragma once #include "js_collection.hpp" #include "js_object.hpp" #include "js_results.hpp" #include "js_util.hpp" #include "shared_realm.hpp" #include "list.hpp" #include "object_accessor.hpp" #include "parser.hpp" #include "query_builder.hpp" #include using RJSAccessor = realm::NativeAccessor; namespace realm { namespace js { template struct List { using ContextType = typename T::Context; using ObjectType = typename T::Object; using ValueType = typename T::Value; using ReturnType = typename T::Return; static void GetLength(ContextType ctx, ObjectType thisObject, ReturnType &ret); static void GetIndex(ContextType ctx, ObjectType thisObject, size_t index, ReturnType &ret); static void SetIndex(ContextType ctx, ObjectType thisObject, size_t index, ValueType value); static void Push(ContextType ctx, ObjectType thisObject, size_t argCount, const ValueType args[], ReturnType &ret); static void Pop(ContextType ctx, ObjectType thisObject, size_t argCount, const ValueType args[], ReturnType &ret); static void Unshift(ContextType ctx, ObjectType thisObject, size_t argCount, const ValueType args[], ReturnType &ret); static void Shift(ContextType ctx, ObjectType thisObject, size_t argCount, const ValueType args[], ReturnType &ret); static void Splice(ContextType ctx, ObjectType thisObject, size_t argCount, const ValueType args[], ReturnType &ret); static void StaticResults(ContextType ctx, ObjectType thisObject, size_t argCount, const ValueType args[], ReturnType &ret); static void Filtered(ContextType ctx, ObjectType thisObject, size_t argCount, const ValueType args[], ReturnType &ret); static void Sorted(ContextType ctx, ObjectType thisObject, size_t argCount, const ValueType args[], ReturnType &ret); }; template void List::GetLength(ContextType ctx, ObjectType object, ReturnType &ret) { realm::List *list = RJSGetInternal(object); RJSSetReturnNumber(ctx, ret, list->size()); } template void List::GetIndex(ContextType ctx, ObjectType object, size_t index, ReturnType &ret) { realm::List *list = RJSGetInternal(object); ret = RJSObjectCreate(ctx, Object(list->get_realm(), list->get_object_schema(), list->get(index))); } template void List::SetIndex(ContextType ctx, ObjectType object, size_t index, ValueType value) { realm::List *list = RJSGetInternal(object); list->set(ctx, value, index); } template void List::Push(ContextType ctx, ObjectType thisObject, size_t argumentCount, const ValueType arguments[], ReturnType &returnObject) { realm::List *list = RJSGetInternal(thisObject); RJSValidateArgumentCountIsAtLeast(argumentCount, 1); for (size_t i = 0; i < argumentCount; i++) { list->add(ctx, arguments[i]); } RJSSetReturnNumber(ctx, returnObject, list->size()); } template void List::Pop(ContextType ctx, ObjectType thisObject, size_t argumentCount, const ValueType arguments[], ReturnType &returnObject) { realm::List *list = RJSGetInternal(thisObject); RJSValidateArgumentCount(argumentCount, 0); size_t size = list->size(); if (size == 0) { list->verify_in_transaction(); RJSSetReturnUndefined(ctx, returnObject); } else { size_t index = size - 1; returnObject = RJSObjectCreate(ctx, Object(list->get_realm(), list->get_object_schema(), list->get(index))); list->remove(index); } } template void List::Unshift(ContextType ctx, ObjectType thisObject, size_t argumentCount, const ValueType arguments[], ReturnType &returnObject) { realm::List *list = RJSGetInternal(thisObject); RJSValidateArgumentCountIsAtLeast(argumentCount, 1); for (size_t i = 0; i < argumentCount; i++) { list->insert(ctx, arguments[i], i); } RJSSetReturnNumber(ctx, returnObject, list->size()); } template void List::Shift(ContextType ctx, ObjectType thisObject, size_t argumentCount, const ValueType arguments[], ReturnType &returnObject) { realm::List *list = RJSGetInternal(thisObject); RJSValidateArgumentCount(argumentCount, 0); if (list->size() == 0) { list->verify_in_transaction(); RJSSetReturnUndefined(ctx, returnObject); } else { returnObject = RJSObjectCreate(ctx, Object(list->get_realm(), list->get_object_schema(), list->get(0))); list->remove(0); } } template void List::Splice(ContextType ctx, ObjectType thisObject, size_t argumentCount, const ValueType arguments[], ReturnType &returnObject) { realm::List *list = RJSGetInternal(thisObject); size_t size = list->size(); RJSValidateArgumentCountIsAtLeast(argumentCount, 1); long index = std::min(RJSValidatedValueToNumber(ctx, arguments[0]), size); if (index < 0) { index = std::max(size + index, 0); } long remove; if (argumentCount < 2) { remove = size - index; } else { remove = std::max(RJSValidatedValueToNumber(ctx, arguments[1]), 0); remove = std::min(remove, size - index); } std::vector removedObjects(remove); for (size_t i = 0; i < remove; i++) { removedObjects[i] = RJSObjectCreate(ctx, Object(list->get_realm(), list->get_object_schema(), list->get(index))); list->remove(index); } for (size_t i = 2; i < argumentCount; i++) { list->insert(ctx, arguments[i], index + i - 2); } RJSSetReturnArray(ctx, remove, removedObjects.data(), returnObject); } template void List::StaticResults(ContextType ctx, ObjectType thisObject, size_t argumentCount, const ValueType arguments[], ReturnType &returnObject) { realm::List *list = RJSGetInternal(thisObject); RJSValidateArgumentCount(argumentCount, 0); returnObject = RJSResultsCreate(ctx, list->get_realm(), list->get_object_schema(), std::move(list->get_query()), false); } template void List::Filtered(ContextType ctx, ObjectType thisObject, size_t argumentCount, const ValueType arguments[], ReturnType &returnObject) { realm::List *list = RJSGetInternal(thisObject); RJSValidateArgumentCountIsAtLeast(argumentCount, 1); SharedRealm sharedRealm = *RJSGetInternal(thisObject); returnObject = RJSResultsCreateFiltered(ctx, sharedRealm, list->get_object_schema(), std::move(list->get_query()), argumentCount, arguments); } template void List::Sorted(ContextType ctx, ObjectType thisObject, size_t argumentCount, const ValueType arguments[], ReturnType &returnObject) { realm::List *list = RJSGetInternal(thisObject); RJSValidateArgumentRange(argumentCount, 1, 2); SharedRealm sharedRealm = *RJSGetInternal(thisObject); returnObject = RJSResultsCreateSorted(ctx, sharedRealm, list->get_object_schema(), std::move(list->get_query()), argumentCount, arguments); } } }