efl/src/bindings/mono/eina_mono/eina_inlist.cs

463 lines
13 KiB
C#
Raw Normal View History

/*
* Copyright 2019 by its authors. See AUTHORS.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#pragma warning disable 1591
using System;
using System.Runtime.InteropServices;
using System.Collections.Generic;
using System.ComponentModel;
csharp: Change to new class API. Summary: As discussed in T7204: - Eo Interfaces/mixins -> C# Interfaces with concrete class implementations - Eo Regular/Abstracts -> Proper C# classes - Added some new generators and helper methods. - Refactored the class generator, splitting into helper methods Eo handles now are stored only in the "root" class in any given inheritance tree (generally, Efl.Object), and accessible to each child. Methods also are defined in a single place instead of repeatedly generated in everyfile, reducing the size of the generated .dll from 30MB to around 4.5MB. Mixins are generated as C# interfaces but any regular class it inherits from is lost, as we can't have interfaces inheriting from regular classes. This will be dealt with in a later commit. Summary of API Changes: - Merged Inherit/Concrete classes. (These suffixes disappear from regular classes). - Interface still have implementations with 'Concrete' suffix for when they are returned from methods. - Removed 'I' from interface names. - Removed interfaces for regular/abstract Eo classes. - Concrete classes for interfaces/mixins hold the event argument struct. - Removed '_' from classes, enums, structs, etc, as indicated in C# naming conventions. - Namespaces are now Camel.Cased. - Renamed IWrapper's raw_handle/raw_klass to NativeHandle/NativeClass Also renamed the test classes as after the namespace change, the test namespace Test can conflict with the helper Test namespace. (And use more meaningful names than Test.Testing...) Also Fixes T7336 by removing a deprecated example and adding efl_loop_timer_example to build system. Fixes T7451 by hiding the class_get DllImports and renaming the IWrapper fields. The native handlers are used in the manual binding. Still need to work: - As there are still some events names clashing (e.g. Efl.Ui.Bg with "resize" from Efl.Gfx.Entity and Efl.Gfx.Image), Events are currently declared on the interface and implemented "namespaced" in the classes, requiring the cast to the interface to access the event. - The Mixin Conundrum. Mixin inheritance will be dealt in a future commit. Depends on D7260 Reviewers: segfaultxavi, vitor.sousa, felipealmeida, Jaehyun_Cho Reviewed By: vitor.sousa Subscribers: cedric, #reviewers, #committers Tags: #efl Maniphest Tasks: T7451, T7336 Differential Revision: https://phab.enlightenment.org/D7262
2018-11-29 15:04:37 -08:00
using static Eina.TraitFunctions;
using static Eina.InlistNativeFunctions;
using Eina.Callbacks;
namespace Eina
{
[EditorBrowsable(EditorBrowsableState.Never)]
public static class InlistNativeFunctions
{
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_append(IntPtr in_list, IntPtr in_item);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_prepend(IntPtr in_list, IntPtr in_item);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_append_relative(IntPtr in_list, IntPtr in_item, IntPtr in_relative);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_prepend_relative(IntPtr in_list, IntPtr in_item, IntPtr in_relative);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_remove(IntPtr in_list, IntPtr in_item);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_find(IntPtr in_list, IntPtr in_item);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_promote(IntPtr list, IntPtr item);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_demote(IntPtr list, IntPtr item);
[DllImport(efl.Libs.Eina)] internal static extern uint
eina_inlist_count(IntPtr list);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_iterator_new(IntPtr in_list);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_accessor_new(IntPtr in_list);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_sorted_insert(IntPtr list, IntPtr item, IntPtr func);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_sorted_state_new();
[DllImport(efl.Libs.Eina)] internal static extern int
eina_inlist_sorted_state_init(IntPtr state, IntPtr list);
[DllImport(efl.Libs.Eina)] internal static extern void
eina_inlist_sorted_state_free(IntPtr state);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_sorted_state_insert(IntPtr list, IntPtr item, IntPtr func, IntPtr state);
[DllImport(efl.Libs.Eina)] internal static extern IntPtr
eina_inlist_sort(IntPtr head, IntPtr func);
[DllImport(efl.Libs.CustomExports)] internal static extern IntPtr
eina_inlist_first_custom_export_mono(IntPtr list);
[DllImport(efl.Libs.CustomExports)] internal static extern IntPtr
eina_inlist_last_custom_export_mono(IntPtr list);
[DllImport(efl.Libs.CustomExports)] internal static extern IntPtr
eina_inlist_next_custom_export_mono(IntPtr list);
[DllImport(efl.Libs.CustomExports)] internal static extern IntPtr
eina_inlist_prev_custom_export_mono(IntPtr list);
[DllImport(efl.Libs.CustomExports)] internal static extern IntPtr
eina_inlist_iterator_wrapper_new_custom_export_mono(IntPtr in_list);
}
/// <summary>Wrapper around an inplace list.
/// <para>Since EFL 1.23.</para>
/// </summary>
public class Inlist<T> : IEnumerable<T>, IDisposable
{
[EditorBrowsable(EditorBrowsableState.Never)]
public IntPtr Handle {get;set;} = IntPtr.Zero;
/// <summary>Whether this wrapper owns the native buffer.
/// <para>Since EFL 1.23.</para>
/// </summary>
public bool Own {get;set;}
/// <summary>Who is in charge of releasing the resources wrapped by
/// this instance.
/// <para>Since EFL 1.23.</para>
/// </summary>
public bool OwnContent {get;set;}
public int Length
{
get { return Count(); }
}
private void InitNew()
{
Handle = IntPtr.Zero;
Own = true;
OwnContent = true;
}
private IntPtr InternalFirst()
{
return eina_inlist_first_custom_export_mono(Handle);
}
private IntPtr InternalLast()
{
return eina_inlist_last_custom_export_mono(Handle);
}
private IntPtr InternalAt(int idx)
{
if (idx < 0)
{
return IntPtr.Zero;
}
IntPtr curr = Handle;
for (int n = 0; n != idx && curr != IntPtr.Zero; ++n)
{
curr = InternalNext(curr);
}
return curr;
}
private static IntPtr InternalNext(IntPtr inlist)
{
return eina_inlist_next_custom_export_mono(inlist);
}
private static IntPtr InternalPrev(IntPtr inlist)
{
return eina_inlist_prev_custom_export_mono(inlist);
}
/// <summary>
/// Default constructor.
/// <para>Since EFL 1.23.</para>
/// </summary>
public Inlist()
{
InitNew();
}
[EditorBrowsable(EditorBrowsableState.Never)]
public Inlist(IntPtr handle, bool own)
{
Handle = handle;
Own = own;
OwnContent = own;
}
[EditorBrowsable(EditorBrowsableState.Never)]
public Inlist(IntPtr handle, bool own, bool ownContent)
{
Handle = handle;
Own = own;
OwnContent = ownContent;
}
/// <summary>
/// Finalizer to be called from the Garbage Collector.
/// <para>Since EFL 1.23.</para>
/// </summary>
~Inlist()
{
Dispose(false);
}
/// <summary>Disposes of this wrapper, releasing the native array if owned.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <param name="disposing">True if this was called from <see cref="Dispose()"/> public method. False if
/// called from the C# finalizer.</param>
protected virtual void Dispose(bool disposing)
{
IntPtr h = Handle;
Handle = IntPtr.Zero;
if (h == IntPtr.Zero)
{
return;
}
if (OwnContent)
{
for (IntPtr curr = h; curr != IntPtr.Zero; curr = InternalNext(curr))
{
NativeFreeInlistNodeElement<T>(curr);
}
}
if (Own)
{
while (h != IntPtr.Zero)
{
var aux = h;
h = eina_inlist_remove(h, h);
NativeFreeInlistNode<T>(aux, false);
}
}
}
/// <summary>Releases the native resources held by this instance.
/// <para>Since EFL 1.23.</para>
/// </summary>
public void Dispose()
{
Dispose(true);
GC.SuppressFinalize(this);
}
/// <summary>Releases the native resources held by this instance.
/// <para>Since EFL 1.23.</para>
/// </summary>
public void Free()
{
Dispose();
}
/// <summary>
/// Releases the native inlist.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <returns>The native inlist.</returns>
public IntPtr Release()
{
IntPtr h = Handle;
Handle = IntPtr.Zero;
return h;
}
/// <summary>Sets all ownership.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <param name="ownAll">If the hash own for all ownerships.</param>
public void SetOwnership(bool ownAll)
{
Own = ownAll;
OwnContent = ownAll;
}
/// <summary>Sets own individually.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <param name="own">If own the object.</param>
/// <param name="ownContent">If own the content's object.</param>
public void SetOwnership(bool own, bool ownContent)
{
Own = own;
OwnContent = ownContent;
}
/// <summary>
/// Gets the count of the number of items.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <returns>The number of members in the list.</returns>
public int Count()
{
return (int)eina_inlist_count(Handle);
}
/// <summary>
/// Cleanup the inlist.
/// <para>Since EFL 1.23.</para>
/// </summary>
public void Clean()
{
while (Handle != IntPtr.Zero)
{
var aux = Handle;
Handle = eina_inlist_remove(Handle, Handle);
NativeFreeInlistNode<T>(aux, OwnContent);
}
}
/// <summary>
/// Adds a new value to the end.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <param name="val">The value to be added.</param>
public void Append(T val)
{
IntPtr node = ManagedToNativeAllocInlistNode(val);
Handle = eina_inlist_append(Handle, node);
}
/// <summary>
/// Adds a new value to the begin.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <param name="val">The value to be added.</param>
public void Prepend(T val)
{
IntPtr node = ManagedToNativeAllocInlistNode(val);
Handle = eina_inlist_prepend(Handle, node);
}
/// <summary>
/// Removes value at the specified position from list.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <param name="idx">The given position.</param>
public void Remove(int idx)
{
IntPtr node = InternalAt(idx);
Handle = eina_inlist_remove(Handle, node);
NativeFreeInlistNode<T>(node, OwnContent);
}
/// <summary>
/// Returns the element of the list at the specified position.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <param name="idx">The position of the desired element.</param>
/// <returns>The element at the specified position</returns>
public T At(int idx)
{
IntPtr node = InternalAt(idx);
if (node == IntPtr.Zero)
{
throw new IndexOutOfRangeException();
}
return NativeToManagedInlistNode<T>(node);
}
/// <summary>
/// Replaces the element at the specified position.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <param name="idx">The position of the desired element.</param>
/// <param name="val">The value of the element to be inserted.</param>
public void DataSet(int idx, T val)
{
IntPtr old = InternalAt(idx);
if (old == IntPtr.Zero)
{
throw new IndexOutOfRangeException();
}
IntPtr new_node = ManagedToNativeAllocInlistNode(val);
Handle = eina_inlist_append_relative(Handle, new_node, old);
Handle = eina_inlist_remove(Handle, old);
NativeFreeInlistNode<T>(old, OwnContent);
}
/// <summary>
/// Accessor by index to the elements of this list.
/// <para>Since EFL 1.23.</para>
/// </summary>
public T this[int idx]
{
get
{
return At(idx);
}
set
{
DataSet(idx, value);
}
}
/// <summary>
/// Returns a array containing all of the elements in proper sequence.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <returns>A array</returns>
public T[] ToArray()
{
var managed = new T[Count()];
int i = 0;
for (IntPtr curr = Handle; curr != IntPtr.Zero; ++i, curr = InternalNext(curr))
{
managed[i] = NativeToManagedInlistNode<T>(curr);
}
return managed;
}
/// <summary>
/// Adds a array to the end.
/// <para>Since EFL 1.23.</para>
/// </summary>
/// <param name="values">The values to be added.</param>
public void AppendArray(T[] values)
{
foreach (T v in values)
{
Append(v);
}
}
/// <summary> Gets an Iterator for this list.
/// <para>Since EFL 1.23.</para>
/// </summary>
csharp: Change to new class API. Summary: As discussed in T7204: - Eo Interfaces/mixins -> C# Interfaces with concrete class implementations - Eo Regular/Abstracts -> Proper C# classes - Added some new generators and helper methods. - Refactored the class generator, splitting into helper methods Eo handles now are stored only in the "root" class in any given inheritance tree (generally, Efl.Object), and accessible to each child. Methods also are defined in a single place instead of repeatedly generated in everyfile, reducing the size of the generated .dll from 30MB to around 4.5MB. Mixins are generated as C# interfaces but any regular class it inherits from is lost, as we can't have interfaces inheriting from regular classes. This will be dealt with in a later commit. Summary of API Changes: - Merged Inherit/Concrete classes. (These suffixes disappear from regular classes). - Interface still have implementations with 'Concrete' suffix for when they are returned from methods. - Removed 'I' from interface names. - Removed interfaces for regular/abstract Eo classes. - Concrete classes for interfaces/mixins hold the event argument struct. - Removed '_' from classes, enums, structs, etc, as indicated in C# naming conventions. - Namespaces are now Camel.Cased. - Renamed IWrapper's raw_handle/raw_klass to NativeHandle/NativeClass Also renamed the test classes as after the namespace change, the test namespace Test can conflict with the helper Test namespace. (And use more meaningful names than Test.Testing...) Also Fixes T7336 by removing a deprecated example and adding efl_loop_timer_example to build system. Fixes T7451 by hiding the class_get DllImports and renaming the IWrapper fields. The native handlers are used in the manual binding. Still need to work: - As there are still some events names clashing (e.g. Efl.Ui.Bg with "resize" from Efl.Gfx.Entity and Efl.Gfx.Image), Events are currently declared on the interface and implemented "namespaced" in the classes, requiring the cast to the interface to access the event. - The Mixin Conundrum. Mixin inheritance will be dealt in a future commit. Depends on D7260 Reviewers: segfaultxavi, vitor.sousa, felipealmeida, Jaehyun_Cho Reviewed By: vitor.sousa Subscribers: cedric, #reviewers, #committers Tags: #efl Maniphest Tasks: T7451, T7336 Differential Revision: https://phab.enlightenment.org/D7262
2018-11-29 15:04:37 -08:00
public Eina.Iterator<T> GetIterator()
{
return new Eina.Iterator<T>(eina_inlist_iterator_wrapper_new_custom_export_mono(Handle), true);
}
/// <summary> Gets an Enumerator for this list.
/// <para>Since EFL 1.23.</para>
/// </summary>
public IEnumerator<T> GetEnumerator()
{
for (IntPtr curr = Handle; curr != IntPtr.Zero; curr = InternalNext(curr))
{
yield return NativeToManagedInlistNode<T>(curr);
}
}
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
{
return this.GetEnumerator();
}
/// <summary> Gets an Accessor for this List.
/// <para>Since EFL 1.23.</para>
/// </summary>
csharp: Change to new class API. Summary: As discussed in T7204: - Eo Interfaces/mixins -> C# Interfaces with concrete class implementations - Eo Regular/Abstracts -> Proper C# classes - Added some new generators and helper methods. - Refactored the class generator, splitting into helper methods Eo handles now are stored only in the "root" class in any given inheritance tree (generally, Efl.Object), and accessible to each child. Methods also are defined in a single place instead of repeatedly generated in everyfile, reducing the size of the generated .dll from 30MB to around 4.5MB. Mixins are generated as C# interfaces but any regular class it inherits from is lost, as we can't have interfaces inheriting from regular classes. This will be dealt with in a later commit. Summary of API Changes: - Merged Inherit/Concrete classes. (These suffixes disappear from regular classes). - Interface still have implementations with 'Concrete' suffix for when they are returned from methods. - Removed 'I' from interface names. - Removed interfaces for regular/abstract Eo classes. - Concrete classes for interfaces/mixins hold the event argument struct. - Removed '_' from classes, enums, structs, etc, as indicated in C# naming conventions. - Namespaces are now Camel.Cased. - Renamed IWrapper's raw_handle/raw_klass to NativeHandle/NativeClass Also renamed the test classes as after the namespace change, the test namespace Test can conflict with the helper Test namespace. (And use more meaningful names than Test.Testing...) Also Fixes T7336 by removing a deprecated example and adding efl_loop_timer_example to build system. Fixes T7451 by hiding the class_get DllImports and renaming the IWrapper fields. The native handlers are used in the manual binding. Still need to work: - As there are still some events names clashing (e.g. Efl.Ui.Bg with "resize" from Efl.Gfx.Entity and Efl.Gfx.Image), Events are currently declared on the interface and implemented "namespaced" in the classes, requiring the cast to the interface to access the event. - The Mixin Conundrum. Mixin inheritance will be dealt in a future commit. Depends on D7260 Reviewers: segfaultxavi, vitor.sousa, felipealmeida, Jaehyun_Cho Reviewed By: vitor.sousa Subscribers: cedric, #reviewers, #committers Tags: #efl Maniphest Tasks: T7451, T7336 Differential Revision: https://phab.enlightenment.org/D7262
2018-11-29 15:04:37 -08:00
public Eina.Accessor<T> GetAccessor()
{
csharp: Change to new class API. Summary: As discussed in T7204: - Eo Interfaces/mixins -> C# Interfaces with concrete class implementations - Eo Regular/Abstracts -> Proper C# classes - Added some new generators and helper methods. - Refactored the class generator, splitting into helper methods Eo handles now are stored only in the "root" class in any given inheritance tree (generally, Efl.Object), and accessible to each child. Methods also are defined in a single place instead of repeatedly generated in everyfile, reducing the size of the generated .dll from 30MB to around 4.5MB. Mixins are generated as C# interfaces but any regular class it inherits from is lost, as we can't have interfaces inheriting from regular classes. This will be dealt with in a later commit. Summary of API Changes: - Merged Inherit/Concrete classes. (These suffixes disappear from regular classes). - Interface still have implementations with 'Concrete' suffix for when they are returned from methods. - Removed 'I' from interface names. - Removed interfaces for regular/abstract Eo classes. - Concrete classes for interfaces/mixins hold the event argument struct. - Removed '_' from classes, enums, structs, etc, as indicated in C# naming conventions. - Namespaces are now Camel.Cased. - Renamed IWrapper's raw_handle/raw_klass to NativeHandle/NativeClass Also renamed the test classes as after the namespace change, the test namespace Test can conflict with the helper Test namespace. (And use more meaningful names than Test.Testing...) Also Fixes T7336 by removing a deprecated example and adding efl_loop_timer_example to build system. Fixes T7451 by hiding the class_get DllImports and renaming the IWrapper fields. The native handlers are used in the manual binding. Still need to work: - As there are still some events names clashing (e.g. Efl.Ui.Bg with "resize" from Efl.Gfx.Entity and Efl.Gfx.Image), Events are currently declared on the interface and implemented "namespaced" in the classes, requiring the cast to the interface to access the event. - The Mixin Conundrum. Mixin inheritance will be dealt in a future commit. Depends on D7260 Reviewers: segfaultxavi, vitor.sousa, felipealmeida, Jaehyun_Cho Reviewed By: vitor.sousa Subscribers: cedric, #reviewers, #committers Tags: #efl Maniphest Tasks: T7451, T7336 Differential Revision: https://phab.enlightenment.org/D7262
2018-11-29 15:04:37 -08:00
return new Eina.AccessorInList<T>(eina_inlist_accessor_new(Handle), Ownership.Managed);
}
}
}