Chromium Code Reviews| Index: third_party/WebKit/Source/core/dom/custom/CustomElementReactionQueueTest.cpp |
| diff --git a/third_party/WebKit/Source/core/dom/custom/CustomElementReactionQueueTest.cpp b/third_party/WebKit/Source/core/dom/custom/CustomElementReactionQueueTest.cpp |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..41ee46d2818b42d7c9ee9c2da08546fea616765a |
| --- /dev/null |
| +++ b/third_party/WebKit/Source/core/dom/custom/CustomElementReactionQueueTest.cpp |
| @@ -0,0 +1,146 @@ |
| +// Copyright 2016 The Chromium Authors. All rights reserved. |
| +// Use of this source code is governed by a BSD-style license that can be |
| +// found in the LICENSE file. |
| + |
| +#include "core/dom/custom/CustomElementReactionQueue.h" |
| + |
| +#include "core/dom/custom/CustomElementReaction.h" |
| +#include "testing/gtest/include/gtest/gtest.h" |
| +#include "wtf/text/AtomicString.h" |
| +#include <initializer_list> |
| +#include <vector> |
| + |
| +namespace blink { |
| + |
| +class Command : public GarbageCollectedFinalized<Command> { |
| + WTF_MAKE_NONCOPYABLE(Command); |
| +public: |
| + Command() { } |
| + virtual ~Command() { } |
| + DEFINE_INLINE_VIRTUAL_TRACE() { } |
| + virtual void run(Element*) = 0; |
| +}; |
| + |
| +class Log : public Command { |
| + WTF_MAKE_NONCOPYABLE(Log); |
| +public: |
| + Log(char what, std::vector<char>& where) : m_what(what), m_where(where) { } |
| + virtual ~Log() { } |
| + void run(Element*) override { m_where.push_back(m_what); } |
| +private: |
| + char m_what; |
| + std::vector<char>& m_where; |
| +}; |
| + |
| +class Recurse : public Command { |
| + WTF_MAKE_NONCOPYABLE(Recurse); |
| +public: |
| + Recurse(CustomElementReactionQueue* queue) : m_queue(queue) { } |
| + virtual ~Recurse() { } |
| + DEFINE_INLINE_VIRTUAL_TRACE() |
| + { |
| + Command::trace(visitor); |
| + visitor->trace(m_queue); |
| + } |
| + void run(Element* element) override { m_queue->invokeReactions(element); } |
| +private: |
| + Member<CustomElementReactionQueue> m_queue; |
| +}; |
| + |
| +class Enqueue : public Command { |
| + WTF_MAKE_NONCOPYABLE(Enqueue); |
| +public: |
| + Enqueue(CustomElementReactionQueue* queue, CustomElementReaction* reaction) |
| + : m_queue(queue) |
| + , m_reaction(reaction) |
| + { |
| + } |
| + virtual ~Enqueue() { } |
| + DEFINE_INLINE_VIRTUAL_TRACE() |
| + { |
| + Command::trace(visitor); |
| + visitor->trace(m_queue); |
| + visitor->trace(m_reaction); |
| + } |
| + void run(Element*) override |
| + { |
| + m_queue->add(m_reaction); |
| + } |
| +private: |
| + Member<CustomElementReactionQueue> m_queue; |
| + Member<CustomElementReaction> m_reaction; |
| +}; |
| + |
| +class TestReaction : public CustomElementReaction { |
| + WTF_MAKE_NONCOPYABLE(TestReaction); |
| +public: |
| + TestReaction(std::initializer_list<Command*> commands) |
| + { |
| + // TODO(dominicc): Simply pass the initializer list when |
| + // HeapVector supports initializer lists like Vector. |
| + for (auto& command : commands) |
| + m_commands.append(command); |
| + } |
| + virtual ~TestReaction() = default; |
| + DEFINE_INLINE_VIRTUAL_TRACE() |
| + { |
| + CustomElementReaction::trace(visitor); |
| + visitor->trace(m_commands); |
| + } |
| + void invoke(Element* element) override |
| + { |
| + for (auto& command : m_commands) |
| + command->run(element); |
| + } |
| + |
| +private: |
| + HeapVector<Member<Command>> m_commands; |
| +}; |
| + |
| +TEST(CustomElementReactionQueueTest, invokeReactions_one) |
| +{ |
| + std::vector<char> log; |
| + CustomElementReactionQueue* queue = new CustomElementReactionQueue(); |
| + queue->add(new TestReaction({new Log('a', log)})); |
| + queue->invokeReactions(nullptr); |
| + EXPECT_EQ(log, std::vector<char>({'a'})) |
| + << "the reaction should have been invoked"; |
| +} |
| + |
| +TEST(CustomElementReactionQueueTest, invokeReactions_many) |
| +{ |
| + std::vector<char> log; |
| + CustomElementReactionQueue* queue = new CustomElementReactionQueue(); |
| + queue->add(new TestReaction({new Log('a', log)})); |
| + queue->add(new TestReaction({new Log('b', log)})); |
| + queue->add(new TestReaction({new Log('c', log)})); |
| + queue->invokeReactions(nullptr); |
| + EXPECT_EQ(log, std::vector<char>({'a', 'b', 'c'})) |
| + << "the reaction should have been invoked"; |
| +} |
| + |
| +TEST(CustomElementReactionQueueTest, invokeReactions_recursive) |
| +{ |
| + std::vector<char> log; |
| + CustomElementReactionQueue* queue = new CustomElementReactionQueue(); |
| + |
| + CustomElementReaction* third = new TestReaction({ |
| + new Log('c', log), |
| + new Recurse(queue)}); // "Empty" recursion |
| + |
| + CustomElementReaction* second = new TestReaction({ |
| + new Log('b', log), |
| + new Enqueue(queue, third)}); // Unwinds one level of recursion |
| + |
| + CustomElementReaction* first = new TestReaction({ |
| + new Log('a', log), |
| + new Enqueue(queue, second), |
| + new Recurse(queue)}); // Non-empty recursion |
| + |
| + queue->add(first); |
| + queue->invokeReactions(nullptr); |
| + EXPECT_EQ(log, std::vector<char>({'a', 'b', 'c'})) |
|
yosin_UTC9
2016/06/01 02:14:01
Wow, gtest's *_EQ(expect, actual) is now historica
|
| + << "the reactions should have been invoked"; |
| +} |
| + |
| +} // namespace blink |