Mercurial Hosting > junotu
changeset 82:e476baaaab2c
Started on TabCalendarBoard
Just copied TabBoard and barely fixed it up.
author | Fox |
---|---|
date | Wed, 18 Jan 2023 04:16:50 +0100 |
parents | b2a17ae9c9e2 |
children | 8759379a27a4 |
files | src/junotu/TabCalendarBoard.java src/junotu/TabSimpleSearch.java src/junotu/Window.java |
diffstat | 3 files changed, 657 insertions(+), 7 deletions(-) [+] |
line wrap: on
line diff
diff -r b2a17ae9c9e2 -r e476baaaab2c src/junotu/TabCalendarBoard.java --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/src/junotu/TabCalendarBoard.java Wed Jan 18 04:16:50 2023 +0100 @@ -0,0 +1,642 @@ +package junotu; + +/* TODO: Clean-up imports. */ + +import java.lang.RuntimeException; + +import java.awt.Dimension; +import java.awt.Font; +import java.awt.GridBagConstraints; +import java.awt.event.InputEvent; +import java.awt.event.KeyEvent; +import java.awt.event.MouseEvent; +import java.awt.event.ActionEvent; +import java.awt.event.ActionListener; +import java.awt.event.MouseListener; +import java.awt.event.MouseAdapter; +import java.awt.event.FocusEvent; +import java.awt.event.FocusListener; + +import javax.swing.KeyStroke; +import javax.swing.SwingUtilities; +import javax.swing.event.DocumentListener; +import javax.swing.event.DocumentEvent; + +import javax.swing.JPanel; +import javax.swing.Box; + +import java.awt.BorderLayout; +import java.awt.GridBagLayout; +import javax.swing.BoxLayout; + +import java.awt.Component; +import javax.swing.JButton; +import javax.swing.JLabel; +import javax.swing.JTextArea; +import javax.swing.JTextField; +import javax.swing.JScrollPane; +import javax.swing.JScrollBar; + +import javax.swing.BorderFactory; +import javax.swing.border.TitledBorder; + +import static java.lang.Math.min; +import static java.lang.Math.max; + +import junotu.Main; +import junotu.Window.Tab; +import junotu.Card; + +public class TabCalendarBoard extends JPanel implements ActionListener, MouseListener { + + final static int COLUMN_CONTENT_WIDTH = 256; + final static int COLUMN_WIDTH = COLUMN_CONTENT_WIDTH+16; + + public final String KEY_ACTION_CARD_UP = "card_up"; + public final String KEY_ACTION_CARD_DOWN = "card_down"; + public final String KEY_ACTION_CARD_FULL_UP = "card_full_up"; + public final String KEY_ACTION_CARD_FULL_DOWN = "card_full_down"; + public final String KEY_ACTION_CARD_LEFT = "card_left"; + public final String KEY_ACTION_CARD_RIGHT = "card_right"; + public final String KEY_ACTION_CARD_FULL_LEFT = "card_full_left"; + public final String KEY_ACTION_CARD_FULL_RIGHT = "card_full_right"; + + private class ColumnWidget extends JPanel implements ActionListener, MouseListener { + + long identifier; + boolean newCard; + + TitledBorder titledBorder; + Box cards; + JButton addCard; + + public ColumnWidget( TabCalendarBoard parent, Card card ) + { + this.setLayout( new GridBagLayout() ); + + cards = Box.createVerticalBox(); + addCard = new JButton("+"); + + addCard.setFont( new Font( "Monospaced", Font.BOLD, 32 ) ); + + GridBagConstraints constraints = new GridBagConstraints(); + constraints.anchor = GridBagConstraints.NORTHWEST; + constraints.fill = GridBagConstraints.HORIZONTAL; + constraints.weightx = 1.0; + constraints.weighty = 0.0; + constraints.gridx = 0; + constraints.gridy = 0; + + this.add( cards, constraints ); + constraints.gridy++; + this.add( addCard, constraints ); + constraints.gridy++; + constraints.weighty = 1.0; + this.add( Box.createVerticalGlue(), constraints ); + + addCard.setPreferredSize( new Dimension( COLUMN_CONTENT_WIDTH, 64 ) ); + addCard.setMaximumSize( new Dimension( COLUMN_CONTENT_WIDTH, 64 ) ); + addCard.setAlignmentX( JButton.CENTER_ALIGNMENT ); + + //this.setPreferredSize( new Dimension( COLUMN_WIDTH, 384 ) ); + this.setMaximumSize( new Dimension( COLUMN_WIDTH, 1000000 ) ); + + titledBorder = BorderFactory.createTitledBorder( + BorderFactory.createEtchedBorder(), + "", + TitledBorder.LEADING, + TitledBorder.TOP, + new Font( "Monospaced", Font.BOLD, 16 ) + ); + + this.setBorder( + BorderFactory.createCompoundBorder( + BorderFactory.createEmptyBorder( 0, 8, 0, 8 ), + titledBorder + ) + ); + + addMouseListener(this); + registerKeyboardAction( + this, + KEY_ACTION_CARD_UP, + KeyStroke.getKeyStroke( KeyEvent.VK_UP, InputEvent.ALT_DOWN_MASK ), + WHEN_ANCESTOR_OF_FOCUSED_COMPONENT + ); + registerKeyboardAction( + this, + KEY_ACTION_CARD_DOWN, + KeyStroke.getKeyStroke( KeyEvent.VK_DOWN, InputEvent.ALT_DOWN_MASK ), + WHEN_ANCESTOR_OF_FOCUSED_COMPONENT + ); + registerKeyboardAction( + this, + KEY_ACTION_CARD_FULL_UP, + KeyStroke.getKeyStroke( KeyEvent.VK_PAGE_UP, InputEvent.ALT_DOWN_MASK ), + WHEN_ANCESTOR_OF_FOCUSED_COMPONENT + ); + registerKeyboardAction( + this, + KEY_ACTION_CARD_FULL_DOWN, + KeyStroke.getKeyStroke( KeyEvent.VK_PAGE_DOWN, InputEvent.ALT_DOWN_MASK ), + WHEN_ANCESTOR_OF_FOCUSED_COMPONENT + ); + registerKeyboardAction( + parent, + KEY_ACTION_CARD_LEFT, + KeyStroke.getKeyStroke( KeyEvent.VK_LEFT, InputEvent.ALT_DOWN_MASK ), + WHEN_ANCESTOR_OF_FOCUSED_COMPONENT + ); + registerKeyboardAction( + parent, + KEY_ACTION_CARD_RIGHT, + KeyStroke.getKeyStroke( KeyEvent.VK_RIGHT, InputEvent.ALT_DOWN_MASK ), + WHEN_ANCESTOR_OF_FOCUSED_COMPONENT + ); + registerKeyboardAction( + parent, + KEY_ACTION_CARD_FULL_LEFT, + KeyStroke.getKeyStroke( KeyEvent.VK_HOME, InputEvent.ALT_DOWN_MASK ), + WHEN_ANCESTOR_OF_FOCUSED_COMPONENT + ); + registerKeyboardAction( + parent, + KEY_ACTION_CARD_FULL_RIGHT, + KeyStroke.getKeyStroke( KeyEvent.VK_END, InputEvent.ALT_DOWN_MASK ), + WHEN_ANCESTOR_OF_FOCUSED_COMPONENT + ); + + addCard.addActionListener(this); + + addCard.setToolTipText("Add card."); + + newCard = card == null; + + if( newCard ) { + return; + } + + identifier = card.identifierGet(); + titleSet(card.titleGet()); + + String cardsString = card.<String>tagGetAsOr( Card.TAG_BOARD_COLUMN_CARDS, "" ); + Card[] cardsSplit = TagUtility.parseCardList(cardsString); + + for( int i = 0; i < cardsSplit.length; i++ ) { + + if( cardsSplit[i] == null ) { + System.out.print("Column '"+card.titleGet()+"', identifier "+Long.toString(card.identifierGet())+": Failed to retrieve card identifier by index "+Integer.toString(i)+", aborting cards loading. Full cards tag: '"+cardsString+"'\n"); + break; + } + + insertCard( cardsSplit[i], -1 ); + + } + + } + + public void titleSet( String title ) + { + titledBorder.setTitle(title); + repaint(); + } + + public String titleGet() + { + return titledBorder.getTitle(); + } + + public void insertCard( Card card, int at ) + { + ColumnCardWidget cardWidget = new ColumnCardWidget( card ); + insertCardRaw( cardWidget, at ); + } + + public void insertCardRaw( ColumnCardWidget cardWidget, int at ) + { + if( at == -1 ) { + at = cards.getComponentCount(); + } + cardWidget.addMouseListener(this); + /* TODO: Check if works properly. */ + cards.add( cardWidget, at ); + cards.revalidate(); + } + + public void moveCard( int at, int to ) + { + if( at < 0 ) { + return; + } + ColumnCardWidget cardWidget = (ColumnCardWidget)cards.getComponent(at); + boolean focused = cardWidget.isSelected(); + cards.remove(at); + cards.add( cardWidget, to ); + cards.revalidate(); + if( focused ) { + cardWidget.select(); + } + } + + public ColumnCardWidget popCard( int at ) + { + if( at < 0 ) { + return null; + } + ColumnCardWidget cardWidget = (ColumnCardWidget)cards.getComponent(at); + cards.remove(at); + cardWidget.removeMouseListener(this); + return cardWidget; + } + + public int selectedCard() + { + Component[] cardList = cards.getComponents(); + for( int i = 0; i < cardList.length; i++ ) { + if( ((ColumnCardWidget)cardList[i]).isSelected() ) { + return i; + } + } + System.out.print("Selected card not found."); + return -1; + } + + public int cardCount() + { + return cards.getComponentCount(); + } + + public void save() + { + Component[] cardList = cards.getComponents(); + String cardIdentifiers = ""; + for( int i = 0; i < cardList.length; i++ ) { + ColumnCardWidget cardWidget = (ColumnCardWidget)cardList[i]; + cardWidget.save(); + + if( cardIdentifiers.length() > 0 ) { + cardIdentifiers += " "; + } + cardIdentifiers += Long.toString(cardWidget.identifier); + } + + Card card; + + if( newCard ) { + card = new Card(); + } else { + try { + card = Main.database.cardGetByIdentifier(identifier); + } catch( Exception e ) { + throw new RuntimeException(e); + } + + if( card == null ) { + throw new RuntimeException("Board column update: card not found."); + } + } + + card.titleSet( titleGet() ); + card.tagValueSetOnly( Card.TAG_BOARD_COLUMN_CARDS, cardIdentifiers ); + card.tagValueSetOnly( Card.TAG_BOARD_COLUMN, null ); + + try { + if( newCard ) { + identifier = Main.database.cardAdd( card ); + newCard = false; + } else { + Main.database.cardUpdate(card); + } + } catch( Exception e ) { + throw new RuntimeException(e); + } + + } + + public void delete() + { + save(); + Main.database.cardDeleteByIdentifier(identifier); + } + + public void actionPerformed( ActionEvent e ) + { + Object source = e.getSource(); + if( source == this ) { + int selected = selectedCard(); + int length = cards.getComponentCount(); + + switch( e.getActionCommand() ) { + + case KEY_ACTION_CARD_UP: { + System.out.print("Move card up.\n"); + moveCard( selected, max(selected-1, 0) ); + break; + } + + case KEY_ACTION_CARD_DOWN: { + System.out.print("Move card down.\n"); + moveCard( selected, min(selected+1, cards.getComponentCount()-1) ); + break; + } + + case KEY_ACTION_CARD_FULL_UP: { + System.out.print("Move card full up.\n"); + moveCard( selected, 0 ); + break; + } + + case KEY_ACTION_CARD_FULL_DOWN: { + System.out.print("Move card full down.\n"); + moveCard( selected, cards.getComponentCount()-1 ); + break; + } + + } + } else if( source == addCard ) { + insertCard( null, -1 ); + } + } + + public void mouseClicked( MouseEvent e ) + { + Object source = e.getSource(); + if( source instanceof ColumnCardWidget ) { + if( e.getButton() == MouseEvent.BUTTON2 ) { + ColumnCardWidget cardWidget = (ColumnCardWidget)e.getSource(); + cardWidget.delete(); + cards.remove(cardWidget); + cards.revalidate(); + } + } + } + + public void mouseEntered( MouseEvent e ) {} + public void mouseExited( MouseEvent e ) {} + public void mousePressed( MouseEvent e ) {} + public void mouseReleased( MouseEvent e ) {} + + } + + private class ColumnCardWidget extends JPanel { + + public boolean newCard; + public long identifier; + public JTextArea title; + + public ColumnCardWidget( Card card ) + { + this.setLayout( new BorderLayout() ); + + title = new JTextArea(""); + + title.setFont( new Font( "Monospaced", Font.BOLD, 16 ) ); + + this.setMinimumSize( new Dimension( COLUMN_CONTENT_WIDTH, 64 ) ); + this.setPreferredSize( new Dimension( COLUMN_CONTENT_WIDTH, 64 ) ); + this.setMaximumSize( new Dimension( COLUMN_CONTENT_WIDTH, 128 ) ); + + //title.setMinimumSize( new Dimension( 32, 32 ) ); + + this.setBorder( BorderFactory.createRaisedBevelBorder() ); + title.setEditable( true ); + title.setLineWrap( true ); + title.setWrapStyleWord( true ); + title.setOpaque( false ); + + this.add( title, BorderLayout.CENTER ); + + MouseAdapter mouseListener = new MouseAdapter() + { + @Override + public void mouseClicked( MouseEvent e ) + { + e.setSource(ColumnCardWidget.this); + processMouseEvent(e); + } + }; + + title.addMouseListener( mouseListener ); + + newCard = card == null; + if( !newCard ) { + identifier = card.identifierGet(); + title.setText(card.titleGet()); + } + + } + + public void save() + { + try { + if( newCard ) { + Card card = new Card(); + card.titleSet( title.getText() ); + card.tagValueSetOnly( Card.TAG_BOARD_COLUMN_CARD, Card.VALUE_BOARD_COLUMN_CARD_ONLY ); + + identifier = Main.database.cardAdd(card); + newCard = false; + + } else { + Card card = Main.database.cardGetByIdentifier(identifier); + + if( card == null ) { + throw new RuntimeException("Null card on update try."); + } + + card.titleSet( title.getText() ); + Main.database.cardUpdate(card); + + } + } catch( Exception e ) { + throw new RuntimeException(e); + } + } + + public void delete() + { + if( newCard ) { + return; + } + + Main.database.cardDeleteByIdentifier(identifier, true); + } + + public boolean isSelected() + { + return title.isFocusOwner(); + } + + public void select() + { + title.requestFocusInWindow(); + } + + } + + static long identifier = -1; + Box columns; + JScrollPane scroll; + + JButton back; + + public TabCalendarBoard() + { + this.setLayout( new BorderLayout() ); + + back = new JButton("Back"); + + Box bottom = Box.createHorizontalBox(); + columns = Box.createHorizontalBox(); + scroll = new JScrollPane( columns ); + + bottom.add( back ); + bottom.add( Box.createHorizontalGlue() ); + this.add( scroll, BorderLayout.CENTER ); + this.add( bottom, BorderLayout.SOUTH ); + + back.addActionListener(this); + + back.setToolTipText("Go back to where the card was accessed from."); + + } + + public void boardEdit() + { + if( identifier == -1 ) { + // TODO: Find or create calendar board card. + } + } + + public Card boardSave() + { + Card card; + + card = Main.database.cardGetByIdentifier(identifier); + + if( card == null ) { + throw new RuntimeException(); + } + + // TODO: Proper tags. + card.titleSet( "JUnotu calendar board" ); + card.tagValueSetOnly( Card.TAG_BOARD, null ); + + Main.database.cardUpdate(card); + Main.refreshSearches(); + + return card; + } + + public void boardReset() + { + columns.removeAll(); + } + + public void insertColumn() + { + ColumnWidget column = new ColumnWidget(this, null); + column.titleSet("New column"); + column.addMouseListener(this); + columns.add(column); + columns.revalidate(); + } + + public void insertColumnRaw( ColumnWidget column ) + { + column.addMouseListener(this); + columns.add(column); + columns.revalidate(); + } + + public int findColumn( ColumnWidget columnWidget ) + { + Component[] columnsList = columns.getComponents(); + for( int i = 0; i < columnsList.length; i++ ) { + if( columnsList[i] == columnWidget ) { + return i; + } + } + return -1; + } + + public void removeColumn( ColumnWidget columnWidget ) { + columnWidget.delete(); + columns.remove( columnWidget ); + columns.validate(); + columns.repaint(); + } + + public void moveCard( int from, int at, int to ) + { + ColumnWidget fromColumn = (ColumnWidget)columns.getComponent(from); + ColumnWidget toColumn = (ColumnWidget)columns.getComponent(to); + ColumnCardWidget cardWidget = fromColumn.popCard(at); + if( cardWidget == null ) { + return; + } + toColumn.insertCardRaw(cardWidget, min(at, toColumn.cardCount()) ); + cardWidget.select(); + } + + public void buttonClickedAsCard() + { + Card card = boardSave(); + Window window = (Window)this.getTopLevelAncestor(); + window.tabEdit.cardEdit(card); + window.tabSwitch( Tab.EDIT ); + } + + public void buttonClickedBack() + { + boardSave(); + boardReset(); + Window window = (Window)this.getTopLevelAncestor(); + window.tabSwitch( Tab.SEARCH ); + } + + public void actionPerformed( ActionEvent e ) + { + if( e.getSource() == back ) { + buttonClickedBack(); + return; + } + + ColumnWidget sourceColumn = (ColumnWidget)e.getSource(); + if( sourceColumn != null ) { + int columnIndex = findColumn(sourceColumn); + switch( e.getActionCommand() ){ + + case KEY_ACTION_CARD_LEFT: { + moveCard( columnIndex, sourceColumn.selectedCard(), max( columnIndex-1, 0) ); + } + case KEY_ACTION_CARD_RIGHT: { + moveCard( columnIndex, sourceColumn.selectedCard(), min( columnIndex+1, columns.getComponentCount()-1) ); + } + case KEY_ACTION_CARD_FULL_LEFT: { + moveCard( columnIndex, sourceColumn.selectedCard(), 0 ); + } + case KEY_ACTION_CARD_FULL_RIGHT: { + moveCard( columnIndex, sourceColumn.selectedCard(), columns.getComponentCount()-1 ); + } + + } + } + } + + public void mouseClicked( MouseEvent e ) + { + if( e.getButton() == MouseEvent.BUTTON2 ) { + if( !(e.getSource() instanceof ColumnWidget) ) + return; + + removeColumn( (ColumnWidget)e.getSource() ); + } + } + + public void mouseEntered( MouseEvent e ) {} + public void mouseExited( MouseEvent e ) {} + public void mousePressed( MouseEvent e ) {} + public void mouseReleased( MouseEvent e ) {} + +}
diff -r b2a17ae9c9e2 -r e476baaaab2c src/junotu/TabSimpleSearch.java --- a/src/junotu/TabSimpleSearch.java Sat Jan 07 13:38:25 2023 +0100 +++ b/src/junotu/TabSimpleSearch.java Wed Jan 18 04:16:50 2023 +0100 @@ -221,7 +221,8 @@ } else if( source == context ) { menu.show( (Component)source, 0, 0 ); } else if( source == menu_calendar ) { - System.out.print("Calendar option selected!\n"); + Window window = (Window)this.getTopLevelAncestor(); + window.tabSwitch( Tab.CALENDAR_BOARD ); } }
diff -r b2a17ae9c9e2 -r e476baaaab2c src/junotu/Window.java --- a/src/junotu/Window.java Sat Jan 07 13:38:25 2023 +0100 +++ b/src/junotu/Window.java Wed Jan 18 04:16:50 2023 +0100 @@ -20,6 +20,7 @@ import junotu.TabEdit; import junotu.TabSimpleSearch; import junotu.TabBoard; +import junotu.TabCalendarBoard; public class Window extends JFrame { @@ -31,17 +32,20 @@ SEARCH, EDIT, BOARD, + CALENDAR_BOARD, }; private static final String[] TAB_NAMES = { "Search", "Edit", "Board", + "Calendar board", }; - public TabSimpleSearch tabSearch; - public TabEdit tabEdit; - public TabBoard tabBoard; + public TabSimpleSearch tabSearch; + public TabEdit tabEdit; + public TabBoard tabBoard; + public TabCalendarBoard tabCalendarBoard; private JPanel tabs; private CardLayout tabsLayout; @@ -104,14 +108,17 @@ tabSearch = new TabSimpleSearch(); tabEdit = new TabEdit(); tabBoard = new TabBoard(); + tabCalendarBoard = new TabCalendarBoard(); this.add(tabs); tabs.add(tabSearch); tabs.add(tabEdit); tabs.add(tabBoard); - tabsLayout.addLayoutComponent( tabSearch, TAB_NAMES[Tab.SEARCH.ordinal()] ); - tabsLayout.addLayoutComponent( tabEdit, TAB_NAMES[Tab.EDIT.ordinal()] ); - tabsLayout.addLayoutComponent( tabBoard, TAB_NAMES[Tab.BOARD.ordinal()] ); + tabs.add(tabCalendarBoard); + tabsLayout.addLayoutComponent( tabSearch, TAB_NAMES[Tab.SEARCH.ordinal()] ); + tabsLayout.addLayoutComponent( tabEdit, TAB_NAMES[Tab.EDIT.ordinal()] ); + tabsLayout.addLayoutComponent( tabBoard, TAB_NAMES[Tab.BOARD.ordinal()] ); + tabsLayout.addLayoutComponent( tabCalendarBoard, TAB_NAMES[Tab.CALENDAR_BOARD.ordinal()] ); }