summary refs log tree commit diff stats
path: root/src/list.rs
diff options
context:
space:
mode:
Diffstat (limited to 'src/list.rs')
-rw-r--r--src/list.rs413
1 files changed, 413 insertions, 0 deletions
diff --git a/src/list.rs b/src/list.rs
new file mode 100644
index 0000000..257ef95
--- /dev/null
+++ b/src/list.rs
@@ -0,0 +1,413 @@
+// This file is part of Hexchat Plugin API Bindings for Rust
+// Copyright (C) 2022 Soni L.
+//
+// This program is free software: you can redistribute it and/or modify
+// it under the terms of the GNU General Public License as
+// published by the Free Software Foundation, either version 3 of the
+// License, or (at your option) any later version.
+//
+// This program is distributed in the hope that it will be useful,
+// but WITHOUT ANY WARRANTY; without even the implied warranty of
+// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+// GNU General Public License for more details.
+//
+// You should have received a copy of the GNU General Public License
+// along with this program.  If not, see <https://www.gnu.org/licenses/>.
+
+//! List support module.
+
+use std::borrow::Cow;
+use std::cell::Cell;
+use std::ffi::CStr;
+use std::rc::Rc;
+
+use crate::Context;
+use crate::wrap_context;
+
+mod sealed {
+    pub trait Sealed {
+    }
+}
+
+/// A list. This trait is sealed.
+pub trait List: sealed::Sealed {
+    fn name(&self) -> Cow<'static, str>;
+}
+
+// List types
+
+/// A "channels" list.
+pub struct Contexts;
+
+/// A "dcc" list.
+pub struct Dcc;
+
+/// An "ignore" list.
+pub struct Ignore;
+
+/// A "notify" list.
+pub struct Notify;
+
+/// An "users" list.
+pub struct Users;
+
+/// Entries.
+pub struct Entries<'a, 'ph, T> {
+    pub(crate) context: &'a crate::ValidContext<'a, 'ph>,
+    pub(crate) list: *mut crate::internals::HexchatList,
+    pub(crate) t: &'a T,
+    pub(crate) valid: Rc<Cell<usize>>,
+}
+
+/// Fields.
+pub struct Fields<'a, 'ph, T> {
+    context: &'a crate::ValidContext<'a, 'ph>,
+    list: *mut crate::internals::HexchatList,
+    _t: &'a T,
+    id: usize,
+    valid: Rc<Cell<usize>>,
+}
+
+// impls
+
+impl sealed::Sealed for Contexts {}
+impl sealed::Sealed for Dcc {}
+impl sealed::Sealed for Ignore {}
+impl sealed::Sealed for Notify {}
+impl sealed::Sealed for Users {}
+
+impl List for Contexts {
+    fn name(&self) -> Cow<'static, str> { "channels".into() }
+}
+
+impl List for Dcc {
+    fn name(&self) -> Cow<'static, str> { "dcc".into() }
+}
+
+impl List for Ignore {
+    fn name(&self) -> Cow<'static, str> { "ignore".into() }
+}
+
+impl List for Notify {
+    fn name(&self) -> Cow<'static, str> { "notify".into() }
+}
+
+impl List for Users {
+    fn name(&self) -> Cow<'static, str> { "users".into() }
+}
+
+impl<'a, 'ph, T> Drop for Entries<'a, 'ph, T> {
+    fn drop(&mut self) {
+        let ph = &self.context.ph;
+        unsafe {
+            ph_call!(hexchat_list_free(ph, self.list));
+        }
+    }
+}
+
+impl<'a, 'ph, T> Iterator for Entries<'a, 'ph, T> {
+    type Item = Fields<'a, 'ph, T>;
+
+    fn next(&mut self) -> Option<Self::Item> {
+        let ph = &self.context.ph;
+        let id = self.valid.get();
+        if unsafe {
+            ph_call!(hexchat_list_next(ph, self.list))
+        } != 0 {
+            debug_assert!(self.valid.get() < usize::MAX);
+            self.valid.set(id + 1);
+            Some(Fields {
+                context: self.context,
+                list: self.list,
+                _t: self.t,
+                valid: Rc::clone(&self.valid),
+                id: id,
+            })
+        } else {
+            self.valid.set(usize::MAX);
+            None
+        }
+    }
+}
+
+macro_rules! field {
+    ($(#[$m:meta])* $f:ident, unsafe {$n:expr}, $r:ty, $ph:ident, $c:ident, $($conv:tt)+) => {
+        $(#[$m])* pub fn $f(&self) -> $r {
+            assert_eq!(
+                self.id, self.valid.get(),
+                "instances of Fields are invalidated by next()",
+            );
+            let $ph = &self.context.ph;
+            const NAME: &'static CStr = unsafe {
+                CStr::from_bytes_with_nul_unchecked(
+                    $n.as_bytes()
+                )
+            };
+            match unsafe {
+                ph_call!($c($ph, self.list, NAME.as_ptr()))
+            } {
+                $($conv)+
+            }
+        }
+    };
+    ($(#[$m:meta])* $f:ident, $r:ty, $ph:ident, $c:ident, $($conv:tt)+) => {
+        field!(
+            $(#[$m])* $f,
+            unsafe { ::std::concat!(::std::stringify!($f), "\0") },
+            $r,
+            $ph,
+            $c,
+            $($conv)+
+        );
+    };
+}
+
+macro_rules! field_int {
+    ($(#[$m:meta])* $f:ident, unsafe { $n:expr }) => {
+        field!($(#[$m])* $f, unsafe { $n }, i32, ph, hexchat_list_int, r => r as i32);
+    };
+    ($(#[$m:meta])* $f:ident) => {
+        field_int!($(#[$m])* $f, unsafe { ::std::concat!(::std::stringify!($f), "\0") });
+    };
+}
+
+macro_rules! field_bool {
+    ($(#[$m:meta])* $f:ident, unsafe { $n:expr }) => {
+        field!($(#[$m])* $f, unsafe { $n }, bool, ph, hexchat_list_int, r => r != 0);
+    };
+    ($(#[$m:meta])* $f:ident) => {
+        field_bool!($(#[$m])* $f, unsafe { ::std::concat!(::std::stringify!($f), "\0") });
+    };
+}
+
+macro_rules! field_str {
+    ($(#[$m:meta])* $f:ident, unsafe { $n:expr }) => {
+        field!(
+            $(#[$m])* $f, unsafe { $n }, Option<String>, ph, hexchat_list_str,
+            r if r.is_null() => {
+                None
+            },
+            r => Some(unsafe {
+                CStr::from_ptr(r)
+            }.to_owned().into_string().unwrap())
+        );
+    };
+    ($(#[$m:meta])* $f:ident) => {
+        field_str!($(#[$m])* $f, unsafe { ::std::concat!(::std::stringify!($f), "\0") });
+    };
+}
+
+
+macro_rules! field_time {
+    ($(#[$m:meta])* $f:ident) => {
+        $(#[$m])* pub fn $f(&self) -> ! {
+            todo!()
+        }
+    };
+}
+
+/// Contexts fields.
+impl<'a, 'ph> Fields<'a, 'ph, Contexts> {
+    field_str!(
+        /// The context's name.
+        name, unsafe { "channel" }
+    );
+    field_str!(
+        /// The channel key.
+        channelkey
+    );
+    field_str!(
+        /// The server's channel modes. Requires HexChat 2.12.2+.
+        chanmodes
+    );
+    field_str!(
+        /// The server's channel types.
+        chantypes
+    );
+    field!(
+        /// The context.
+        context, Context<'ph>, ph, hexchat_list_str, r => unsafe {
+            wrap_context(ph, r as *const _)
+        }
+    );
+    field_int!(
+        /// Raw flags about this context.
+        raw_flags, unsafe { "flags" }
+    );
+    field_int!(
+        /// The server ID.
+        server_id, unsafe { "id" }
+    );
+    field_int!(
+        /// The latency to server in milliseconds.
+        lag
+    );
+    field_int!(
+        /// The maximum number of mode changes per line accepted by the server.
+        maxmodes
+    );
+    field_str!(
+        /// The network name.
+        network
+    );
+    field_str!(
+        /// The server's nick prefixes.
+        nickprefixes
+    );
+    field_str!(
+        /// The server's nick modes.
+        nickmodes
+    );
+    field_int!(
+        /// The current length of the send-queue, in bytes.
+        queue_len, unsafe { "queue" }
+    );
+    field_str!(
+        /// The server's name.
+        server
+    );
+    field_int!(
+        /// The context's raw type.
+        raw_type, unsafe { "type" }
+    );
+    field_int!(
+        /// The number of users in the channel.
+        users
+    );
+}
+
+/// Dcc fields.
+impl<'a, 'ph> Fields<'a, 'ph, Dcc> {
+    field_int!(
+        /// The raw 32-bit IPv4 address of the remote user.
+        address32
+    );
+    field_int!(
+        /// The transfer rate (speed), in bytes per second.
+        rate, unsafe { "cps" }
+    );
+    field_str!(
+        /// The destination file path.
+        destfile
+    );
+    field_str!(
+        /// The filename.
+        file
+    );
+    field_str!(
+        /// The remote user's nick.
+        nick
+    );
+    field_int!(
+        /// The listener's port number.
+        port
+    );
+    field_int!(
+        /// File position, LSB 32 bits.
+        raw_pos, unsafe { "pos" }
+    );
+    field_int!(
+        /// File position, MSB 32 bits.
+        raw_poshigh, unsafe { "poshigh" }
+    );
+    field_int!(
+        /// Resumed position, LSB 32 bits.
+        raw_resume, unsafe { "resume" }
+    );
+    field_int!(
+        /// Resumed position, MSB 32 bits.
+        raw_resumehigh, unsafe { "resumehigh" }
+    );
+    field_int!(
+        /// File size, LSB 32 bits.
+        raw_size, unsafe { "size" }
+    );
+    field_int!(
+        /// File size, MSB 32 bits.
+        raw_sizehigh, unsafe { "sizehigh" }
+    );
+    field_int!(
+        /// Raw DCC status.
+        raw_status, unsafe { "status" }
+    );
+    field_int!(
+        /// Raw type.
+        raw_type, unsafe { "type" }
+    );
+}
+
+/// Ignore fields.
+impl<'a, 'ph> Fields<'a, 'ph, Ignore> {
+    field_str!(
+        /// The ignore mask.
+        mask
+    );
+    field_int!(
+        /// Raw ignore flags.
+        raw_flags, unsafe { "flags" }
+    );
+}
+
+/// Notify fields.
+impl<'a, 'ph> Fields<'a, 'ph, Notify> {
+    field_str!(
+        /// Comma-separated list of networks this notify entry applies to.
+        networks
+    );
+    field_str!(
+        /// The nick.
+        nick
+    );
+    field_int!(
+        /// Raw flags.
+        raw_flags, unsafe { "flags" }
+    );
+    field_time!(
+        /// Time when the user went online. [NYI]
+        on
+    );
+    field_time!(
+        /// Time when the user went offline. [NYI]
+        off
+    );
+    field_time!(
+        /// Time when the user was last seen. [NYI]
+        seen
+    );
+}
+
+/// Users fields.
+impl<'a, 'ph> Fields<'a, 'ph, Users> {
+    field_str!(
+        /// Account name.
+        account
+    );
+    field_bool!(
+        /// Whether the user is away.
+        away
+    );
+    field_time!(
+        /// Time when the user last talked. [NYI]
+        lasttalk
+    );
+    field_str!(
+        /// User's nick.
+        nick
+    );
+    field_str!(
+        /// User's user and hostname (`user@host`).
+        host
+    );
+    field_str!(
+        /// User's prefix.
+        prefix
+    );
+    field_str!(
+        /// User's userdata.
+        userdata, unsafe { "realname" }
+    );
+    field_bool!(
+        /// Whether the user is selected in the UI.
+        selected
+    );
+}