alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 1 | package com.savoirfairelinux.sflphone.model; |
| 2 | |
| 3 | import android.content.Context; |
| 4 | import android.graphics.Bitmap; |
| 5 | import android.graphics.BitmapFactory; |
| 6 | import android.graphics.Canvas; |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 7 | import android.graphics.Color; |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 8 | import android.graphics.Paint; |
Adrien Béraud | 25fc409 | 2013-05-06 15:28:39 +1000 | [diff] [blame] | 9 | import android.graphics.PointF; |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 10 | import android.graphics.PorterDuff.Mode; |
| 11 | import android.graphics.PorterDuffXfermode; |
| 12 | import android.graphics.RectF; |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 13 | import android.util.Log; |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 14 | |
| 15 | import com.savoirfairelinux.sflphone.R; |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 16 | import com.savoirfairelinux.sflphone.adapters.ContactPictureLoader; |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 17 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 18 | public class Bubble { |
alision | 806e18e | 2013-06-21 15:30:17 -0400 | [diff] [blame] | 19 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 20 | // A Bitmap object that is going to be passed to the BitmapShader |
| 21 | private Bitmap internalBMP, externalBMP; |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 22 | |
alision | 806e18e | 2013-06-21 15:30:17 -0400 | [diff] [blame] | 23 | public SipCall associated_call; |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 24 | private PointF pos = new PointF(); |
| 25 | private RectF bounds; |
| 26 | public float target_scale = 1.f; |
| 27 | private float radius; |
| 28 | private float scale = 1.f; |
| 29 | private float density = 1.f; |
| 30 | public PointF speed = new PointF(0, 0); |
| 31 | public PointF last_speed = new PointF(); |
| 32 | public PointF attractor = null; |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 33 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 34 | public boolean dragged = false; |
| 35 | public long last_drag; |
| 36 | public boolean expanded; // determine if we draw the buttons around the bubble |
| 37 | private Bitmap saved_photo; |
alision | 729b0a2 | 2013-07-02 11:57:33 -0400 | [diff] [blame^] | 38 | private float expanded_radius; |
alision | 4a0eb09 | 2013-05-07 13:52:03 -0400 | [diff] [blame] | 39 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 40 | public void setAttractor(PointF attractor) { |
| 41 | this.attractor = attractor; |
| 42 | } |
Adrien Béraud | e0ef0c2 | 2013-05-18 01:56:27 +1000 | [diff] [blame] | 43 | |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 44 | public Bubble(Context context, SipCall call, float x, float y, float size) { |
| 45 | |
| 46 | Bitmap photo = null; |
| 47 | if (call.getContact().getPhoto_id() > 0) { |
| 48 | photo = ContactPictureLoader.loadContactPhoto(context.getContentResolver(), call.getContact().getId()); |
| 49 | } else { |
| 50 | photo = BitmapFactory.decodeResource(context.getResources(), R.drawable.ic_contact_picture); |
| 51 | } |
| 52 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 53 | saved_photo = photo; |
| 54 | internalBMP = Bitmap.createScaledBitmap(photo, (int) size, (int) size, false); |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 55 | |
alision | 806e18e | 2013-06-21 15:30:17 -0400 | [diff] [blame] | 56 | associated_call = call; |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 57 | pos.set(x, y); |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 58 | radius = internalBMP.getWidth() / 2; |
alision | 729b0a2 | 2013-07-02 11:57:33 -0400 | [diff] [blame^] | 59 | expanded_radius = (float) (size * 1.5); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 60 | bounds = new RectF(pos.x - radius, pos.y - radius, pos.x + radius, pos.y + radius); |
| 61 | attractor = new PointF(x, y); |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 62 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 63 | Paint circlePaint = new Paint(Paint.ANTI_ALIAS_FLAG); |
| 64 | circlePaint.setStyle(Paint.Style.FILL); |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 65 | Bitmap circle = Bitmap.createBitmap(internalBMP.getWidth(), internalBMP.getHeight(), Bitmap.Config.ARGB_8888); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 66 | Canvas circle_drawer = new Canvas(circle); |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 67 | circle_drawer.drawOval(new RectF(0, 0, internalBMP.getWidth(), internalBMP.getHeight()), circlePaint); |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 68 | |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 69 | externalBMP = Bitmap.createBitmap(internalBMP.getWidth(), internalBMP.getHeight(), Bitmap.Config.ARGB_8888); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 70 | Canvas canvas = new Canvas(externalBMP); |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 71 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 72 | circlePaint.setFilterBitmap(false); |
| 73 | canvas.drawBitmap(internalBMP, 0, 0, circlePaint); |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 74 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 75 | circlePaint.setXfermode(new PorterDuffXfermode(Mode.DST_IN)); |
| 76 | canvas.drawBitmap(circle, 0, 0, circlePaint); |
| 77 | } |
Adrien Béraud | 25fc409 | 2013-05-06 15:28:39 +1000 | [diff] [blame] | 78 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 79 | public Bitmap getBitmap() { |
| 80 | return externalBMP; |
| 81 | } |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 82 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 83 | public RectF getBounds() { |
| 84 | return bounds; |
| 85 | } |
Adrien Béraud | 3326888 | 2013-05-18 03:41:15 +1000 | [diff] [blame] | 86 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 87 | public void set(float x, float y, float s) { |
| 88 | scale = s; |
| 89 | pos.x = x; |
| 90 | pos.y = y; |
| 91 | float rad = scale * getRadius() * density; |
| 92 | bounds.set(pos.x - rad, pos.y - rad, pos.x + rad, pos.y + rad); |
| 93 | } |
Adrien Béraud | 25fc409 | 2013-05-06 15:28:39 +1000 | [diff] [blame] | 94 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 95 | public float getPosX() { |
| 96 | return pos.x; |
| 97 | } |
Adrien Béraud | 25fc409 | 2013-05-06 15:28:39 +1000 | [diff] [blame] | 98 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 99 | public float getPosY() { |
| 100 | return pos.y; |
| 101 | } |
Adrien Béraud | 6bbce91 | 2013-05-24 00:48:13 +1000 | [diff] [blame] | 102 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 103 | public void setPos(float x, float y) { |
| 104 | set(x, y, scale); |
| 105 | } |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 106 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 107 | public PointF getPos() { |
| 108 | return pos; |
| 109 | } |
Adrien Béraud | 25fc409 | 2013-05-06 15:28:39 +1000 | [diff] [blame] | 110 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 111 | public float getScale() { |
| 112 | return scale; |
| 113 | } |
Adrien Béraud | 25fc409 | 2013-05-06 15:28:39 +1000 | [diff] [blame] | 114 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 115 | public void setScale(float s) { |
| 116 | set(pos.x, pos.y, s); |
| 117 | } |
Adrien Béraud | 25fc409 | 2013-05-06 15:28:39 +1000 | [diff] [blame] | 118 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 119 | public float getRadius() { |
| 120 | return expanded ? expanded_radius : radius; |
| 121 | } |
Adrien Béraud | 6bbce91 | 2013-05-24 00:48:13 +1000 | [diff] [blame] | 122 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 123 | /** |
| 124 | * Point intersection test. |
| 125 | */ |
| 126 | boolean intersects(float x, float y) { |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 127 | float dx = x - pos.x; |
| 128 | float dy = y - pos.y; |
alision | 50fa072 | 2013-06-25 17:29:44 -0400 | [diff] [blame] | 129 | |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 130 | return dx * dx + dy * dy < getRadius() * density * getRadius() * density; |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 131 | } |
Adrien Béraud | 6bbce91 | 2013-05-24 00:48:13 +1000 | [diff] [blame] | 132 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 133 | /** |
| 134 | * Other circle intersection test. |
| 135 | */ |
| 136 | boolean intersects(float x, float y, float radius) { |
| 137 | float dx = x - pos.x, dy = y - pos.y; |
| 138 | float tot_radius = getRadius() + radius; |
| 139 | return dx * dx + dy * dy < tot_radius * tot_radius; |
| 140 | } |
Adrien Béraud | 6bbce91 | 2013-05-24 00:48:13 +1000 | [diff] [blame] | 141 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 142 | public void setDensity(float density) { |
| 143 | this.density = density; |
| 144 | } |
Adrien Béraud | 25fc409 | 2013-05-06 15:28:39 +1000 | [diff] [blame] | 145 | |
alision | 729b0a2 | 2013-07-02 11:57:33 -0400 | [diff] [blame^] | 146 | public void expand() { |
Adrien Béraud | 0c9bd8f | 2013-05-30 16:16:57 +1000 | [diff] [blame] | 147 | |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 148 | expanded = true; |
| 149 | internalBMP = Bitmap.createScaledBitmap(saved_photo, (int) (2 * radius), (int) (2 * radius), false); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 150 | |
| 151 | bounds = new RectF(pos.x - getRadius(), pos.y - getRadius(), pos.x + getRadius(), pos.y + getRadius()); |
| 152 | |
| 153 | Paint circlePaint = new Paint(Paint.ANTI_ALIAS_FLAG); |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 154 | circlePaint.setAntiAlias(true); |
| 155 | circlePaint.setDither(true); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 156 | circlePaint.setStyle(Paint.Style.FILL); |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 157 | circlePaint.setColor(Color.RED); |
| 158 | |
| 159 | Bitmap circle = Bitmap.createBitmap(internalBMP.getWidth(), internalBMP.getHeight(), Bitmap.Config.ARGB_8888); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 160 | Canvas circle_drawer = new Canvas(circle); |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 161 | circle_drawer.drawOval(new RectF(0, 0, internalBMP.getWidth(), internalBMP.getHeight()), circlePaint); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 162 | |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 163 | Canvas canvas = new Canvas(internalBMP); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 164 | circlePaint.setFilterBitmap(false); |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 165 | canvas.drawBitmap(internalBMP, 0, 0, circlePaint); |
| 166 | |
| 167 | circlePaint.setXfermode(new PorterDuffXfermode(Mode.DST_IN)); |
| 168 | canvas.drawBitmap(circle, 0, 0, circlePaint); |
| 169 | circle_drawer.drawOval(new RectF(0, 0, internalBMP.getWidth(), internalBMP.getHeight()), circlePaint); |
| 170 | |
| 171 | externalBMP = Bitmap.createBitmap((int) (getRadius() * 2), (int) (getRadius() * 2), Bitmap.Config.ARGB_8888); |
| 172 | Canvas canvasf = new Canvas(externalBMP); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 173 | |
| 174 | Paint mPaintPath = new Paint(Paint.ANTI_ALIAS_FLAG); |
| 175 | mPaintPath.setStyle(Paint.Style.FILL); |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 176 | mPaintPath.setColor(0xAA000000); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 177 | |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 178 | Paint fatality = new Paint(Paint.ANTI_ALIAS_FLAG); |
| 179 | fatality.setAntiAlias(true); |
| 180 | fatality.setDither(true); |
| 181 | fatality.setStyle(Paint.Style.FILL); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 182 | |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 183 | canvasf.drawOval(new RectF(0, 0, getRadius() * 2, getRadius() * 2), mPaintPath); // background with buttons |
| 184 | canvasf.drawBitmap(internalBMP, (float) (getRadius() - radius), (float) (getRadius() - radius), fatality); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 185 | |
| 186 | } |
| 187 | |
| 188 | public void retract() { |
| 189 | expanded = false; |
| 190 | internalBMP = Bitmap.createScaledBitmap(saved_photo, (int) (2 * radius), (int) (2 * radius), false); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 191 | |
| 192 | bounds = new RectF(pos.x - getRadius(), pos.y - getRadius(), pos.x + getRadius(), pos.y + getRadius()); |
| 193 | |
| 194 | Paint circlePaint = new Paint(Paint.ANTI_ALIAS_FLAG); |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 195 | circlePaint.setAntiAlias(true); |
| 196 | circlePaint.setDither(true); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 197 | circlePaint.setStyle(Paint.Style.FILL); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 198 | |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 199 | Bitmap circle = Bitmap.createBitmap(internalBMP.getWidth(), internalBMP.getHeight(), Bitmap.Config.ARGB_8888); |
| 200 | Canvas circle_drawer = new Canvas(circle); |
| 201 | circle_drawer.drawOval(new RectF(0, 0, internalBMP.getWidth(), internalBMP.getHeight()), circlePaint); |
| 202 | |
| 203 | externalBMP = Bitmap.createBitmap(internalBMP.getWidth(), internalBMP.getHeight(), Bitmap.Config.ARGB_8888); |
alision | 2ec64f9 | 2013-06-17 17:28:58 -0400 | [diff] [blame] | 204 | Canvas canvas = new Canvas(externalBMP); |
| 205 | |
| 206 | circlePaint.setFilterBitmap(false); |
| 207 | canvas.drawBitmap(internalBMP, 0, 0, circlePaint); |
| 208 | |
| 209 | circlePaint.setXfermode(new PorterDuffXfermode(Mode.DST_IN)); |
| 210 | canvas.drawBitmap(circle, 0, 0, circlePaint); |
| 211 | |
| 212 | } |
alision | df1dac9 | 2013-06-27 17:35:53 -0400 | [diff] [blame] | 213 | |
| 214 | |
| 215 | /** |
| 216 | * Compare bubbles based on call ID |
| 217 | */ |
| 218 | @Override |
| 219 | public boolean equals(Object c) { |
| 220 | if (c instanceof Bubble && ((Bubble) c).associated_call.getCallId().contentEquals(associated_call.getCallId())) { |
| 221 | return true; |
| 222 | } |
| 223 | return false; |
| 224 | |
| 225 | } |
alision | 34673e6 | 2013-06-25 14:40:07 -0400 | [diff] [blame] | 226 | |
| 227 | /** |
| 228 | * When bubble is expanded we need to check on wich action button the user tap |
| 229 | * |
| 230 | * @param x |
| 231 | * @param y |
| 232 | * @return |
| 233 | */ |
| 234 | public int getAction(float x, float y) { |
| 235 | float relativeX = x - pos.x + externalBMP.getWidth() / 2; |
| 236 | float relativeY = y - pos.y + externalBMP.getHeight() / 2; |
| 237 | |
| 238 | Log.i("Bubble","relativeX:"+relativeX); |
| 239 | Log.i("Bubble","relativeY:"+relativeY); |
| 240 | |
| 241 | Log.i("Bubble","pos.x:"+pos.x); |
| 242 | Log.i("Bubble","pos.y:"+pos.y); |
| 243 | |
| 244 | Log.i("Bubble","externalBMP.getWidth():"+externalBMP.getWidth()); |
| 245 | Log.i("Bubble","externalBMP.getHeight():"+externalBMP.getHeight()); |
| 246 | |
| 247 | // Hold - Left |
| 248 | if (relativeX < externalBMP.getWidth() / 3 && relativeY > externalBMP.getHeight() / 3) { |
| 249 | return 1; |
| 250 | } |
| 251 | |
| 252 | // Record - Right |
| 253 | if (relativeX > externalBMP.getWidth() * 2 / 3 && relativeY > externalBMP.getHeight() / 3) { |
| 254 | return 2; |
| 255 | } |
| 256 | |
| 257 | // Transfer - Bottom |
| 258 | if (relativeY > externalBMP.getHeight() * 2 / 3) { |
| 259 | return 3; |
| 260 | } |
| 261 | return 0; |
| 262 | } |
alision | 729b0a2 | 2013-07-02 11:57:33 -0400 | [diff] [blame^] | 263 | |
| 264 | /** |
| 265 | * Always return the normal radius of the bubble |
| 266 | * @return |
| 267 | */ |
| 268 | public float getRetractedRadius() { |
| 269 | return radius; |
| 270 | } |
| 271 | |
| 272 | public int getHoldStatus() { |
| 273 | if(associated_call.isOnHold()) |
| 274 | return R.string.action_call_unhold; |
| 275 | else |
| 276 | return R.string.action_call_hold; |
| 277 | } |
| 278 | |
| 279 | public int getRecordStatus() { |
| 280 | if(associated_call.isRecording()) |
| 281 | return R.string.action_call_stop_record; |
| 282 | else |
| 283 | return R.string.action_call_record; |
| 284 | } |
alision | fe9cf71 | 2013-05-03 17:26:08 -0400 | [diff] [blame] | 285 | } |