cursor_icon/
lib.rs

1// SPDX-License-Identifier: MIT OR Apache-2.0 OR Zlib
2
3#![cfg_attr(not(feature = "std"), no_std)]
4#![deny(rust_2018_idioms)]
5#![deny(rustdoc::broken_intra_doc_links)]
6#![deny(unsafe_op_in_unsafe_fn)]
7#![deny(improper_ctypes, improper_ctypes_definitions)]
8#![deny(clippy::all)]
9#![deny(missing_debug_implementations)]
10#![deny(missing_docs)]
11#![forbid(unsafe_code)]
12#![cfg_attr(feature = "cargo-clippy", deny(warnings))]
13#![cfg_attr(docsrs, feature(doc_auto_cfg))]
14
15//! The cross platform cursor icon type.
16//!
17//! This type is intended to be used as a standard interopability type between
18//! GUI frameworks in order to convey the cursor icon type.
19//!
20//! # Example
21//!
22//! ```
23//! use cursor_icon::CursorIcon;
24//!
25//! # fn main() -> Result<(), Box<dyn std::error::Error>> {
26//! // Parse a cursor icon from the string that describes it.
27//! let cursor_name = "pointer";
28//! let cursor_icon: CursorIcon = cursor_name.parse()?;
29//! println!("The cursor icon is {:?}", cursor_icon);
30//! # Ok(())
31//! # }
32//! ```
33
34// This file contains a portion of the CSS Basic User Interface Module Level 3
35// specification. In particular, the names for the cursor from the #cursor
36// section and documentation for some of the variants were taken.
37//
38// The original document is https://www.w3.org/TR/css-ui-3/#cursor.
39// Copyright © 2018 W3C® (MIT, ERCIM, Keio, Beihang)
40//
41// These documents were used under the terms of the following license. This W3C
42// license as well as the W3C short notice apply to the `CursorIcon` enum's
43// variants and documentation attached to them.
44
45// --------- BEGGINING OF W3C LICENSE
46// --------------------------------------------------------------
47//
48// License
49//
50// By obtaining and/or copying this work, you (the licensee) agree that you have
51// read, understood, and will comply with the following terms and conditions.
52//
53// Permission to copy, modify, and distribute this work, with or without
54// modification, for any purpose and without fee or royalty is hereby granted,
55// provided that you include the following on ALL copies of the work or portions
56// thereof, including modifications:
57//
58// - The full text of this NOTICE in a location viewable to users of the
59//   redistributed or derivative work.
60// - Any pre-existing intellectual property disclaimers, notices, or terms and
61//   conditions. If none exist, the W3C Software and Document Short Notice
62//   should be included.
63// - Notice of any changes or modifications, through a copyright statement on
64//   the new code or document such as "This software or document includes
65//   material copied from or derived from [title and URI of the W3C document].
66//   Copyright © [YEAR] W3C® (MIT, ERCIM, Keio, Beihang)."
67//
68// Disclaimers
69//
70// THIS WORK IS PROVIDED "AS IS," AND COPYRIGHT HOLDERS MAKE NO REPRESENTATIONS
71// OR WARRANTIES, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO, WARRANTIES
72// OF MERCHANTABILITY OR FITNESS FOR ANY PARTICULAR PURPOSE OR THAT THE USE OF
73// THE SOFTWARE OR DOCUMENT WILL NOT INFRINGE ANY THIRD PARTY PATENTS,
74// COPYRIGHTS, TRADEMARKS OR OTHER RIGHTS.
75//
76// COPYRIGHT HOLDERS WILL NOT BE LIABLE FOR ANY DIRECT, INDIRECT, SPECIAL OR
77// CONSEQUENTIAL DAMAGES ARISING OUT OF ANY USE OF THE SOFTWARE OR DOCUMENT.
78//
79// The name and trademarks of copyright holders may NOT be used in advertising
80// or publicity pertaining to the work without specific, written prior
81// permission. Title to copyright in this work will at all times remain with
82// copyright holders.
83//
84// --------- END OF W3C LICENSE
85// --------------------------------------------------------------------
86
87// --------- BEGGINING OF W3C SHORT NOTICE
88// ---------------------------------------------------------
89//
90// winit: https://github.com/rust-windowing/cursor-icon
91//
92// Copyright © 2023 World Wide Web Consortium, (Massachusetts Institute of
93// Technology, European Research Consortium for Informatics and Mathematics,
94// Keio University, Beihang). All Rights Reserved. This work is distributed
95// under the W3C® Software License [1] in the hope that it will be useful, but
96// WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
97// FITNESS FOR A PARTICULAR PURPOSE.
98//
99// [1] http://www.w3.org/Consortium/Legal/copyright-software
100//
101// --------- END OF W3C SHORT NOTICE
102// --------------------------------------------------------------
103
104#[cfg(feature = "serde")]
105#[macro_use]
106extern crate serde;
107
108// XXX for forwards compatibility.
109#[cfg(feature = "alloc")]
110extern crate alloc as _;
111
112/// Describes the appearance of the (usually mouse) cursor icon.
113///
114/// The names are taken from the CSS W3C specification:
115/// <https://www.w3.org/TR/css-ui-3/#cursor>
116///
117/// # Examples
118///
119/// ```
120/// use cursor_icon::CursorIcon;
121///
122/// // Get the cursor icon for the default cursor.
123/// let cursor_icon = CursorIcon::Default;
124/// ```
125#[non_exhaustive]
126#[derive(Debug, Default, Copy, Clone, PartialEq, Eq, Hash)]
127#[cfg_attr(feature = "serde", derive(Serialize, Deserialize))]
128pub enum CursorIcon {
129    /// The platform-dependent default cursor. Often rendered as arrow.
130    #[default]
131    Default,
132
133    /// A context menu is available for the object under the cursor. Often
134    /// rendered as an arrow with a small menu-like graphic next to it.
135    ContextMenu,
136
137    /// Help is available for the object under the cursor. Often rendered as a
138    /// question mark or a balloon.
139    Help,
140
141    /// The cursor is a pointer that indicates a link. Often rendered as the
142    /// backside of a hand with the index finger extended.
143    Pointer,
144
145    /// A progress indicator. The program is performing some processing, but is
146    /// different from [`CursorIcon::Wait`] in that the user may still interact
147    /// with the program.
148    Progress,
149
150    /// Indicates that the program is busy and the user should wait. Often
151    /// rendered as a watch or hourglass.
152    Wait,
153
154    /// Indicates that a cell or set of cells may be selected. Often rendered as
155    /// a thick plus-sign with a dot in the middle.
156    Cell,
157
158    /// A simple crosshair (e.g., short line segments resembling a "+" sign).
159    /// Often used to indicate a two dimensional bitmap selection mode.
160    Crosshair,
161
162    /// Indicates text that may be selected. Often rendered as an I-beam.
163    Text,
164
165    /// Indicates vertical-text that may be selected. Often rendered as a
166    /// horizontal I-beam.
167    VerticalText,
168
169    /// Indicates an alias of/shortcut to something is to be created. Often
170    /// rendered as an arrow with a small curved arrow next to it.
171    Alias,
172
173    /// Indicates something is to be copied. Often rendered as an arrow with a
174    /// small plus sign next to it.
175    Copy,
176
177    /// Indicates something is to be moved.
178    Move,
179
180    /// Indicates that the dragged item cannot be dropped at the current cursor
181    /// location. Often rendered as a hand or pointer with a small circle with a
182    /// line through it.
183    NoDrop,
184
185    /// Indicates that the requested action will not be carried out. Often
186    /// rendered as a circle with a line through it.
187    NotAllowed,
188
189    /// Indicates that something can be grabbed (dragged to be moved). Often
190    /// rendered as the backside of an open hand.
191    Grab,
192
193    /// Indicates that something is being grabbed (dragged to be moved). Often
194    /// rendered as the backside of a hand with fingers closed mostly out of
195    /// view.
196    Grabbing,
197
198    /// The east border to be moved.
199    EResize,
200
201    /// The north border to be moved.
202    NResize,
203
204    /// The north-east corner to be moved.
205    NeResize,
206
207    /// The north-west corner to be moved.
208    NwResize,
209
210    /// The south border to be moved.
211    SResize,
212
213    /// The south-east corner to be moved.
214    SeResize,
215
216    /// The south-west corner to be moved.
217    SwResize,
218
219    /// The west border to be moved.
220    WResize,
221
222    /// The east and west borders to be moved.
223    EwResize,
224
225    /// The south and north borders to be moved.
226    NsResize,
227
228    /// The north-east and south-west corners to be moved.
229    NeswResize,
230
231    /// The north-west and south-east corners to be moved.
232    NwseResize,
233
234    /// Indicates that the item/column can be resized horizontally. Often
235    /// rendered as arrows pointing left and right with a vertical bar
236    /// separating them.
237    ColResize,
238
239    /// Indicates that the item/row can be resized vertically. Often rendered as
240    /// arrows pointing up and down with a horizontal bar separating them.
241    RowResize,
242
243    /// Indicates that the something can be scrolled in any direction. Often
244    /// rendered as arrows pointing up, down, left, and right with a dot in the
245    /// middle.
246    AllScroll,
247
248    /// Indicates that something can be zoomed in. Often rendered as a
249    /// magnifying glass with a "+" in the center of the glass.
250    ZoomIn,
251
252    /// Indicates that something can be zoomed in. Often rendered as a
253    /// magnifying glass with a "-" in the center of the glass.
254    ZoomOut,
255}
256
257impl CursorIcon {
258    /// The name of the cursor icon as defined in w3c standard.
259    ///
260    /// This name most of the time could be passed as is to cursor loading
261    /// libraries on X11/Wayland and could be used as-is on web.
262    ///
263    /// # Examples
264    ///
265    /// ```no_run
266    /// use cursor_icon::CursorIcon;
267    /// use wayland_cursor::CursorTheme;
268    ///
269    /// # use wayland_client::Connection;
270    /// # use wayland_client::protocol::wl_shm::WlShm;
271    /// # fn test(conn: &Connection, shm: WlShm) -> Result<(), Box<dyn std::error::Error>> {
272    /// // Choose a cursor to load.
273    /// let cursor = CursorIcon::Help;
274    ///
275    /// // Load the Wayland cursor theme.
276    /// let mut cursor_theme = CursorTheme::load(conn, shm, 32)?;
277    ///
278    /// // Load the cursor.
279    /// let cursor = cursor_theme.get_cursor(cursor.name());
280    /// if let Some(cursor) = cursor {
281    ///     println!("Total number of images: {}", cursor.image_count());
282    /// }
283    /// # Ok(())
284    /// # }
285    /// ```
286    pub fn name(&self) -> &'static str {
287        match self {
288            CursorIcon::Default => "default",
289            CursorIcon::ContextMenu => "context-menu",
290            CursorIcon::Help => "help",
291            CursorIcon::Pointer => "pointer",
292            CursorIcon::Progress => "progress",
293            CursorIcon::Wait => "wait",
294            CursorIcon::Cell => "cell",
295            CursorIcon::Crosshair => "crosshair",
296            CursorIcon::Text => "text",
297            CursorIcon::VerticalText => "vertical-text",
298            CursorIcon::Alias => "alias",
299            CursorIcon::Copy => "copy",
300            CursorIcon::Move => "move",
301            CursorIcon::NoDrop => "no-drop",
302            CursorIcon::NotAllowed => "not-allowed",
303            CursorIcon::Grab => "grab",
304            CursorIcon::Grabbing => "grabbing",
305            CursorIcon::EResize => "e-resize",
306            CursorIcon::NResize => "n-resize",
307            CursorIcon::NeResize => "ne-resize",
308            CursorIcon::NwResize => "nw-resize",
309            CursorIcon::SResize => "s-resize",
310            CursorIcon::SeResize => "se-resize",
311            CursorIcon::SwResize => "sw-resize",
312            CursorIcon::WResize => "w-resize",
313            CursorIcon::EwResize => "ew-resize",
314            CursorIcon::NsResize => "ns-resize",
315            CursorIcon::NeswResize => "nesw-resize",
316            CursorIcon::NwseResize => "nwse-resize",
317            CursorIcon::ColResize => "col-resize",
318            CursorIcon::RowResize => "row-resize",
319            CursorIcon::AllScroll => "all-scroll",
320            CursorIcon::ZoomIn => "zoom-in",
321            CursorIcon::ZoomOut => "zoom-out",
322        }
323    }
324
325    /// A list of alternative names for the cursor icon as commonly found in
326    /// legacy Xcursor themes.
327    ///
328    /// This should only be used as a fallback in case the cursor theme does not
329    /// adhere to the w3c standard.
330    pub fn alt_names(&self) -> &[&'static str] {
331        match self {
332            CursorIcon::Default => &["left_ptr", "arrow", "top_left_arrow", "left_arrow"],
333            CursorIcon::ContextMenu => &[],
334            CursorIcon::Help => &["question_arrow", "whats_this"],
335            CursorIcon::Pointer => &["hand2", "hand1", "hand", "pointing_hand"],
336            CursorIcon::Progress => &["left_ptr_watch", "half-busy"],
337            CursorIcon::Wait => &["watch"],
338            CursorIcon::Cell => &["plus"],
339            CursorIcon::Crosshair => &["cross"],
340            CursorIcon::Text => &["xterm", "ibeam"],
341            CursorIcon::VerticalText => &[],
342            CursorIcon::Alias => &["link"],
343            CursorIcon::Copy => &[],
344            CursorIcon::Move => &[],
345            CursorIcon::NoDrop => &["circle"],
346            CursorIcon::NotAllowed => &["crossed_circle", "forbidden"],
347            CursorIcon::Grab => &["openhand", "fleur"],
348            CursorIcon::Grabbing => &["closedhand"],
349            CursorIcon::EResize => &["right_side"],
350            CursorIcon::NResize => &["top_side"],
351            CursorIcon::NeResize => &["top_right_corner"],
352            CursorIcon::NwResize => &["top_left_corner"],
353            CursorIcon::SResize => &["bottom_side"],
354            CursorIcon::SeResize => &["bottom_right_corner"],
355            CursorIcon::SwResize => &["bottom_left_corner"],
356            CursorIcon::WResize => &["left_side"],
357            CursorIcon::EwResize => &["h_double_arrow", "size_hor"],
358            CursorIcon::NsResize => &["v_double_arrow", "size_ver"],
359            CursorIcon::NeswResize => &["fd_double_arrow", "size_bdiag"],
360            CursorIcon::NwseResize => &["bd_double_arrow", "size_fdiag"],
361            CursorIcon::ColResize => &["split_h", "h_double_arrow", "sb_h_double_arrow"],
362            CursorIcon::RowResize => &["split_v", "v_double_arrow", "sb_v_double_arrow"],
363            CursorIcon::AllScroll => &["size_all"],
364            CursorIcon::ZoomIn => &[],
365            CursorIcon::ZoomOut => &[],
366        }
367    }
368}
369
370impl core::fmt::Display for CursorIcon {
371    fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
372        f.write_str(self.name())
373    }
374}
375
376impl core::str::FromStr for CursorIcon {
377    type Err = ParseError;
378
379    /// Parse a string slice into [`CursorIcon`].
380    ///
381    /// The `name` is a lower kebab case [`CursorIcon`] varaint name, e.g.
382    /// `nesw-resize`. The set of possible valid `name` values matches exactly
383    /// the set of [`CursorIcon::name`] outputs.
384    fn from_str(name: &str) -> Result<Self, Self::Err> {
385        match name {
386            "default" => Ok(CursorIcon::Default),
387            "context-menu" => Ok(CursorIcon::ContextMenu),
388            "help" => Ok(CursorIcon::Help),
389            "pointer" => Ok(CursorIcon::Pointer),
390            "progress" => Ok(CursorIcon::Progress),
391            "wait" => Ok(CursorIcon::Wait),
392            "cell" => Ok(CursorIcon::Cell),
393            "crosshair" => Ok(CursorIcon::Crosshair),
394            "text" => Ok(CursorIcon::Text),
395            "vertical-text" => Ok(CursorIcon::VerticalText),
396            "alias" => Ok(CursorIcon::Alias),
397            "copy" => Ok(CursorIcon::Copy),
398            "move" => Ok(CursorIcon::Move),
399            "no-drop" => Ok(CursorIcon::NoDrop),
400            "not-allowed" => Ok(CursorIcon::NotAllowed),
401            "grab" => Ok(CursorIcon::Grab),
402            "grabbing" => Ok(CursorIcon::Grabbing),
403            "e-resize" => Ok(CursorIcon::EResize),
404            "n-resize" => Ok(CursorIcon::NResize),
405            "ne-resize" => Ok(CursorIcon::NeResize),
406            "nw-resize" => Ok(CursorIcon::NwResize),
407            "s-resize" => Ok(CursorIcon::SResize),
408            "se-resize" => Ok(CursorIcon::SeResize),
409            "sw-resize" => Ok(CursorIcon::SwResize),
410            "w-resize" => Ok(CursorIcon::WResize),
411            "ew-resize" => Ok(CursorIcon::EwResize),
412            "ns-resize" => Ok(CursorIcon::NsResize),
413            "nesw-resize" => Ok(CursorIcon::NeswResize),
414            "nwse-resize" => Ok(CursorIcon::NwseResize),
415            "col-resize" => Ok(CursorIcon::ColResize),
416            "row-resize" => Ok(CursorIcon::RowResize),
417            "all-scroll" => Ok(CursorIcon::AllScroll),
418            "zoom-in" => Ok(CursorIcon::ZoomIn),
419            "zoom-out" => Ok(CursorIcon::ZoomOut),
420            _ => Err(ParseError { _private: () }),
421        }
422    }
423}
424
425/// An error which could be returned when parsing [`CursorIcon`].
426///
427/// This occurs when the [`FromStr`] implementation of [`CursorIcon`] fails.
428///
429/// [`FromStr`]: core::str::FromStr
430#[derive(Debug, PartialEq, Eq)]
431pub struct ParseError {
432    _private: (),
433}
434
435impl core::fmt::Display for ParseError {
436    fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
437        f.write_str("failed to parse cursor icon")
438    }
439}
440
441#[cfg(feature = "std")]
442impl std::error::Error for ParseError {}