001package jmri.jmrix.bidib.serialdriver; 002 003import java.awt.AWTEvent; 004import java.awt.Color; 005import java.awt.event.ActionEvent; 006import java.awt.event.ActionListener; 007import java.awt.event.FocusEvent; 008import java.awt.event.FocusListener; 009import java.awt.event.ItemEvent; 010import java.util.List; 011import java.util.Vector; 012import javax.swing.JCheckBox; 013import javax.swing.JComboBox; 014import javax.swing.JLabel; 015import javax.swing.JTextField; 016import org.bidib.jbidibc.messages.utils.ByteUtils; 017//import org.bidib.jbidibc.scm.ScmPortIdentifierUtils; 018import org.slf4j.Logger; 019import org.slf4j.LoggerFactory; 020 021/** 022 * Definition of objects to handle configuring a layout connection via a BiDiB 023 * SerialDriverAdapter object. 024 * 025 * @author Bob Jacobsen Copyright (C) 2001, 2003 026 * @author Eckart Meyer Copyright (C) 2019, 2024 027 */ 028public class ConnectionConfig extends jmri.jmrix.AbstractSerialConnectionConfig { 029 030 public final static String NAME = Bundle.getMessage("TypeSerial"); 031// protected JCheckBox useAutoScan = new JCheckBox(Bundle.getMessage("Use Autoscan")); 032 // TODO: use Bundle for localization of the field text 033 protected JCheckBox useAutoScan = new JCheckBox(Bundle.getMessage("SerialConnectionUseAutoscan")); 034 protected JLabel rootNodeLabel = new JLabel(Bundle.getMessage("UniqueIDHex") + ":"); 035 protected JTextField rootNodeField = new JTextField(16); 036 protected JLabel portNameFilterLabel = new JLabel(Bundle.getMessage("SerialConnectionPortNameFilter") + ":"); 037 protected JTextField portNameFilterField = new JTextField(15); 038 039 /** 040 * Ctor for an object being created during load process; Swing init is 041 * deferred. 042 * 043 * @param p SerialPortAdapter for existing adapter 044 */ 045 public ConnectionConfig(jmri.jmrix.SerialPortAdapter p) { 046 super(p); 047 } 048 049 /** 050 * Ctor for a functional Swing object with no existing adapter 051 */ 052 public ConnectionConfig() { 053 super(); 054 } 055 056 @Override 057 public String name() { 058 return NAME; 059 } 060 061 /** 062 * {@inheritDoc} 063 */ 064 @Override 065 protected void setInstance() { 066 log.debug("BiDiB Serial ConnectionConfig.setInstance: {}", adapter); 067 if (adapter == null) { 068 adapter = new SerialDriverAdapter(); 069 log.debug("-- adapter created: {}", adapter); 070 } 071 } 072 073 /** 074 * {@inheritDoc} 075 */ 076 @Override 077// @SuppressWarnings("UseOfObsoleteCollectionType") 078 public Vector<String> getPortNames() { 079 // used for SCM - will prepend "/dev" fpr Linux 080 Vector<String> portNameVector = new Vector<>(); 081 try { 082 List<String> portNameList = ((SerialDriverAdapter)adapter).getPortIdentifiers(); 083 for (String portName : portNameList) { 084 portNameVector.addElement(portName); 085 } 086 } 087 catch (Exception ex) { 088 log.error("Serial adapter not set: ", ex); // NOSONAR 089 } 090// List<String> portNameList = SerialDriverAdapter.getPortIdentifiers(); 091 log.trace("getPortNames done {}", portNameVector); 092 return portNameVector; 093 } 094 095 /** 096 * {@inheritDoc} 097 */ 098 @Override 099 protected void checkInitDone() { 100 super.checkInitDone(); 101 log.debug("checkInitDone"); 102 if (adapter.getSystemConnectionMemo() != null) { 103 SerialDriverAdapter a = (SerialDriverAdapter)adapter; 104 rootNodeField.setText(ByteUtils.formatHexUniqueId(a.getRootNodeUid())); 105 portNameFilterField.setText(a.getPortNameFilter()); 106 useAutoScan.setSelected(a.getUseAutoScan()); 107 if (a.getUseAutoScan()) { 108 rootNodeField.setEnabled(true); 109 portBox.setEnabled(false); 110 } 111 else { 112 rootNodeField.setEnabled(false); 113 portBox.setEnabled(true); 114 } 115 // add listeners 116 // portBox combobox 117 portBox.addActionListener(new ActionListener() { 118 @Override 119 public void actionPerformed(ActionEvent e) { 120 log.debug("portBox action!"); 121 portBoxChanged(e); 122 } 123 }); 124 // rootNode field 125 rootNodeField.addActionListener(new ActionListener() { 126 @Override 127 public void actionPerformed(ActionEvent e) { 128 log.debug("rootNodeUID action!"); 129 rootNodeUidChanged(e); 130 } 131 }); 132 rootNodeField.addFocusListener(new FocusListener() { 133 @Override 134 public void focusLost(FocusEvent e) { 135 log.debug("rootNodeUID focus lost!"); 136 rootNodeUidChanged(e); 137 } 138 @Override 139 public void focusGained(FocusEvent e) { 140 } 141 }); 142 // portNameFilter field 143 portNameFilterField.addActionListener(new ActionListener() { 144 @Override 145 public void actionPerformed(ActionEvent e) { 146 log.debug("portNameFilter action!"); 147 portNameFilterChanged(e); 148 } 149 }); 150 portNameFilterField.addFocusListener(new FocusListener() { 151 @Override 152 public void focusLost(FocusEvent e) { 153 log.debug("portNameFilter focus lost!"); 154 portNameFilterChanged(e); 155 } 156 @Override 157 public void focusGained(FocusEvent e) { 158 } 159 }); 160 // useAutoScan checkbox 161 useAutoScan.addItemListener((ItemEvent e) -> { 162 log.debug("useAutoScan changed!"); 163 a.setUseAutoScan(useAutoScan.isSelected()); 164 rootNodeField.setEnabled(useAutoScan.isSelected()); 165 portBox.setEnabled(!useAutoScan.isSelected()); 166 }); 167 168 baudBox.setEnabled(false);// Bidib SCM always tries to find the the baudrate itself, the user cannot select 169 } 170 } 171 172 @SuppressWarnings (value="unchecked") //because of the unchecked cast of the event object. but we can be sure here... 173 private void portBoxChanged(AWTEvent e) { 174 SerialDriverAdapter a = (SerialDriverAdapter)adapter; 175 String fieldtext = ((JComboBox<String>)e.getSource()).getSelectedItem().toString(); 176 log.debug("portBox selected: {}", fieldtext); 177 if (!a.getUseAutoScan()) { 178 //Long uid = SerialDriverAdapter.checkPort(fieldtext); //call static function 179 Long uid = a.checkPort(fieldtext); //call static function 180 if (uid == null) { 181 a.setRootNodeUid(null); 182 rootNodeField.setText(""); 183 } 184 else { 185 a.setRootNodeUid(uid); 186 //rootNodeField.setText(String.format("0x%X", a.getRootNodeUid() & 0xffffffffffffffL)); 187 rootNodeField.setText(ByteUtils.formatHexUniqueId(a.getRootNodeUid())); 188 } 189 } 190 } 191 192 private void rootNodeUidChanged(AWTEvent e) { 193 SerialDriverAdapter a = (SerialDriverAdapter)adapter; 194 String fieldtext = ((JTextField)e.getSource()).getText(); 195 if (fieldtext.isEmpty()) { 196 //a.setRootNodeUid(null); //don't set 197 } 198 else { 199 try { 200 if (a.getUseAutoScan()) { 201 Long uid = ByteUtils.parseHexUniqueId(fieldtext); 202 String err = a.findPortbyUniqueID(uid); 203 if (err == null) { 204 log.info("found port name for UID {} is {}", 205 ByteUtils.formatHexUniqueId(a.getRootNodeUid()), a.getCurrentPortName()); 206 a.setRootNodeUid(uid); 207 //rootNodeField.setText(String.format("0x%X", a.getRootNodeUid() & 0x0000ffffffffffL)); 208 rootNodeField.setText(ByteUtils.formatHexUniqueId(a.getRootNodeUid())); 209 // TODO set portBox 210 rootNodeField.setForeground(Color.black); 211 portBox.setSelectedItem(a.getCurrentPortName()); 212 } 213 else { 214 log.warn(err); 215 rootNodeField.setForeground(Color.red); 216 } 217 } 218 } 219 catch (NumberFormatException ex) { 220 a.setRootNodeUid(null); 221 rootNodeField.setText(""); 222 log.warn("Exception:", ex); 223 } 224 } 225 } 226 227 private void portNameFilterChanged(AWTEvent e) { 228 SerialDriverAdapter a = (SerialDriverAdapter)adapter; 229 // String s = a.getCurrentPortName(); 230 String fieldtext = ((JTextField)e.getSource()).getText(); 231 a.setPortNameFilter(fieldtext); 232 refreshPortBox(); 233 } 234 235 /** 236 * {@inheritDoc} 237 */ 238 @Override 239 protected void showAdvancedItems() { 240 super.showAdvancedItems(); // we're adding to the normal advanced items. 241 log.debug("showAdvancedItems"); 242 if (adapter.getSystemConnectionMemo() != null) { 243 cR.gridy += 2; 244 cL.gridy += 2; 245 gbLayout.setConstraints(rootNodeLabel, cL); 246 gbLayout.setConstraints(rootNodeField, cR); 247 _details.add(rootNodeLabel); 248 //rootNodeField.setEnabled(false); 249 _details.add(rootNodeField); 250 251 cR.gridy += 2; 252 cL.gridy += 2; 253 gbLayout.setConstraints(useAutoScan, cL); 254 useAutoScan.setFont(useAutoScan.getFont().deriveFont(9f)); 255 useAutoScan.setForeground(Color.blue); 256 _details.add(useAutoScan); 257 258 cR.gridy += 2; 259 cL.gridy += 2; 260 gbLayout.setConstraints(portNameFilterLabel, cL); 261 gbLayout.setConstraints(portNameFilterField, cR); 262 _details.add(portNameFilterLabel); 263 _details.add(portNameFilterField); 264 } 265 if (_details.getParent() != null) { 266 _details.getParent().revalidate(); 267 _details.getParent().repaint(); 268 } 269 } 270 271 @Override 272 public void updateAdapter() { 273 super.updateAdapter(); // we're adding more details to the connection. 274 log.debug("updateAdapter"); 275 SerialDriverAdapter a = (SerialDriverAdapter)adapter; 276 if (adapter.getSystemConnectionMemo() != null) { 277 a.setUseAutoScan(useAutoScan.isSelected()); 278 a.setRootNodeUid(ByteUtils.parseHexUniqueId(rootNodeField.getText())); 279 a.setPortNameFilter(portNameFilterField.getText()); 280 } 281 } 282 283 284 private final static Logger log = LoggerFactory.getLogger(ConnectionConfig.class); 285}