Drag and Drop works ok now.

The main functionality works but only in cosmic desktop. So there are
some issues that need to be worked out yet in regards to libcosmic.
This commit is contained in:
Chris Cochrun 2025-02-20 06:53:54 -06:00
parent 614630bea8
commit a36a1d59c6
10 changed files with 466 additions and 512 deletions

View file

@ -6,7 +6,7 @@ use super::{
model::{get_db, LibraryKind, Model},
service_items::ServiceTrait,
};
use crisp::types::{Keyword, Value};
use crisp::types::{Keyword, Symbol, Value};
use miette::{IntoDiagnostic, Result};
use serde::{Deserialize, Serialize};
use sqlx::{query_as, SqliteConnection};
@ -22,6 +22,12 @@ pub struct Image {
pub path: PathBuf,
}
impl From<&Image> for Value {
fn from(value: &Image) -> Self {
Self::List(vec![Self::Symbol(Symbol("image".into()))])
}
}
impl Content for Image {
fn title(&self) -> String {
self.title.clone()

View file

@ -1,4 +1,4 @@
use crisp::types::{Keyword, Value};
use crisp::types::{Keyword, Symbol, Value};
use miette::{IntoDiagnostic, Result};
use serde::{Deserialize, Serialize};
use sqlx::{
@ -36,6 +36,12 @@ pub struct Presentation {
pub kind: PresKind,
}
impl From<&Presentation> for Value {
fn from(value: &Presentation) -> Self {
Self::List(vec![Self::Symbol(Symbol("presentation".into()))])
}
}
impl Content for Presentation {
fn title(&self) -> String {
self.title.clone()

View file

@ -47,7 +47,7 @@ impl AllowedMimeTypes for ServiceItem {
}
impl AsMimeTypes for ServiceItem {
fn available(&self) -> std::borrow::Cow<'static, [String]> {
fn available(&self) -> Cow<'static, [String]> {
debug!(?self);
Cow::from(vec!["application/service-item".to_string()])
}
@ -67,11 +67,13 @@ impl AsMimeTypes for ServiceItem {
impl From<&ServiceItem> for Value {
fn from(value: &ServiceItem) -> Self {
match &value.kind {
ServiceItemKind::Song(song) => todo!(),
ServiceItemKind::Video(video) => todo!(),
ServiceItemKind::Image(image) => todo!(),
ServiceItemKind::Presentation(presentation) => todo!(),
ServiceItemKind::Content(slide) => todo!(),
ServiceItemKind::Song(song) => Value::from(song),
ServiceItemKind::Video(video) => Value::from(video),
ServiceItemKind::Image(image) => Value::from(image),
ServiceItemKind::Presentation(presentation) => {
Value::from(presentation)
}
ServiceItemKind::Content(slide) => Value::from(slide),
}
}
}

View file

@ -194,6 +194,12 @@ pub struct Slide {
video_end_time: f32,
}
impl From<&Slide> for Value {
fn from(value: &Slide) -> Self {
Self::List(vec![Self::Symbol(Symbol("slide".into()))])
}
}
impl Slide {
pub fn background(&self) -> &Background {
&self.background

View file

@ -36,6 +36,12 @@ pub struct Song {
pub font_size: Option<i32>,
}
impl From<&Song> for Value {
fn from(value: &Song) -> Self {
Self::List(vec![Self::Symbol(Symbol("song".into()))])
}
}
impl Content for Song {
fn title(&self) -> String {
self.title.clone()

View file

@ -8,7 +8,7 @@ use super::{
slide::Slide,
};
use cosmic::iced::Executor;
use crisp::types::{Keyword, Value};
use crisp::types::{Keyword, Symbol, Value};
use miette::{IntoDiagnostic, Result};
use serde::{Deserialize, Serialize};
use sqlx::{query_as, SqliteConnection};
@ -27,6 +27,12 @@ pub struct Video {
pub looping: bool,
}
impl From<&Video> for Value {
fn from(value: &Video) -> Self {
Self::List(vec![Self::Symbol(Symbol("video".into()))])
}
}
impl Content for Video {
fn title(&self) -> String {
self.title.clone()

View file

@ -15,7 +15,8 @@ use cosmic::widget::nav_bar::nav_bar_style;
use cosmic::widget::segmented_button::Entity;
use cosmic::widget::tooltip::Position as TPosition;
use cosmic::widget::{
button, nav_bar, text, tooltip, DndSource, Id, Space,
button, nav_bar, text, tooltip, DndDestination, DndSource, Id,
Space,
};
use cosmic::widget::{icon, slider};
use cosmic::{executor, Application, ApplicationExt, Element};
@ -28,6 +29,7 @@ use std::path::PathBuf;
use tracing::{debug, level_filters::LevelFilter};
use tracing::{error, warn};
use tracing_subscriber::EnvFilter;
use ui::editor::SongEditor;
use ui::library::{self, Library};
pub mod core;
@ -96,12 +98,14 @@ struct App {
library: Option<Library>,
library_open: bool,
library_width: f32,
song_editor: SongEditor,
}
#[derive(Debug, Clone)]
enum Message {
Present(presenter::Message),
Library(library::Message),
SongEditor(editor::Message),
File(PathBuf),
DndEnter(Entity, Vec<String>),
DndDrop(Entity, Option<ServiceItem>, DndAction),
@ -165,6 +169,7 @@ impl cosmic::Application for App {
let presenter = Presenter::with_items(items.clone());
let slides = items.to_slides().unwrap_or_default();
let current_slide = slides[0].clone();
let song_editor = SongEditor::new();
for item in items.iter() {
nav_model.insert().text(item.title()).data(item.clone());
@ -174,6 +179,7 @@ impl cosmic::Application for App {
let mut app = App {
presenter,
song_editor,
core,
nav_model,
file: PathBuf::default(),
@ -221,15 +227,12 @@ impl cosmic::Application for App {
cosmic::app::cosmic::Message::NavBar(id),
)
})
.on_dnd_drop(|entity, data, action| {
cosmic::app::Message::App(Message::DndDrop(
entity, data, action,
))
})
.on_dnd_enter(|entity, data| {
debug!("entered");
cosmic::app::Message::App(Message::DndEnter(entity, data))
})
.on_dnd_leave(|entity| {
debug!("left");
cosmic::app::Message::App(Message::DndLeave(entity))
})
.drag_id(DragId::new())
@ -238,6 +241,12 @@ impl cosmic::Application for App {
cosmic::app::cosmic::Message::NavBarContext(id),
)
})
.on_dnd_drop::<ServiceItem>(|entity, data, action| {
debug!("dropped");
cosmic::app::Message::App(Message::DndDrop(
entity, data, action,
))
})
.context_menu(None)
.into_container()
// XXX both must be shrink to avoid flex layout from ignoring it
@ -248,6 +257,14 @@ impl cosmic::Application for App {
nav = nav.max_width(280);
}
// let dnd = DndDestination::new(
// nav,
// vec!["application/service-item".to_string().into()],
// )
// .data_received_for(|item| {
// cosmic::app::Message::App(Message::DndDrop(item))
// });
let column = column![
text::heading("Service List").center().width(280),
nav
@ -423,6 +440,9 @@ impl cosmic::Application for App {
_ => Task::none(),
}
}
Message::SongEditor(message) => {
todo!()
}
Message::Present(message) => {
// debug!(?message);
if self.presentation_open {
@ -523,8 +543,8 @@ impl cosmic::Application for App {
}
Message::DndDrop(entity, service_item, action) => {
debug!(?entity);
debug!(?service_item);
debug!(?action);
debug!(?service_item);
Task::none()
}
Message::AddLibrary(library) => {
@ -614,6 +634,9 @@ impl cosmic::Application for App {
})
.style(nav_bar_style)
.center(Length::Fill);
let song_editor =
self.song_editor.view().map(|m| Message::SongEditor(m));
// let dnd_source: DndSource<Message, _> = DndSource::with_id(
// drag_item.expect("errors").map(|m| Message::Library(m)),
// Id::new("item"),

View file

@ -1,5 +1,6 @@
use cosmic::{
iced::{alignment::Vertical, Background, Border, Length},
iced_core::widget::tree::State,
iced_widget::{column, row as rowm, text as textm},
widget::{
container, horizontal_space, icon, mouse_area, responsive,
@ -233,6 +234,12 @@ impl Library {
.on_start(Some(Message::DragItem(Some(
(model.kind, index as i32),
))))
.on_finish(Some(Message::DragItem(Some(
(model.kind, index as i32),
))))
.on_cancel(Some(Message::DragItem(Some(
(model.kind, index as i32),
))))
.into()
},
)