1 | #ifndef RXVT_H_ /* include once only */ |
1 | #ifndef RXVT_H_ /* include once only */ |
2 | #define RXVT_H_ |
2 | #define RXVT_H_ |
3 | |
3 | |
4 | #include <cstdio> |
4 | #include <stdio.h> |
5 | #include <cctype> |
5 | #include <ctype.h> |
6 | #include <cerrno> |
6 | #include <errno.h> |
7 | #include <cstdarg> |
7 | #include <stdarg.h> |
8 | #include <cstdlib> |
8 | #include <stdlib.h> |
9 | #ifdef HAVE_STDINT_H |
9 | #ifdef HAVE_STDINT_H |
10 | #include <stdint.h> |
10 | #include <stdint.h> |
11 | #endif |
11 | #endif |
12 | #include <sys/types.h> |
12 | #include <sys/types.h> |
13 | #include <unistd.h> |
13 | #include <unistd.h> |
14 | #include <cstring> |
14 | #include <string.h> |
15 | #include <assert.h> |
15 | #include <assert.h> |
16 | #ifdef HAVE_SYS_IOCTL_H |
16 | #ifdef HAVE_SYS_IOCTL_H |
17 | #include <sys/ioctl.h> |
17 | #include <sys/ioctl.h> |
18 | #endif |
18 | #endif |
19 | #ifdef HAVE_SYS_STRREDIR_H |
19 | #ifdef HAVE_SYS_STRREDIR_H |
20 | #include <sys/strredir.h> |
20 | #include <sys/strredir.h> |
21 | #endif |
21 | #endif |
22 | |
22 | |
23 | #if HAVE_CWCHAR |
|
|
24 | # include <cwchar> |
|
|
25 | #elif HAVE_WCHAR_H |
23 | #if HAVE_WCHAR_H |
26 | # include <wchar.h> |
24 | # include <wchar.h> |
27 | #else |
25 | #else |
28 | // stdlib.h might provide it |
26 | // stdlib.h might provide it |
29 | #endif |
27 | #endif |
30 | |
|
|
31 | using namespace std; |
|
|
32 | |
28 | |
33 | // we assume that Xlib.h defines XPointer, and it does since at least 1994... |
29 | // we assume that Xlib.h defines XPointer, and it does since at least 1994... |
34 | |
30 | |
35 | extern "C" { |
31 | extern "C" { |
36 | #include <X11/Xlib.h> |
32 | #include <X11/Xlib.h> |
… | |
… | |
74 | #include <X11/cursorfont.h> |
70 | #include <X11/cursorfont.h> |
75 | #include <X11/keysym.h> |
71 | #include <X11/keysym.h> |
76 | #include <X11/keysymdef.h> |
72 | #include <X11/keysymdef.h> |
77 | #include <X11/Xatom.h> |
73 | #include <X11/Xatom.h> |
78 | |
74 | |
79 | #ifdef HAVE_AFTERIMAGE |
|
|
80 | # include <afterimage.h> |
|
|
81 | # undef min |
|
|
82 | # undef max |
|
|
83 | #endif |
|
|
84 | |
|
|
85 | #ifdef HAVE_PIXBUF |
75 | #ifdef HAVE_PIXBUF |
86 | # include <gdk-pixbuf/gdk-pixbuf.h> |
76 | # include <gdk-pixbuf/gdk-pixbuf.h> |
87 | #endif |
77 | #endif |
88 | |
78 | |
89 | #if defined(BG_IMAGE_FROM_FILE) || defined(ENABLE_TRANSPARENCY) |
79 | #if defined(BG_IMAGE_FROM_FILE) || defined(ENABLE_TRANSPARENCY) |
90 | # define HAVE_BG_PIXMAP 1 |
80 | # define HAVE_BG_PIXMAP 1 |
91 | #endif |
81 | #endif |
92 | |
82 | |
|
|
83 | #include <ecb.h> |
93 | #include "encoding.h" |
84 | #include "encoding.h" |
94 | #include "rxvtutil.h" |
85 | #include "rxvtutil.h" |
95 | #include "rxvtfont.h" |
86 | #include "rxvtfont.h" |
96 | #include "rxvttoolkit.h" |
87 | #include "rxvttoolkit.h" |
97 | #include "scrollbar.h" |
88 | #include "scrollbar.h" |
… | |
… | |
217 | ~localise_env () |
208 | ~localise_env () |
218 | { |
209 | { |
219 | environ = orig_env; |
210 | environ = orig_env; |
220 | } |
211 | } |
221 | }; |
212 | }; |
|
|
213 | |
|
|
214 | #ifdef HAVE_BG_PIXMAP |
|
|
215 | # ifdef BG_IMAGE_FROM_FILE |
|
|
216 | enum { |
|
|
217 | IM_IS_SET = 1 << 0, |
|
|
218 | IM_IS_SIZE_SENSITIVE = 1 << 1, |
|
|
219 | IM_KEEP_ASPECT = 1 << 2, |
|
|
220 | IM_ROOT_ALIGN = 1 << 3, |
|
|
221 | IM_TILE = 1 << 4, |
|
|
222 | IM_GEOMETRY_FLAGS = IM_KEEP_ASPECT | IM_ROOT_ALIGN | IM_TILE, |
|
|
223 | }; |
|
|
224 | |
|
|
225 | enum { |
|
|
226 | noScale = 0, |
|
|
227 | windowScale = 100, |
|
|
228 | defaultScale = windowScale, |
|
|
229 | centerAlign = 50, |
|
|
230 | defaultAlign = centerAlign, |
|
|
231 | }; |
|
|
232 | |
|
|
233 | struct rxvt_image |
|
|
234 | { |
|
|
235 | uint8_t flags; |
|
|
236 | unsigned int h_scale, v_scale; /* percents of the window size */ |
|
|
237 | int h_align, v_align; /* percents of the window size: |
|
|
238 | 0 - left align, 50 - center, 100 - right */ |
|
|
239 | |
|
|
240 | # ifdef HAVE_PIXBUF |
|
|
241 | GdkPixbuf *pixbuf; |
|
|
242 | |
|
|
243 | ~rxvt_image () |
|
|
244 | { |
|
|
245 | if (pixbuf) |
|
|
246 | g_object_unref (pixbuf); |
|
|
247 | } |
|
|
248 | |
|
|
249 | int width () |
|
|
250 | { |
|
|
251 | return gdk_pixbuf_get_width (pixbuf); |
|
|
252 | } |
|
|
253 | int height () |
|
|
254 | { |
|
|
255 | return gdk_pixbuf_get_height (pixbuf); |
|
|
256 | } |
|
|
257 | # endif |
|
|
258 | |
|
|
259 | bool set_file (const char *file); |
|
|
260 | bool set_geometry (const char *geom, bool update = false); |
|
|
261 | }; |
|
|
262 | # endif |
|
|
263 | #endif |
222 | |
264 | |
223 | /* |
265 | /* |
224 | ***************************************************************************** |
266 | ***************************************************************************** |
225 | * STRUCTURES AND TYPEDEFS |
267 | * STRUCTURES AND TYPEDEFS |
226 | ***************************************************************************** |
268 | ***************************************************************************** |
… | |
… | |
779 | // that are not representable in unicode, as well as characters |
821 | // that are not representable in unicode, as well as characters |
780 | // not fitting in the BMP. |
822 | // not fitting in the BMP. |
781 | struct compose_char |
823 | struct compose_char |
782 | { |
824 | { |
783 | unicode_t c1, c2; // any chars != NOCHAR are valid |
825 | unicode_t c1, c2; // any chars != NOCHAR are valid |
|
|
826 | #if __cplusplus >= 201103L || ECB_GCC_VERSION(4,4) |
|
|
827 | compose_char () = default; |
|
|
828 | #endif |
784 | compose_char (unicode_t c1, unicode_t c2) |
829 | compose_char (unicode_t c1, unicode_t c2) |
785 | : c1(c1), c2(c2) |
830 | : c1(c1), c2(c2) |
786 | { } |
831 | { } |
787 | }; |
832 | }; |
788 | |
833 | |
… | |
… | |
870 | * END······················= total_rows |
915 | * END······················= total_rows |
871 | */ |
916 | */ |
872 | |
917 | |
873 | struct TermWin_t |
918 | struct TermWin_t |
874 | { |
919 | { |
|
|
920 | int vt_width; /* actual window width [pixels] */ |
|
|
921 | int vt_height; /* actual window height [pixels] */ |
875 | int width; /* window width [pixels] */ |
922 | int width; /* window width [pixels] */ |
876 | int height; /* window height [pixels] */ |
923 | int height; /* window height [pixels] */ |
877 | int fwidth; /* font width [pixels] */ |
924 | int fwidth; /* font width [pixels] */ |
878 | int fheight; /* font height [pixels] */ |
925 | int fheight; /* font height [pixels] */ |
879 | int fbase; /* font ascent (baseline) [pixels] */ |
926 | int fbase; /* font ascent (baseline) [pixels] */ |
… | |
… | |
891 | int view_start; /* scrollback view starts here */ |
938 | int view_start; /* scrollback view starts here */ |
892 | int top_row; /* topmost row index of scrollback */ |
939 | int top_row; /* topmost row index of scrollback */ |
893 | Window parent; /* parent identifier */ |
940 | Window parent; /* parent identifier */ |
894 | Window vt; /* vt100 window */ |
941 | Window vt; /* vt100 window */ |
895 | GC gc; /* GC for drawing */ |
942 | GC gc; /* GC for drawing */ |
896 | Pixmap pixmap; |
|
|
897 | rxvt_drawable *drawable; |
943 | rxvt_drawable *drawable; |
898 | rxvt_fontset *fontset[4]; |
944 | rxvt_fontset *fontset[4]; |
899 | }; |
945 | }; |
900 | |
946 | |
901 | /* |
947 | /* |
… | |
… | |
1028 | hidden_pointer:1, |
1074 | hidden_pointer:1, |
1029 | #endif |
1075 | #endif |
1030 | enc_utf8:1, /* whether locale uses utf-8 */ |
1076 | enc_utf8:1, /* whether locale uses utf-8 */ |
1031 | seen_input:1, /* whether we have seen some program output yet */ |
1077 | seen_input:1, /* whether we have seen some program output yet */ |
1032 | seen_resize:1, /* whether we had a resize event */ |
1078 | seen_resize:1, /* whether we had a resize event */ |
|
|
1079 | init_done:1, |
1033 | parsed_geometry:1; |
1080 | parsed_geometry:1; |
1034 | |
1081 | |
1035 | unsigned char refresh_type, |
1082 | unsigned char refresh_type, |
1036 | #ifdef META8_OPTION |
1083 | #ifdef META8_OPTION |
1037 | meta_char; /* Alt-key prefix */ |
1084 | meta_char; /* Alt-key prefix */ |
… | |
… | |
1071 | allowedxerror; |
1118 | allowedxerror; |
1072 | /* ---------- */ |
1119 | /* ---------- */ |
1073 | unsigned int ModLevel3Mask, |
1120 | unsigned int ModLevel3Mask, |
1074 | ModMetaMask, |
1121 | ModMetaMask, |
1075 | ModNumLockMask; |
1122 | ModNumLockMask; |
1076 | int old_width, /* last used width in screen resize */ |
|
|
1077 | old_height; /* last used height in screen resize */ |
|
|
1078 | unsigned long priv_modes, |
1123 | unsigned long priv_modes, |
1079 | SavedModes; |
1124 | SavedModes; |
1080 | /* ---------- */ |
1125 | /* ---------- */ |
1081 | Atom *xa; |
1126 | Atom *xa; |
1082 | /* ---------- */ |
1127 | /* ---------- */ |
… | |
… | |
1092 | #ifdef HAVE_BG_PIXMAP |
1137 | #ifdef HAVE_BG_PIXMAP |
1093 | void bg_init (); |
1138 | void bg_init (); |
1094 | void bg_destroy (); |
1139 | void bg_destroy (); |
1095 | |
1140 | |
1096 | enum { |
1141 | enum { |
1097 | //subset returned by make_transparency_pixmap |
|
|
1098 | BG_IS_VALID = 1 << 0, |
1142 | BG_IS_VALID = 1 << 0, |
1099 | BG_NEEDS_TINT = 1 << 1, |
|
|
1100 | BG_NEEDS_BLUR = 1 << 2, |
|
|
1101 | |
1143 | |
1102 | BG_EFFECTS_FLAGS = BG_NEEDS_TINT | BG_NEEDS_BLUR, |
|
|
1103 | |
|
|
1104 | BG_KEEP_ASPECT = 1 << 3, |
|
|
1105 | BG_ROOT_ALIGN = 1 << 4, |
|
|
1106 | BG_TILE = 1 << 14, |
|
|
1107 | BG_GEOMETRY_FLAGS = BG_KEEP_ASPECT | BG_ROOT_ALIGN | BG_TILE, |
|
|
1108 | |
|
|
1109 | BG_TINT_SET = 1 << 5, |
|
|
1110 | BG_TINT_BITAND = 1 << 6, |
|
|
1111 | BG_TINT_FLAGS = BG_NEEDS_TINT | BG_TINT_BITAND, |
|
|
1112 | |
|
|
1113 | BG_HAS_RENDER = 1 << 7, |
|
|
1114 | BG_HAS_RENDER_CONV = 1 << 8, |
|
|
1115 | BG_CLIENT_RENDER = 1 << 9, |
|
|
1116 | |
|
|
1117 | BG_IS_TRANSPARENT = 1 << 10, |
1144 | BG_IS_TRANSPARENT = 1 << 1, |
1118 | BG_NEEDS_REFRESH = 1 << 11, |
1145 | BG_NEEDS_REFRESH = 1 << 2, |
1119 | BG_IS_SIZE_SENSITIVE = 1 << 12, |
1146 | |
1120 | BG_IS_FROM_FILE = 1 << 13, |
1147 | BG_HAS_RENDER = 1 << 3, |
|
|
1148 | BG_HAS_RENDER_CONV = 1 << 4, |
|
|
1149 | BG_CLIENT_RENDER = 1 << 5, |
|
|
1150 | |
|
|
1151 | BG_TINT_SET = 1 << 6, |
|
|
1152 | BG_TINT_BITAND = 1 << 7, |
1121 | }; |
1153 | }; |
1122 | |
1154 | |
1123 | unsigned int bg_flags; |
1155 | uint8_t bg_flags; |
1124 | |
1156 | |
1125 | # ifdef BG_IMAGE_FROM_FILE |
1157 | # ifdef BG_IMAGE_FROM_FILE |
|
|
1158 | rxvt_image bg_image; |
1126 | void get_image_geometry (int image_width, int image_height, int &w, int &h, int &x, int &y); |
1159 | void get_image_geometry (rxvt_image &image, int &w, int &h, int &x, int &y); |
1127 | bool render_image (unsigned long tr_flags); |
1160 | bool render_image (rxvt_image &image); |
1128 | |
|
|
1129 | enum { |
|
|
1130 | noScale = 0, |
|
|
1131 | windowScale = 100, |
|
|
1132 | defaultScale = windowScale, |
|
|
1133 | centerAlign = 50, |
|
|
1134 | defaultAlign = centerAlign, |
|
|
1135 | }; |
|
|
1136 | |
|
|
1137 | unsigned int h_scale, v_scale; /* percents of the window size */ |
|
|
1138 | int h_align, v_align; /* percents of the window size: |
|
|
1139 | 0 - left align, 50 - center, 100 - right */ |
|
|
1140 | |
|
|
1141 | bool bg_set_geometry (const char *geom, bool update = false); |
|
|
1142 | void bg_set_default_geometry () |
|
|
1143 | { |
|
|
1144 | h_scale = v_scale = defaultScale; |
|
|
1145 | h_align = v_align = defaultAlign; |
|
|
1146 | } |
|
|
1147 | |
|
|
1148 | bool bg_set_file (const char *file); |
|
|
1149 | # endif |
1161 | # endif |
1150 | |
1162 | |
1151 | # ifdef ENABLE_TRANSPARENCY |
1163 | # ifdef ENABLE_TRANSPARENCY |
1152 | Pixmap root_pixmap; /* current root pixmap set */ |
1164 | Pixmap root_pixmap; /* current root pixmap set */ |
1153 | rxvt_color tint; |
1165 | rxvt_color tint; |
1154 | int shade; |
1166 | int shade; |
1155 | int h_blurRadius, v_blurRadius; |
1167 | int h_blurRadius, v_blurRadius; |
1156 | |
1168 | |
1157 | bool bg_set_transparent (); |
1169 | void bg_set_transparent () |
|
|
1170 | { |
|
|
1171 | bg_flags |= BG_IS_TRANSPARENT; |
|
|
1172 | } |
1158 | void bg_set_root_pixmap (); |
1173 | void bg_set_root_pixmap (); |
1159 | void set_tint_shade_flags (); |
|
|
1160 | bool bg_set_tint (rxvt_color &new_tint); |
1174 | bool bg_set_tint (rxvt_color &new_tint); |
1161 | bool bg_set_shade (const char *shade_str); |
1175 | bool bg_set_shade (const char *shade_str); |
1162 | bool bg_set_blur (const char *geom); |
1176 | bool bg_set_blur (const char *geom); |
1163 | |
1177 | |
1164 | bool blur_pixmap (Pixmap pixmap, Visual *visual, int width, int height); |
1178 | bool blur_pixmap (Pixmap pixmap, int width, int height); |
1165 | bool tint_pixmap (Pixmap pixmap, Visual *visual, int width, int height); |
1179 | bool tint_pixmap (Pixmap pixmap, int width, int height); |
1166 | void tint_ximage (Visual *visual, XImage *ximage); |
1180 | void tint_ximage (XImage *ximage); |
1167 | unsigned long make_transparency_pixmap (); |
1181 | bool make_transparency_pixmap (); |
1168 | # endif |
1182 | # endif |
1169 | |
1183 | |
1170 | ev_tstamp bg_valid_since; |
1184 | ev_tstamp bg_valid_since; |
1171 | |
1185 | |
1172 | Pixmap bg_pixmap; |
1186 | Pixmap bg_pixmap; |
… | |
… | |
1182 | void bg_invalidate () |
1196 | void bg_invalidate () |
1183 | { |
1197 | { |
1184 | bg_flags &= ~BG_IS_VALID; |
1198 | bg_flags &= ~BG_IS_VALID; |
1185 | } |
1199 | } |
1186 | #endif |
1200 | #endif |
1187 | #ifdef HAVE_AFTERIMAGE |
|
|
1188 | ASImage *original_asim; |
|
|
1189 | ASVisual *asv; |
|
|
1190 | ASImageManager *asimman; |
|
|
1191 | |
|
|
1192 | void init_asv () |
|
|
1193 | { |
|
|
1194 | if (!asv) |
|
|
1195 | asv = create_asvisual_for_id (dpy, display->screen, depth, XVisualIDFromVisual (visual), cmap, NULL); |
|
|
1196 | } |
|
|
1197 | #endif |
|
|
1198 | #ifdef HAVE_PIXBUF |
1201 | #ifdef HAVE_PIXBUF |
1199 | GdkPixbuf *pixbuf; |
|
|
1200 | bool pixbuf_to_pixmap (GdkPixbuf *pixbuf, Pixmap pixmap, GC gc, |
1202 | bool pixbuf_to_pixmap (GdkPixbuf *pixbuf, Pixmap pixmap, GC gc, |
1201 | int src_x, int src_y, int dst_x, int dst_y, |
1203 | int src_x, int src_y, int dst_x, int dst_y, |
1202 | unsigned int width, unsigned int height); |
1204 | unsigned int width, unsigned int height, bool argb); |
1203 | #endif |
1205 | #endif |
1204 | |
1206 | |
1205 | #if ENABLE_OVERLAY |
1207 | #if ENABLE_OVERLAY |
1206 | overlay_base ov; |
1208 | overlay_base ov; |
1207 | |
1209 | |
… | |
… | |
1214 | void scr_overlay_set (int x, int y, const char *s) NOTHROW; |
1216 | void scr_overlay_set (int x, int y, const char *s) NOTHROW; |
1215 | void scr_overlay_set (int x, int y, const wchar_t *s) NOTHROW; |
1217 | void scr_overlay_set (int x, int y, const wchar_t *s) NOTHROW; |
1216 | #endif |
1218 | #endif |
1217 | |
1219 | |
1218 | vector<void *> allocated; // free these memory blocks with free() |
1220 | vector<void *> allocated; // free these memory blocks with free() |
1219 | |
|
|
1220 | char env_windowid[21]; /* environmental variable WINDOWID */ |
|
|
1221 | char env_colorfgbg[sizeof ("COLORFGBG=default;default;bg") + 1]; |
|
|
1222 | char *env_display; /* environmental variable DISPLAY */ |
|
|
1223 | char *env_term; /* environmental variable TERM */ |
|
|
1224 | |
1221 | |
1225 | char *locale; |
1222 | char *locale; |
1226 | char charsets[4]; |
1223 | char charsets[4]; |
1227 | char *v_buffer; /* pointer to physical buffer */ |
1224 | char *v_buffer; /* pointer to physical buffer */ |
1228 | unsigned int v_buflen; /* size of area to write */ |
1225 | unsigned int v_buflen; /* size of area to write */ |
… | |
… | |
1288 | void cmdbuf_append (const char *str, size_t count); |
1285 | void cmdbuf_append (const char *str, size_t count); |
1289 | bool pty_fill (); |
1286 | bool pty_fill (); |
1290 | void pty_cb (ev::io &w, int revents); ev::io pty_ev; |
1287 | void pty_cb (ev::io &w, int revents); ev::io pty_ev; |
1291 | |
1288 | |
1292 | #ifdef CURSOR_BLINK |
1289 | #ifdef CURSOR_BLINK |
|
|
1290 | void cursor_blink_reset (); |
1293 | void cursor_blink_cb (ev::timer &w, int revents); ev::timer cursor_blink_ev; |
1291 | void cursor_blink_cb (ev::timer &w, int revents); ev::timer cursor_blink_ev; |
1294 | #endif |
1292 | #endif |
1295 | #ifdef TEXT_BLINK |
1293 | #ifdef TEXT_BLINK |
1296 | void text_blink_cb (ev::timer &w, int revents); ev::timer text_blink_ev; |
1294 | void text_blink_cb (ev::timer &w, int revents); ev::timer text_blink_ev; |
1297 | #endif |
1295 | #endif |
… | |
… | |
1415 | void set_mbstring_property (Atom prop, const char *str, int len = -1); |
1413 | void set_mbstring_property (Atom prop, const char *str, int len = -1); |
1416 | void set_utf8_property (Atom prop, const char *str, int len = -1); |
1414 | void set_utf8_property (Atom prop, const char *str, int len = -1); |
1417 | void set_title (const char *str); |
1415 | void set_title (const char *str); |
1418 | void set_icon_name (const char *str); |
1416 | void set_icon_name (const char *str); |
1419 | void set_window_color (int idx, const char *color); |
1417 | void set_window_color (int idx, const char *color); |
1420 | void set_colorfgbg (); |
1418 | char *get_colorfgbg (); |
1421 | bool set_color (rxvt_color &color, const char *name); |
1419 | bool set_color (rxvt_color &color, const char *name); |
1422 | void alias_color (int dst, int src); |
1420 | void alias_color (int dst, int src); |
1423 | void set_widthheight (unsigned int newwidth, unsigned int newheight); |
1421 | void set_widthheight (unsigned int newwidth, unsigned int newheight); |
1424 | void get_window_origin (int &x, int &y); |
1422 | void get_window_origin (int &x, int &y); |
1425 | Pixmap get_pixmap_property (Atom property); |
1423 | Pixmap get_pixmap_property (Atom property); |
… | |
… | |
1571 | const char **get_options (int argc, const char *const *argv); |
1569 | const char **get_options (int argc, const char *const *argv); |
1572 | int parse_keysym (const char *str, const char *arg); |
1570 | int parse_keysym (const char *str, const char *arg); |
1573 | const char *x_resource (const char *name); |
1571 | const char *x_resource (const char *name); |
1574 | void extract_resources (); |
1572 | void extract_resources (); |
1575 | void extract_keysym_resources (); |
1573 | void extract_keysym_resources (); |
|
|
1574 | void find_resources (const char *n_prefix, const char *c_prefix, int mode, |
|
|
1575 | Bool (*proc)(XrmDatabase *, XrmBindingList, XrmQuarkList, XrmRepresentation *, XrmValue *, XPointer)); |
|
|
1576 | bool parse_bool_resource (const char *str) |
|
|
1577 | { |
|
|
1578 | return (!strcasecmp (str, "TRUE") |
|
|
1579 | || !strcasecmp (str, "YES") |
|
|
1580 | || !strcasecmp (str, "ON") |
|
|
1581 | || !strcasecmp (str, "1")); |
|
|
1582 | } |
1576 | }; |
1583 | }; |
1577 | |
1584 | |
1578 | #endif /* _RXVT_H_ */ |
1585 | #endif /* _RXVT_H_ */ |
1579 | |
1586 | |