1
0
mirror of https://github.com/laurent22/joplin.git synced 2024-12-12 08:54:00 +02:00
joplin/ElectronClient/app/gui/NoteList.jsx

164 lines
4.5 KiB
React
Raw Normal View History

2017-11-04 18:40:34 +02:00
const { ItemList } = require('./ItemList.min.js');
const React = require('react');
const { connect } = require('react-redux');
2017-11-10 22:11:48 +02:00
const { time } = require('lib/time-utils.js');
2017-11-08 19:51:55 +02:00
const { themeStyle } = require('../theme.js');
const { _ } = require('lib/locale.js');
const { bridge } = require('electron').remote.require('./bridge');
const Menu = bridge().Menu;
const MenuItem = bridge().MenuItem;
2017-11-04 18:40:34 +02:00
class NoteListComponent extends React.Component {
2017-11-09 21:21:10 +02:00
style() {
const theme = themeStyle(this.props.theme);
const itemHeight = 34;
2017-11-09 21:21:10 +02:00
let style = {
root: {
backgroundColor: theme.backgroundColor,
},
listItem: {
height: itemHeight,
boxSizing: 'border-box',
display: 'flex',
2017-11-10 22:11:48 +02:00
alignItems: 'stretch',
2017-11-09 21:21:10 +02:00
backgroundColor: theme.backgroundColor,
borderBottom: '1px solid ' + theme.dividerColor,
},
listItemSelected: {
backgroundColor: theme.selectedColor,
},
2017-11-10 22:11:48 +02:00
listItemTitle: {
fontFamily: theme.fontFamily,
fontSize: theme.fontSize,
textDecoration: 'none',
color: theme.color,
cursor: 'default',
whiteSpace: 'nowrap',
flex: 1,
display: 'flex',
alignItems: 'center',
overflow: 'hidden',
},
listItemTitleCompleted: {
opacity: 0.5,
textDecoration: 'line-through',
},
2017-11-09 21:21:10 +02:00
};
return style;
}
2017-11-08 19:51:55 +02:00
itemContextMenu(event) {
const noteId = event.target.getAttribute('data-id');
if (!noteId) throw new Error('No data-id on element');
const menu = new Menu()
2017-11-08 23:22:24 +02:00
menu.append(new MenuItem({label: _('Delete'), click: async () => {
2017-11-08 19:51:55 +02:00
const ok = bridge().showConfirmMessageBox(_('Delete note?'));
if (!ok) return;
await Note.delete(noteId);
}}));
menu.append(new MenuItem({label: _('Switch between note and to-do'), click: async () => {
const note = await Note.load(noteId);
await Note.save(Note.toggleIsTodo(note));
2017-11-08 19:51:55 +02:00
}}))
2017-11-08 19:51:55 +02:00
menu.popup(bridge().window());
}
2017-11-10 22:11:48 +02:00
itemRenderer(item, theme, width) {
const onTitleClick = async (event, item) => {
2017-11-05 01:27:13 +02:00
this.props.dispatch({
2017-11-08 23:22:24 +02:00
type: 'NOTE_SELECT',
id: item.id,
2017-11-05 01:27:13 +02:00
});
}
2017-11-10 22:11:48 +02:00
const onCheckboxClick = async (event) => {
const checked = event.target.checked;
const newNote = {
id: item.id,
todo_completed: checked ? time.unixMs() : 0,
}
await Note.save(newNote);
}
const padding = 6;
let style = Object.assign({ width: width }, this.style().listItem);
2017-11-09 21:21:10 +02:00
if (this.props.selectedNoteId === item.id) style = Object.assign(style, this.style().listItemSelected);
2017-11-08 19:51:55 +02:00
2017-11-10 22:11:48 +02:00
// Setting marginBottom = 1 because it makes the checkbox looks more centered, at least on Windows
// but don't know how it will look in other OSes.
const checkbox = item.is_todo ?
<div style={{display: 'flex', height: style.height, alignItems: 'center', paddingLeft: padding}}>
<input style={{margin:0, marginBottom:1}} type="checkbox" defaultChecked={!!item.todo_completed} onClick={(event) => { onCheckboxClick(event, item) }}/>
</div>
: null;
let listItemTitleStyle = Object.assign({}, this.style().listItemTitle);
2017-11-10 22:11:48 +02:00
listItemTitleStyle.paddingLeft = checkbox ? padding : 4;
if (item.is_todo && !!item.todo_completed) listItemTitleStyle = Object.assign(listItemTitleStyle, this.style().listItemTitleCompleted);
2017-11-10 22:11:48 +02:00
return <div key={item.id} style={style}>
{checkbox}
<a
data-id={item.id}
className="list-item"
onContextMenu={(event) => this.itemContextMenu(event)}
href="#"
style={listItemTitleStyle}
onClick={(event) => { onTitleClick(event, item) }}
>
{item.title}
</a>
</div>
2017-11-04 18:40:34 +02:00
}
render() {
2017-11-08 19:51:55 +02:00
const theme = themeStyle(this.props.theme);
2017-11-10 19:58:17 +02:00
const style = this.props.style;
if (!this.props.notes.length) {
const padding = 10;
const emptyDivStyle = Object.assign({
padding: padding + 'px',
fontSize: theme.fontSize,
color: theme.color,
backgroundColor: theme.backgroundColor,
fontFamily: theme.fontFamily,
}, style);
emptyDivStyle.width = emptyDivStyle.width - padding * 2;
emptyDivStyle.height = emptyDivStyle.height - padding * 2;
return <div style={emptyDivStyle}>{_('No notes in here. Create one by clicking on "New note".')}</div>
}
2017-11-08 19:51:55 +02:00
2017-11-04 18:40:34 +02:00
return (
<ItemList
itemHeight={this.props.itemHeight}
2017-11-10 19:58:17 +02:00
style={style}
className={"note-list"}
items={this.props.notes}
2017-11-10 22:11:48 +02:00
itemRenderer={ (item) => { return this.itemRenderer(item, theme, style.width) } }
2017-11-05 01:27:13 +02:00
></ItemList>
2017-11-04 18:40:34 +02:00
);
}
}
const mapStateToProps = (state) => {
return {
2017-11-05 01:27:13 +02:00
notes: state.notes,
2017-11-06 01:55:01 +02:00
selectedNoteId: state.selectedNoteId,
2017-11-08 19:51:55 +02:00
theme: state.settings.theme,
2017-11-04 18:40:34 +02:00
};
};
const NoteList = connect(mapStateToProps)(NoteListComponent);
module.exports = { NoteList };