- 
          
- 
                Notifications
    You must be signed in to change notification settings 
- Fork 4.2k
Basic clipboard support #19106
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
base: main
Are you sure you want to change the base?
Basic clipboard support #19106
Conversation
Enabled with the `bevy_clipboard` feature. Supports setting and getting text to and from the clipboard. Only supports desktop targets.
Enabled with the `bevy_clipboard` feature. Supports setting and getting text to and from the clipboard. Only supports desktop targets.
| Marked as X-Controversial due to a new crate, but I strongly think we need this as part of the engine in some form. Both for games and the editor! | 
| I don't like the crate for that which is pretty much just a resource wrapping arboard... And not sure I like the fake resource that can't work on not supported platforms | 
        
          
                crates/bevy_clipboard/src/dummy.rs
              
                Outdated
          
        
      | /// | ||
| /// Returns error if clipboard is empty or contents are not UTF-8 text. | ||
| pub fn get_text(&mut self) -> Result<String, arboard::Error> { | ||
| Err(arboard::Error::ClipboardNotSupported) | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Question: why can't Clipboard store the string itself? Then use arboard in addition when it is supported?
I guess that only works when setting directly through the resource. But might be an okay tradeoff
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I thought about this but felt like it might be confusing for users if copy and paste is only working internally.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What platforms are unsupported? It might be an okay tradeoff to still allow some amount of clipboard behavior
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
arboard seems to support Linux, Mac and Windows. So there's at least Web that is unsupported, and many no_std targets.
I also feel like ultra basic support is still better than no support at all ? Maybe with a warning emitted if the plugin is enabled on an unsupported platform ?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I've added wasm32 support now, it seems to work okay but needs testing.
All enabling the plugin on an unsupported platform does is init a unit Clipboard resource with accessor functions that always return ClipboardError::ClipboardNotSupported, I don't know if it needs any warnings.
        
          
                crates/bevy_clipboard/src/desktop.rs
              
                Outdated
          
        
      | &mut self, | ||
| text: T, | ||
| ) -> Result<(), arboard::Error> { | ||
| arboard::Clipboard::new().and_then(|mut clipboard| clipboard.set_text(text)) | 
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think dropping the Clipboard may not work on Linux ?
See arboard's documentation about platform-specific behaviors (and also issue #88 and PR #89 for more info)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yeah seems like drop-after-use is only needed on windows.
| 
 This PR is really trivial, but image and android and wasm support will add a lot more complexity. I'm being lazy here, I've got way too many things I working on atm so I was thinking if I just get desktop text clipboard support going someone else will pick this up and fill out the rest of the implementation. I'm not sure it's an important enough feature to deserve its own crate, but I'm not sure if it belongs in any of the exisiting crates either. Maybe  
 The intent is that you can only use the clipboard with a  | 
Co-authored-by: Gilles Henaux <[email protected]>
* On unix targets an instance of `arboard::Clipboard` is stored in the `Clipboard` resource. * Removed the desktop and dummy modules. * Made `arboard` dependency conditional.
| For crates candidates, I've had a look and did not find much either. Doesn't fit in  There's a potential savior in  | 
| 
 It'd be ideal if winit just supported it and we didn't have to do anything but I don't like the idea of adding it to the winit crate before they have some built in support, it seems like a lie. And it might still need a separate solution when targeting wasm32? | 
… support for platform.
Clipboard bevy/main merge
| I need that, ty! PR lgtm (but I'm not a skilled rustacean). Some tweaks that come to mind: 
 I like the simple and easy sync API (from user standpoint), I guess most of the time it'll be the tool for the job. Also, #[derive(Debug)]
pub enum ClipboardRead {
    /// The clipboard contents are ready to be accessed.
    Ready(Result<String, ClipboardError>),
    #[cfg(target_arch = "wasm32")]
    /// The clipboard contents are being fetched asynchronously.
    Pending(Arc<Mutex<Option<Result<String, ClipboardError>>>>),
}Seem an awful lot like a Task, right?. :) AsyncComputeTaskPool::get().spawn(<ready future that did sync clipboard read>)But then, if it makes sense so far 😄, I guess we could go async-first, That's all that came to mind.. hope any of that is of value. :) | 
| Very helpful feedback, both on the testing and design side. Thanks! | 
| Oh, we'd also need arboard's  | 
| Yes thanks a lot for the review. 
 I think the clipboard api returns a  
 That's what I'm using I think, it gets the clipboard from a  
 Sound good. 
 This was actually how I wrote it initially but most of the multi-platform code in bevy does it this way so I rewrote it because I figured that way it would be more likely to get merged. I guess this way the advantage is that it ensures that the interface is the same for each platform. | 
Objective
Add a platform-agnostic interface for interacting with the clipboard.
Solution
New crate
bevy_clipboardwith aClipboardPluginthat adds aClipboardresource. The clipboard is accessed using the methodsfetch_textandset_texton theClipboardresource.fetch_textreturns aClipboardReadwith apoll_resultmethod that's used to get the actual value once it's ready.The
windowsanduniximplementations both use thearboardcrate. On windows theClipboardresource is a unit struct and a new arboard clipboard instance is created and dropped for each clipboard access. On unix targets theClipboardresource holds a clipboard instance it reuses each time. On both targetsfetch_textandset_textwork instantly.On
wasm32Clipboardis a unit struct. Thefetch_textandset_textfunctions spawn async tasks. The task spawned byfetch_textupdates the shared arc mutex option once the future is evaluated to get the clipboard text.It works but it feels like the design of
ClipboardReadis a bit clumsy and not very idiomatic. I don't tend to do much asynchronous programming, maybe a reviewer can suggest an improved construction.Also added an alternative async
fetch_text_asyncfunction now that returns aResult<String, ClipboardError>future.Notes
Testing
The PR includes a basic example
clipboardthat can be used for testing.