Change LGPL-2.1+ to LGPL-2.1-or-later
[platform/upstream/glib.git] / gio / gbytesicon.c
1 /* GIO - GLib Input, Output and Streaming Library
2  *
3  * Copyright © 2013 Canonical Limited
4  *
5  * SPDX-License-Identifier: LGPL-2.1-or-later
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General
18  * Public License along with this library; if not, see <http://www.gnu.org/licenses/>.
19  *
20  * Author: Ryan Lortie <desrt@desrt.ca>
21  */
22
23 #include "config.h"
24
25 #include "gbytesicon.h"
26 #include "gbytes.h"
27 #include "gicon.h"
28 #include "glibintl.h"
29 #include "gloadableicon.h"
30 #include "gmemoryinputstream.h"
31 #include "gtask.h"
32 #include "gioerror.h"
33
34
35 /**
36  * SECTION:gbytesicon
37  * @short_description: An icon stored in memory as a GBytes
38  * @include: gio/gio.h
39  * @see_also: #GIcon, #GLoadableIcon, #GBytes
40  *
41  * #GBytesIcon specifies an image held in memory in a common format (usually
42  * png) to be used as icon.
43  *
44  * Since: 2.38
45  **/
46
47 typedef GObjectClass GBytesIconClass;
48
49 struct _GBytesIcon
50 {
51   GObject parent_instance;
52
53   GBytes *bytes;
54 };
55
56 enum
57 {
58   PROP_0,
59   PROP_BYTES
60 };
61
62 static void g_bytes_icon_icon_iface_init          (GIconIface          *iface);
63 static void g_bytes_icon_loadable_icon_iface_init (GLoadableIconIface  *iface);
64 G_DEFINE_TYPE_WITH_CODE (GBytesIcon, g_bytes_icon, G_TYPE_OBJECT,
65                          G_IMPLEMENT_INTERFACE (G_TYPE_ICON, g_bytes_icon_icon_iface_init)
66                          G_IMPLEMENT_INTERFACE (G_TYPE_LOADABLE_ICON, g_bytes_icon_loadable_icon_iface_init))
67
68 static void
69 g_bytes_icon_get_property (GObject    *object,
70                            guint       prop_id,
71                            GValue     *value,
72                            GParamSpec *pspec)
73 {
74   GBytesIcon *icon = G_BYTES_ICON (object);
75
76   switch (prop_id)
77     {
78       case PROP_BYTES:
79         g_value_set_boxed (value, icon->bytes);
80         break;
81
82       default:
83         G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
84     }
85 }
86
87 static void
88 g_bytes_icon_set_property (GObject      *object,
89                           guint         prop_id,
90                           const GValue *value,
91                           GParamSpec   *pspec)
92 {
93   GBytesIcon *icon = G_BYTES_ICON (object);
94
95   switch (prop_id)
96     {
97       case PROP_BYTES:
98         icon->bytes = g_value_dup_boxed (value);
99         break;
100
101       default:
102         G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
103     }
104 }
105
106 static void
107 g_bytes_icon_finalize (GObject *object)
108 {
109   GBytesIcon *icon;
110
111   icon = G_BYTES_ICON (object);
112
113   g_bytes_unref (icon->bytes);
114
115   G_OBJECT_CLASS (g_bytes_icon_parent_class)->finalize (object);
116 }
117
118 static void
119 g_bytes_icon_class_init (GBytesIconClass *klass)
120 {
121   GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
122
123   gobject_class->get_property = g_bytes_icon_get_property;
124   gobject_class->set_property = g_bytes_icon_set_property;
125   gobject_class->finalize = g_bytes_icon_finalize;
126
127   /**
128    * GBytesIcon:bytes:
129    *
130    * The bytes containing the icon.
131    */
132   g_object_class_install_property (gobject_class, PROP_BYTES,
133                                    g_param_spec_boxed ("bytes",
134                                                        P_("bytes"),
135                                                        P_("The bytes containing the icon"),
136                                                        G_TYPE_BYTES,
137                                                        G_PARAM_CONSTRUCT_ONLY | G_PARAM_READWRITE | G_PARAM_STATIC_STRINGS));
138 }
139
140 static void
141 g_bytes_icon_init (GBytesIcon *bytes)
142 {
143 }
144
145 /**
146  * g_bytes_icon_new:
147  * @bytes: a #GBytes.
148  *
149  * Creates a new icon for a bytes.
150  *
151  * This cannot fail, but loading and interpreting the bytes may fail later on
152  * (for example, if g_loadable_icon_load() is called) if the image is invalid.
153  *
154  * Returns: (transfer full) (type GBytesIcon): a #GIcon for the given
155  *   @bytes.
156  *
157  * Since: 2.38
158  **/
159 GIcon *
160 g_bytes_icon_new (GBytes *bytes)
161 {
162   g_return_val_if_fail (bytes != NULL, NULL);
163
164   return g_object_new (G_TYPE_BYTES_ICON, "bytes", bytes, NULL);
165 }
166
167 /**
168  * g_bytes_icon_get_bytes:
169  * @icon: a #GIcon.
170  *
171  * Gets the #GBytes associated with the given @icon.
172  *
173  * Returns: (transfer none): a #GBytes.
174  *
175  * Since: 2.38
176  **/
177 GBytes *
178 g_bytes_icon_get_bytes (GBytesIcon *icon)
179 {
180   g_return_val_if_fail (G_IS_BYTES_ICON (icon), NULL);
181
182   return icon->bytes;
183 }
184
185 static guint
186 g_bytes_icon_hash (GIcon *icon)
187 {
188   GBytesIcon *bytes_icon = G_BYTES_ICON (icon);
189
190   return g_bytes_hash (bytes_icon->bytes);
191 }
192
193 static gboolean
194 g_bytes_icon_equal (GIcon *icon1,
195                     GIcon *icon2)
196 {
197   GBytesIcon *bytes1 = G_BYTES_ICON (icon1);
198   GBytesIcon *bytes2 = G_BYTES_ICON (icon2);
199
200   return g_bytes_equal (bytes1->bytes, bytes2->bytes);
201 }
202
203 static GVariant *
204 g_bytes_icon_serialize (GIcon *icon)
205 {
206   GBytesIcon *bytes_icon = G_BYTES_ICON (icon);
207
208   return g_variant_new ("(sv)", "bytes",
209                         g_variant_new_from_bytes (G_VARIANT_TYPE_BYTESTRING, bytes_icon->bytes, TRUE));
210 }
211
212 static void
213 g_bytes_icon_icon_iface_init (GIconIface *iface)
214 {
215   iface->hash = g_bytes_icon_hash;
216   iface->equal = g_bytes_icon_equal;
217   iface->serialize = g_bytes_icon_serialize;
218 }
219
220 static GInputStream *
221 g_bytes_icon_load (GLoadableIcon  *icon,
222                    int            size,
223                    char          **type,
224                    GCancellable   *cancellable,
225                    GError        **error)
226 {
227   GBytesIcon *bytes_icon = G_BYTES_ICON (icon);
228
229   if (type)
230     *type = NULL;
231
232   return g_memory_input_stream_new_from_bytes (bytes_icon->bytes);
233 }
234
235 static void
236 g_bytes_icon_load_async (GLoadableIcon       *icon,
237                          int                  size,
238                          GCancellable        *cancellable,
239                          GAsyncReadyCallback  callback,
240                          gpointer             user_data)
241 {
242   GBytesIcon *bytes_icon = G_BYTES_ICON (icon);
243   GTask *task;
244
245   task = g_task_new (icon, cancellable, callback, user_data);
246   g_task_set_source_tag (task, g_bytes_icon_load_async);
247   g_task_return_pointer (task, g_memory_input_stream_new_from_bytes (bytes_icon->bytes), g_object_unref);
248   g_object_unref (task);
249 }
250
251 static GInputStream *
252 g_bytes_icon_load_finish (GLoadableIcon  *icon,
253                           GAsyncResult   *res,
254                           char          **type,
255                           GError        **error)
256 {
257   g_return_val_if_fail (g_task_is_valid (res, icon), NULL);
258
259   if (type)
260     *type = NULL;
261
262   return g_task_propagate_pointer (G_TASK (res), error);
263 }
264
265 static void
266 g_bytes_icon_loadable_icon_iface_init (GLoadableIconIface *iface)
267 {
268   iface->load = g_bytes_icon_load;
269   iface->load_async = g_bytes_icon_load_async;
270   iface->load_finish = g_bytes_icon_load_finish;
271 }