Skip to contentMethod: lambda$pmFor$5(SimpleDciEntity)
1: /*
2: * *************************************************************************************************************************************************************
3: *
4: * SteelBlue: DCI User Interfaces
5: * http://tidalwave.it/projects/steelblue
6: *
7: * Copyright (C) 2015 - 2025 by Tidalwave s.a.s. (http://tidalwave.it)
8: *
9: * *************************************************************************************************************************************************************
10: *
11: * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License.
12: * You may obtain a copy of the License at
13: *
14: * http://www.apache.org/licenses/LICENSE-2.0
15: *
16: * 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
17: * CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.
18: *
19: * *************************************************************************************************************************************************************
20: *
21: * git clone https://bitbucket.org/tidalwave/steelblue-src
22: * git clone https://github.com/tidalwave-it/steelblue-src
23: *
24: * *************************************************************************************************************************************************************
25: */
26: package it.tidalwave.ui.example.presentation.impl;
27:
28: import jakarta.annotation.Nonnull;
29: import jakarta.annotation.PostConstruct;
30: import java.util.List;
31: import java.nio.file.Path;
32: import java.nio.file.Paths;
33: import org.springframework.stereotype.Component;
34: import it.tidalwave.util.annotation.VisibleForTesting;
35: import it.tidalwave.role.Aggregate;
36: import it.tidalwave.ui.core.BoundProperty;
37: import it.tidalwave.ui.core.message.PowerOnEvent;
38: import it.tidalwave.ui.core.role.Displayable;
39: import it.tidalwave.ui.core.MenuBarControl.MenuPlacement;
40: import it.tidalwave.ui.core.role.PresentationModel;
41: import it.tidalwave.ui.core.role.PresentationModelAggregate;
42: import it.tidalwave.ui.core.role.Selectable;
43: import it.tidalwave.ui.core.role.UserAction;
44: import it.tidalwave.ui.core.role.UserActionProvider;
45: import it.tidalwave.ui.core.role.Visibility;
46: import it.tidalwave.ui.example.model.Dao;
47: import it.tidalwave.ui.example.model.SimpleDciEntity;
48: import it.tidalwave.ui.example.model.SimpleEntity;
49: import it.tidalwave.ui.example.presentation.MainPanelPresentation;
50: import it.tidalwave.ui.example.presentation.MainPanelPresentation.Bindings;
51: import it.tidalwave.ui.example.presentation.MainPanelPresentationControl;
52: import it.tidalwave.messagebus.annotation.ListensTo;
53: import lombok.Getter;
54: import lombok.RequiredArgsConstructor;
55: import static it.tidalwave.util.Parameters.r;
56: import static it.tidalwave.util.ui.UserNotificationWithFeedback.*;
57: import static it.tidalwave.ui.core.role.Presentable._Presentable_;
58: import static it.tidalwave.ui.core.role.spi.PresentationModelCollectors.toCompositePresentationModel;
59:
60: /***************************************************************************************************************************************************************
61: *
62: * @stereotype Control
63: *
64: * @author Fabrizio Giudici
65: *
66: **************************************************************************************************************************************************************/
67: @Component @RequiredArgsConstructor
68: // @SimpleMessageSubscriber
69: public class DefaultMainPanelPresentationControl implements MainPanelPresentationControl
70: {
71: private static final Path USER_HOME = Paths.get(System.getProperty("user.home"));
72:
73: // START SNIPPET: injections
74: @Nonnull
75: private final Dao dao;
76:
77: @Nonnull
78: private final MainPanelPresentation presentation;
79: // END SNIPPET: injections
80:
81: // For each button on the presentation that can do something, a UserAction is provided.
82: // START SNIPPET: userActions
83: @Getter
84: private final UserAction actionButton = UserAction.of(this::onButtonPressed,
85: Displayable.of("Press me"));
86:
87: @Getter
88: private final UserAction actionDialogOk = UserAction.of(this::onButtonDialogOkPressed,
89: Displayable.of("Dialog with ok"));
90:
91: @Getter
92: private final UserAction actionDialogCancelOk = UserAction.of(this::onButtonDialogOkCancelPressed, List.of(
93: Displayable.of("Dialog with ok/cancel"),
94: MenuPlacement.under("Tools")));
95:
96: @Getter
97: private final UserAction actionPickFile = UserAction.of(this::onButtonPickFilePressed, List.of(
98: Displayable.of("Open file..."),
99: MenuPlacement.under("File")));
100:
101: @Getter
102: private final UserAction actionPickDirectory = UserAction.of(this::onButtonPickDirectoryPressed, List.of(
103: Displayable.of("Open directory..."),
104: MenuPlacement.under("File")));
105: // END SNIPPET: userActions
106: // START SNIPPET: bindings
107: private final Bindings bindings = Bindings.builder()
108: .actionButton(actionButton)
109: .actionDialogOk(actionDialogOk)
110: .actionDialogCancelOk(actionDialogCancelOk)
111: .actionPickFile(actionPickFile)
112: .actionPickDirectory(actionPickDirectory)
113: .build();
114: // END SNIPPET: bindings
115:
116: // Then there can be a set of variables that represent the internal state of the control.
117: @VisibleForTesting int status = 1;
118:
119: /***********************************************************************************************************************************************************
120: * At {@link PostConstruct} time the control just performs the binding to the presentation.
121: **********************************************************************************************************************************************************/
122: // START SNIPPET: initialization
123: @PostConstruct
124: @VisibleForTesting void initialize()
125: {
126: presentation.bind(bindings);
127: }
128: // END SNIPPET: initialization
129:
130: /***********************************************************************************************************************************************************
131: * {@inheritDoc}
132: *
133: * This method demonstrates the typical idiom for populating model:
134: *
135: * 1. A dao is called to provide raw model - let's say in form of collections;
136: * 2. Objects in the collection are transformed into PresentationModels.
137: * 3. The PresentationModels are then passed to the presentation.
138: **********************************************************************************************************************************************************/
139: // START SNIPPET: populate
140: @Override
141: public void populate ()
142: {
143: final var entities1 = dao.getSimpleEntities();
144: final var entities2 = dao.getDciEntities();
145: final var files = dao.getFiles();
146: final var pmEnt1 = entities1.stream().map(this::pmFor).collect(toCompositePresentationModel());
147: final var pmEnt2 = entities2.stream().map(this::pmFor).collect(toCompositePresentationModel());
148: final var pmFiles = files.stream()
149: .map(item -> item.as(_Presentable_).createPresentationModel())
150: .collect(toCompositePresentationModel(r(Visibility.INVISIBLE)));
151: presentation.populate(pmEnt1, pmEnt2, pmFiles);
152: }
153: // END SNIPPET: populate
154:
155: /***********************************************************************************************************************************************************
156: * Alternatively to expose methods to a public interface, a pubsub facility can be used. This method is called back at application initialisation.
157: * @param event the 'power on' event
158: **********************************************************************************************************************************************************/
159: // START SNIPPET: onPowerOn
160: @VisibleForTesting void onPowerOn (@ListensTo final PowerOnEvent event)
161: {
162: presentation.bind(bindings);
163: populate();
164: }
165: // END SNIPPET: onPowerOn
166:
167: /***********************************************************************************************************************************************************
168: * Factory method for the PresentationModel of SimpleEntity instances.
169: *
170: * It aggregates a few extra roles into the PresentationModel that are used by the control, such as callbacks
171: * for action associated to the context menu. Also a Displayable role is usually injected to control the rendering
172: * of entities.
173: **********************************************************************************************************************************************************/
174: // START SNIPPET: pmSimpleEntity
175: @Nonnull
176: private PresentationModel pmFor (@Nonnull final SimpleEntity entity)
177: {
178: final Selectable selectable = () -> onSelected(entity);
179: final var action1 = UserAction.of(() -> action1(entity), Displayable.of("Action 1"));
180: final var action2 = UserAction.of(() -> action2(entity), Displayable.of("Action 2"));
181: final var action3 = UserAction.of(() -> action3(entity), Displayable.of("Action 3"));
182: return PresentationModel.of(entity, r(Displayable.of("Item #" + entity.getName()),
183: selectable,
184: UserActionProvider.of(action1, action2, action3)));
185: }
186: // END SNIPPET: pmSimpleEntity
187:
188: /***********************************************************************************************************************************************************
189: * Factory method for the PresentationModel of SimpleDciEntity instances.
190: **********************************************************************************************************************************************************/
191: // START SNIPPET: pmSimpleDciEntity
192: @Nonnull
193: private PresentationModel pmFor (@Nonnull final SimpleDciEntity entity)
194: {
195: // FIXME: column names
196: final Aggregate<PresentationModel> aggregate = PresentationModelAggregate.newInstance()
197: .withPmOf("C1", r(Displayable.of(entity.getName())))
198: .withPmOf("C2", r(Displayable.of("" + entity.getAttribute1())))
199: .withPmOf("C3", r(Displayable.of("" + entity.getAttribute2())));
200: final Selectable selectable = () -> onSelected(entity);
201: final var action1 = UserAction.of(() -> action1(entity), Displayable.of("Action 1"));
202: final var action2 = UserAction.of(() -> action2(entity), Displayable.of("Action 2"));
203: final var action3 = UserAction.of(() -> action3(entity), Displayable.of("Action 3"));
204: // No explicit Displayable here, as the one inside SimpleDciEntity is used.
205: return PresentationModel.of(entity, r(aggregate, selectable, UserActionProvider.of(action1, action2, action3)));
206: }
207: // END SNIPPET: pmSimpleDciEntity
208:
209: // Below simple business methods, as per usual business.
210:
211: /***********************************************************************************************************************************************************
212: *
213: **********************************************************************************************************************************************************/
214: // START SNIPPET: onButtonPressed
215: private void onButtonPressed()
216: {
217: presentation.notify("Button pressed");
218: status++;
219: bindings.textProperty.set(Integer.toString(status));
220: }
221: // END SNIPPET: onButtonPressed
222:
223: /***********************************************************************************************************************************************************
224: *
225: **********************************************************************************************************************************************************/
226: // START SNIPPET: onButtonDialogOkPressed
227: private void onButtonDialogOkPressed()
228: {
229: presentation.notify(notificationWithFeedback()
230: .withCaption("Notification")
231: .withText("Now press the button")
232: .withFeedback(feedback().withOnConfirm(() -> presentation.notify("Pressed ok"))));
233: }
234: // END SNIPPET: onButtonDialogOkPressed
235:
236: /***********************************************************************************************************************************************************
237: *
238: **********************************************************************************************************************************************************/
239: // START SNIPPET: onButtonDialogOkCancelPressed
240: private void onButtonDialogOkCancelPressed()
241: {
242: presentation.notify(notificationWithFeedback()
243: .withCaption("Notification")
244: .withText("Now press the button")
245: .withFeedback(feedback().withOnConfirm(() -> presentation.notify("Pressed ok"))
246: .withOnCancel(() -> presentation.notify("Pressed cancel"))));
247: }
248: // END SNIPPET: onButtonDialogOkCancelPressed
249:
250: /***********************************************************************************************************************************************************
251: * This method demonstrates how to pick a file name by using the proper UI dialog.
252: **********************************************************************************************************************************************************/
253: // START SNIPPET: onButtonPickFilePressed
254: private void onButtonPickFilePressed()
255: {
256: final var selectedFile = new BoundProperty<>(USER_HOME);
257: presentation.pickFile(selectedFile,
258: notificationWithFeedback()
259: .withCaption("Pick a file")
260: .withFeedback(feedback().withOnConfirm(() -> presentation.notify("Selected file: " + selectedFile.get()))
261: .withOnCancel(() -> presentation.notify("Selection cancelled"))));
262: }
263: // END SNIPPET: onButtonPickFilePressed
264:
265: /***********************************************************************************************************************************************************
266: * This method demonstrates how to pick a directory name by using the proper UI dialog.
267: **********************************************************************************************************************************************************/
268: // START SNIPPET: onButtonPickDirectoryPressed
269: private void onButtonPickDirectoryPressed()
270: {
271: final var selectedFolder = new BoundProperty<>(USER_HOME);
272: presentation.pickDirectory(selectedFolder,
273: notificationWithFeedback()
274: .withCaption("Pick a directory")
275: .withFeedback(feedback().withOnConfirm(() -> presentation.notify("Selected directory: " + selectedFolder.get()))
276: .withOnCancel(() -> presentation.notify("Selection cancelled"))));
277: }
278: // END SNIPPET: onButtonPickDirectoryPressed
279:
280: /***********************************************************************************************************************************************************
281: *
282: **********************************************************************************************************************************************************/
283: // START SNIPPET: onSelected
284: private void onSelected (@Nonnull final Object object)
285: {
286: presentation.notify("Selected " + object);
287: }
288: // END SNIPPET: onSelected
289:
290: /***********************************************************************************************************************************************************
291: *
292: **********************************************************************************************************************************************************/
293: // START SNIPPET: action1
294: private void action1 (@Nonnull final Object object)
295: {
296: presentation.notify("Action 1 on " + object);
297: }
298: // END SNIPPET: action1
299:
300: /***********************************************************************************************************************************************************
301: *
302: **********************************************************************************************************************************************************/
303: private void action2 (@Nonnull final Object object)
304: {
305: presentation.notify("Action 2 on " + object);
306: }
307:
308: /***********************************************************************************************************************************************************
309: *
310: **********************************************************************************************************************************************************/
311: private void action3 (@Nonnull final Object object)
312: {
313: presentation.notify("Action 3 on " + object);
314: }
315: }