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 }