Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Remove usage of the getTag and setTag methods #1740

Open
wants to merge 10 commits into
base: georgie
Choose a base branch
from
Open
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -62,6 +62,8 @@
## 1.5.20

### Enhancements
* Removed usage of deprecated APIs from the framework and examples. Including the following:
- Removed usage of `WComponents::getTag` and `WComponents::setTag` from `WDefinitionList`.

* Removed usage of deprecated APIs from the framework and examples. Including the following:
- Replaced usage of `WBeanContainer` with `WContainer`. #1733
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,9 @@

import com.github.bordertech.wcomponents.util.Duplet;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.stream.Collectors;

/**
* <p>
Expand All @@ -16,6 +16,8 @@
public class WDefinitionList extends AbstractNamingContextContainer implements AjaxTarget,
SubordinateTarget, Marginable {

private static final String TERM_ATTRIBUTE = "WDefinitionList.term";
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Even though the change now sets the term on the tag and the attribute, I think planning ahead for getTag() being removed, it would still be good for the TERM_ATTRIBUTE to be made public and explain how they will need to access the term via the attribute in the change log.

It would also be good to include how to use the attribute in the javadoc of WDefinitionList.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I don't understand why TERM_ATTRIBUTE should be made public. It should be considered an implementation detail. To add a term you can use addTerm(String, WComponent...) to get all of the terms you can use getTerms() or to get a specific term use getComponentsForTerm(String).

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I could envision a completely different internal implementation where WContainers were used as buckets for the components of each term, and each WComponent bucket, which is unknown to the user, has an attribute with the name of the term. In fact this might be a better implementation.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You are right. The whole implementation of WDefinitionList smells as it originally hijacked the getTag method from the template logic and using the attribute is just as smelly. It might not even be worth doing this change until we consider how it could be refactored and deleting getTag is way off in the future anyway. Let me know what you think.


/**
* The layout options.
*/
Expand Down Expand Up @@ -100,24 +102,27 @@ public Margin getMargin() {
* @param data the term data.
*/
public void addTerm(final String term, final WComponent... data) {
for (WComponent component : data) {
if (component != null) {
content.add(component, term);
}
}

// If the term doesn't exist, we may need to add a dummy component
if (getComponentsForTerm(term).isEmpty()) {
content.add(new DefaultWComponent(), term);
}
getOrAddTermContainer(term)
.ifPresent(container -> {
for (WComponent datum : data) {
datum.setTag(term);
container.add(datum);
}
});
}

/**
* {@inheritDoc}
*/
@Override
public void remove(final WComponent child) {
content.remove(child);
getTermContainers()
.stream()
.forEach(container -> {
if (container.getChildren().contains(child)) {
container.remove(child);
}
});
}

/**
Expand All @@ -126,9 +131,7 @@ public void remove(final WComponent child) {
* @param term the term to remove.
*/
public void removeTerm(final String term) {
for (WComponent child : getComponentsForTerm(term)) {
content.remove(child);
}
getTermContainer(term).ifPresent(container -> content.remove(container));
}

/**
Expand All @@ -137,53 +140,42 @@ public void removeTerm(final String term) {
* @return a list of this definition list's children grouped by their terms.
*/
public List<Duplet<String, ArrayList<WComponent>>> getTerms() {
Map<String, Duplet<String, ArrayList<WComponent>>> componentsByTerm = new HashMap<>();
List<Duplet<String, ArrayList<WComponent>>> result = new ArrayList<>();
return getTermContainers()
.stream()
.map(container ->
new Duplet<>((String) container.getAttribute(TERM_ATTRIBUTE),
new ArrayList<>(container.getChildren())))
.collect(Collectors.toList());
}

List<WComponent> childList = content.getComponentModel().getChildren();
private List<WContainer> getTermContainers() {
return content
.getChildren()
.stream()
.filter(child -> (child instanceof WContainer) && ((WContainer) child).getAttribute(TERM_ATTRIBUTE) != null)
.map(child -> (WContainer) child)
.collect(Collectors.toList());
}

if (childList != null) {
for (int i = 0; i < childList.size(); i++) {
WComponent child = childList.get(i);
String term = child.getTag();
private Optional<WContainer> getOrAddTermContainer(final String term) {

Duplet<String, ArrayList<WComponent>> termComponents = componentsByTerm.get(term);
final Optional<WContainer> container = getTermContainer(term);

if (termComponents == null) {
termComponents = new Duplet<>(term,
new ArrayList<WComponent>());
componentsByTerm.put(term, termComponents);
result.add(termComponents);
}

termComponents.getSecond().add(child);
}
if (term != null && !container.isPresent()) {
WContainer newContainer = new WContainer();
newContainer.setAttribute(TERM_ATTRIBUTE, term);
content.add(newContainer);
return Optional.of(newContainer);
}

return result;
return container;
}

/**
* Retrieves the components for the given term.
*
* @param term the term of the children to be retrieved.
* @return the child components for the given term, may be empty.
*/
private List<WComponent> getComponentsForTerm(final String term) {
List<WComponent> childList = content.getComponentModel().getChildren();
List<WComponent> result = new ArrayList<>();

if (childList != null) {
for (int i = 0; i < childList.size(); i++) {
WComponent child = childList.get(i);

if (term.equals(child.getTag())) {
result.add(child);
}
}
}

return result;
private Optional<WContainer> getTermContainer(final String term) {
return getTermContainers()
.stream()
.filter(container -> term != null && term.equals(container.getAttribute(TERM_ATTRIBUTE)))
.findFirst();
}

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -46,15 +46,21 @@ public void testAddTerm() {
WComponent term2data3 = new DefaultWComponent();

// Test addition of term with no component
Assert.assertTrue("There should not be any terms", list.getTerms().isEmpty());

list.addTerm(term1);
Container termContainer = (Container) list.getChildAt(0);
Assert.assertEquals("Term1 should have been added", 1, termContainer.getChildCount());
Assert.assertEquals("Incorrect value for Term1", term1, termContainer.getChildAt(0).getTag());
Assert.assertEquals("There should have been 1 term added", 1, list.getTerms().size());
Assert.assertEquals("The first term's name is incorrect", term1, list.getTerms().get(0).getFirst());
Assert.assertTrue("The first term should not have any components", list.getTerms().get(0).getSecond().isEmpty());

// Test addition of term with multiple components
list.addTerm(term2, term2data1, term2data2);
list.addTerm(term2, term2data3);

Assert.assertEquals("There should have been 2 terms added", 2, list.getTerms().size());
Assert.assertEquals("The second term's name is incorrect", term2, list.getTerms().get(1).getFirst());
Assert.assertEquals("The second term should have 3 components", 3, list.getTerms().get(1).getSecond().size());

Assert.assertEquals("Incorrect term for component 1", term2, term2data1.getTag());
Assert.assertEquals("Incorrect term for component 2", term2, term2data2.getTag());
Assert.assertEquals("Incorrect term for component 3", term2, term2data3.getTag());
Expand All @@ -72,12 +78,30 @@ public void testRemoveTerm() {

list.addTerm(term1);
list.addTerm(term2, term2data1, term2data2);
Container termContainer = (Container) list.getChildAt(0);

Assert.assertEquals("Incorrect term data", 3, termContainer.getChildren().size());
Assert.assertEquals("There should have been 2 terms added", 2, list.getTerms().size());

list.removeTerm(term2);
Assert.assertEquals("Incorrect term data", 1, termContainer.getChildCount());
Assert.assertEquals("Incorrect value for Term1", term1, termContainer.getChildAt(0).getTag());
Assert.assertEquals("There should be only 1 term remaining", 1, list.getTerms().size());

Assert.assertEquals("Incorrect tag for component 1", term2, term2data1.getTag());
Assert.assertEquals("Incorrect tag for component 2", term2, term2data2.getTag());
}

@Test
public void testRemoveComponent() {
WDefinitionList list = new WDefinitionList();

String term1 = "WDefinitionList_Test.testRemoveComponent.term1";

WComponent data1 = new DefaultWComponent();
WComponent data2 = new DefaultWComponent();

list.addTerm(term1, data1, data2);
Assert.assertEquals("There should have been 1 terms added", 1, list.getTerms().size());
Assert.assertEquals("There should have been 2 components added", 2, list.getTerms().get(0).getSecond().size());

list.remove(data2);

Assert.assertEquals("There should be only 1 component remaining", 1, list.getTerms().get(0).getSecond().size());
}
}