Ви не можете вибрати більше 25 тем Теми мають розпочинатися з літери або цифри, можуть містити дефіси (-) і не повинні перевищувати 35 символів.

ActionComponentChain.java 5.2KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. /*
  2. * Copyright (c) 2006-2015 DMDirc Developers
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a copy
  5. * of this software and associated documentation files (the "Software"), to deal
  6. * in the Software without restriction, including without limitation the rights
  7. * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  8. * copies of the Software, and to permit persons to whom the Software is
  9. * furnished to do so, subject to the following conditions:
  10. *
  11. * The above copyright notice and this permission notice shall be included in
  12. * all copies or substantial portions of the Software.
  13. *
  14. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  19. * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  20. * SOFTWARE.
  21. */
  22. package com.dmdirc.actions;
  23. import com.dmdirc.Precondition;
  24. import com.dmdirc.interfaces.ActionController;
  25. import com.dmdirc.interfaces.actions.ActionComponent;
  26. import java.util.ArrayList;
  27. import java.util.List;
  28. import static com.google.common.base.Preconditions.checkArgument;
  29. /**
  30. * An action component chain supports chaining of multiple action components together.
  31. */
  32. public class ActionComponentChain implements ActionComponent {
  33. /**
  34. * A list of components in this chain.
  35. */
  36. private final List<ActionComponent> components = new ArrayList<>();
  37. /**
  38. * Creates a new component chain from the specified text representation. Chains are separated
  39. * with full stops (.).
  40. *
  41. * @param source The class that this chain needs to start with
  42. * @param chain The textual representation of the chain
  43. * @param manager The action manager to use to look up components
  44. */
  45. public ActionComponentChain(final Class<?> source, final String chain,
  46. final ActionController manager) {
  47. Class<?> current = source;
  48. for (String componentName : chain.split("\\.")) {
  49. final ActionComponent component = manager.getComponent(componentName);
  50. if (component == null) {
  51. throw new IllegalArgumentException("Component " + componentName
  52. + " not found");
  53. } else if (component.appliesTo().isAssignableFrom(current)) {
  54. components.add(component);
  55. current = component.getType();
  56. } else {
  57. throw new IllegalArgumentException("Component " + componentName
  58. + " cannot be applied to " + current.getName());
  59. }
  60. }
  61. }
  62. @Override
  63. public Object get(final Object arg) {
  64. Object res = arg;
  65. for (ActionComponent component : components) {
  66. if (res == null) {
  67. return null;
  68. }
  69. res = component.get(res);
  70. }
  71. return res;
  72. }
  73. @Precondition("This component chain has one or more components")
  74. @Override
  75. public Class<?> appliesTo() {
  76. checkArgument(!components.isEmpty());
  77. return components.get(0).appliesTo();
  78. }
  79. @Precondition("This component chain has one or more components")
  80. @Override
  81. public Class<?> getType() {
  82. checkArgument(!components.isEmpty());
  83. return components.get(components.size() - 1).getType();
  84. }
  85. @Precondition("This component chain has one or more components")
  86. @Override
  87. public String getName() {
  88. checkArgument(!components.isEmpty());
  89. final StringBuilder name = new StringBuilder();
  90. for (ActionComponent component : components) {
  91. name.append("'s ");
  92. name.append(component.getName());
  93. }
  94. return name.substring(3);
  95. }
  96. @Override
  97. @Precondition("This component chain has one or more components")
  98. public String toString() {
  99. checkArgument(!components.isEmpty());
  100. final StringBuilder name = new StringBuilder();
  101. for (ActionComponent component : components) {
  102. name.append('.');
  103. name.append(component.toString());
  104. }
  105. return name.substring(1);
  106. }
  107. @Override
  108. public String name() {
  109. return toString();
  110. }
  111. /**
  112. * Determines if any components in this chain require a server to have an established connection
  113. * in order to function.
  114. *
  115. * @since 0.6.4
  116. * @return True iff at least one component requires a connection
  117. */
  118. public boolean requiresConnection() {
  119. boolean res = false;
  120. for (ActionComponent component : components) {
  121. try {
  122. final ComponentOptions options = component.getClass()
  123. .getMethod("get", Object.class).getAnnotation(ComponentOptions.class);
  124. if (options != null) {
  125. res |= options.requireConnected();
  126. }
  127. } catch (NoSuchMethodException ex) {
  128. // Do nothing
  129. }
  130. }
  131. return res;
  132. }
  133. }