]> de.git.xonotic.org Git - xonotic/xonotic-data.pk3dir.git/blob - qcsrc/menu/item/image.c
Allow zooming even if the image would have me.forcedAspect set to 0
[xonotic/xonotic-data.pk3dir.git] / qcsrc / menu / item / image.c
1 #ifdef INTERFACE
2 CLASS(Image) EXTENDS(Item)
3         METHOD(Image, configureImage, void(entity, string))
4         METHOD(Image, draw, void(entity))
5         METHOD(Image, toString, string(entity))
6         METHOD(Image, resizeNotify, void(entity, vector, vector, vector, vector))
7         METHOD(Image, updateAspect, void(entity))
8         METHOD(Image, setZoom, void(entity, float, float))
9         METHOD(Image, drag_setStartPos, float(entity, vector))
10         METHOD(Image, drag, float(entity, vector))
11         ATTRIB(Image, src, string, string_null)
12         ATTRIB(Image, color, vector, '1 1 1')
13         ATTRIB(Image, forcedAspect, float, 0) // special values: -1 keep image aspect ratio, -2 keep image size but bound to the containing box, -3 always keep image size
14         ATTRIB(Image, initialForcedZoom, float, 0) // used by forcedAspect -2 when the image is larger than the containing box
15         ATTRIB(Image, zoomFactor, float, 1)
16         ATTRIB(Image, zoomOffset, vector, '0.5 0.5 0')
17         ATTRIB(Image, zoomTime, float, 0)
18         ATTRIB(Image, start_zoomOffset, vector, '0 0 0')
19         ATTRIB(Image, start_coords, vector, '0 0 0')
20         ATTRIB(Image, imgOrigin, vector, '0 0 0')
21         ATTRIB(Image, imgSize, vector, '0 0 0')
22 ENDCLASS(Image)
23 #endif
24
25 #ifdef IMPLEMENTATION
26 string Image_toString(entity me)
27 {
28         return me.src;
29 }
30 void Image_configureImage(entity me, string path)
31 {
32         me.src = path;
33         me.zoomOffset = '0.5 0.5 0';
34         me.zoomFactor = 1;
35         if (me.forcedAspect == -2)
36                 me.initialForcedZoom = -1; // calculate initialForcedZoom at the first updateAspect call
37 }
38 void Image_draw(entity me)
39 {
40         if(me.imgSize_x > 1 || me.imgSize_y > 1)
41                 draw_SetClip();
42         draw_Picture(me.imgOrigin, me.src, me.imgSize, me.color, 1);
43         if(me.imgSize_x > 1 || me.imgSize_y > 1)
44                 draw_ClearClip();
45 }
46 void Image_updateAspect(entity me)
47 {
48         float asp;
49         if(me.size_x <= 0 || me.size_y <= 0)
50                 return;
51         if(me.forcedAspect == 0)
52         {
53                 me.imgOrigin = '0 0 0';
54                 me.imgSize = '1 1 0';
55         }
56         else
57         {
58                 if(me.forcedAspect < 0)
59                 {
60                         vector sz;
61                         sz = draw_PictureSize(me.src);
62                         asp = sz_x / sz_y;
63                 }
64                 else
65                         asp = me.forcedAspect;
66
67                 if(me.forcedAspect <= -2)
68                 {
69                         me.imgSize_x = sz_x / me.size_x;
70                         me.imgSize_y = sz_y / me.size_y;
71                         if(me.initialForcedZoom < 0 && (me.imgSize_x > 1 || me.imgSize_y > 1))
72                         {
73                                 // image larger than the containing box, zoom it out to fit into the box
74                                 if(me.size_x > asp * me.size_y)
75                                         me.initialForcedZoom = (me.size_y * asp / me.size_x) / me.imgSize_x;
76                                 else
77                                         me.initialForcedZoom = (me.size_x / (asp * me.size_y)) / me.imgSize_y;
78                                 me.zoomFactor = me.initialForcedZoom;
79                         }
80                 }
81                 else
82                 {
83                         if(me.size_x > asp * me.size_y)
84                         {
85                                 // x too large, so center x-wise
86                                 me.imgSize = eY + eX * (me.size_y * asp / me.size_x);
87                         }
88                         else
89                         {
90                                 // y too large, so center y-wise
91                                 me.imgSize = eX + eY * (me.size_x / (asp * me.size_y));
92                         }
93                 }
94         }
95
96         if (me.zoomFactor)
97                 me.imgSize = me.imgSize * me.zoomFactor;
98
99         if(me.imgSize_x > 1 || me.imgSize_y > 1)
100         {
101                 me.zoomOffset_x = bound(0, me.zoomOffset_x, 1);
102                 me.zoomOffset_y = bound(0, me.zoomOffset_y, 1);
103         }
104         else
105                 me.zoomOffset = '0.5 0.5 0';
106
107         me.imgOrigin_x = 0.5 - me.zoomOffset_x * me.imgSize_x;
108         me.imgOrigin_y = 0.5 - me.zoomOffset_y * me.imgSize_y;
109 }
110 float Image_drag_setStartPos(entity me, vector coords)
111 {
112         //if(me.imgSize_x > 1 || me.imgSize_y > 1) // check disabled: mousewheel zoom may start from a non-zoomed-in image
113         {
114                 me.start_zoomOffset = me.zoomOffset;
115                 me.start_coords = coords;
116         }
117         return 1;
118 }
119 float Image_drag(entity me, vector coords)
120 {
121         if(me.imgSize_x > 1 || me.imgSize_y > 1)
122         {
123                 me.zoomOffset_x = me.start_zoomOffset_x + (me.start_coords_x - coords_x) / me.imgSize_x;
124                 me.zoomOffset_y = me.start_zoomOffset_y + (me.start_coords_y - coords_y) / me.imgSize_y;
125                 me.updateAspect(me);
126         }
127         return 1;
128 }
129 void Image_setZoom(entity me, float z, float atMousePosition)
130 {
131         float prev_zoomFactor;
132         prev_zoomFactor = me.zoomFactor;
133         if (z < 0) // multiply by the current zoomFactor
134         {
135                 me.zoomFactor *= -z;
136                 float one_in_the_middle, initialZoom_in_the_middle;
137                 one_in_the_middle = ((prev_zoomFactor - 1) * (me.zoomFactor - 1) < 0);
138                 initialZoom_in_the_middle = (me.initialForcedZoom > 0 && (prev_zoomFactor - me.initialForcedZoom) * (me.zoomFactor - me.initialForcedZoom) < 0);
139                 if (one_in_the_middle && initialZoom_in_the_middle)
140                 {
141                         // snap to real dimensions or to box
142                         if (prev_zoomFactor < me.zoomFactor)
143                                 me.zoomFactor = min(1, me.initialForcedZoom);
144                         else
145                                 me.zoomFactor = max(1, me.initialForcedZoom);
146                 }
147                 else if (one_in_the_middle)
148                         me.zoomFactor = 1; // snap to real dimensions
149                 else if (initialZoom_in_the_middle)
150                         me.zoomFactor = me.initialForcedZoom; // snap to box
151         }
152         else if (z == 0) // reset (no zoom)
153         {
154                 if (me.initialForcedZoom > 0)
155                         me.zoomFactor = me.initialForcedZoom;
156                 else
157                         me.zoomFactor = 1;
158         }
159         else // directly set
160                 me.zoomFactor = z;
161         me.zoomFactor = bound(1/16, me.zoomFactor, 16);
162         if (prev_zoomFactor != me.zoomFactor)
163         {
164                 me.zoomTime = time;
165                 if (atMousePosition)
166                 {
167                         me.zoomOffset_x = me.start_zoomOffset_x + (me.start_coords_x - 0.5) / me.imgSize_x;
168                         me.zoomOffset_y = me.start_zoomOffset_y + (me.start_coords_y - 0.5) / me.imgSize_y;
169                         // updateAspect will reset however zoomOffset to '0.5 0.5 0' if with
170                         // this zoomFactor the image will not be zoomed (updateAspect will check
171                         // the new values of imgSize).
172                 }
173         }
174         me.updateAspect(me);
175 }
176 void Image_resizeNotify(entity me, vector relOrigin, vector relSize, vector absOrigin, vector absSize)
177 {
178         SUPER(Image).resizeNotify(me, relOrigin, relSize, absOrigin, absSize);
179         me.updateAspect(me);
180 }
181 #endif