1 /* 2 * Copyright (C) 2009 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package android.webkit; 18 19 import android.os.Handler; 20 import android.os.Message; 21 import android.util.Log; 22 import java.util.HashMap; 23 import java.util.HashSet; 24 import java.util.Map; 25 import java.util.Set; 26 27 28 /** 29 * Implements the Java side of GeolocationPermissions. Simply marshalls calls 30 * from the UI thread to the WebKit thread. 31 */ 32 public final class GeolocationPermissions { 33 /** 34 * Callback interface used by the browser to report a Geolocation permission 35 * state set by the user in response to a permissions prompt. 36 */ 37 public interface Callback { invoke(String origin, boolean allow, boolean remember)38 public void invoke(String origin, boolean allow, boolean remember); 39 }; 40 41 // Log tag 42 private static final String TAG = "geolocationPermissions"; 43 44 // Global instance 45 private static GeolocationPermissions sInstance; 46 47 private Handler mHandler; 48 private Handler mUIHandler; 49 50 // Members used to transfer the origins and permissions between threads. 51 private Set<String> mOrigins; 52 private boolean mAllowed; 53 private Set<String> mOriginsToClear; 54 private Set<String> mOriginsToAllow; 55 56 // Message ids 57 static final int GET_ORIGINS = 0; 58 static final int GET_ALLOWED = 1; 59 static final int CLEAR = 2; 60 static final int ALLOW = 3; 61 static final int CLEAR_ALL = 4; 62 63 // Message ids on the UI thread 64 static final int RETURN_ORIGINS = 0; 65 static final int RETURN_ALLOWED = 1; 66 67 private static final String ORIGINS = "origins"; 68 private static final String ORIGIN = "origin"; 69 private static final String CALLBACK = "callback"; 70 private static final String ALLOWED = "allowed"; 71 72 /** 73 * Gets the singleton instance of the class. 74 * @hide 75 */ getInstance()76 public static GeolocationPermissions getInstance() { 77 if (sInstance == null) { 78 sInstance = new GeolocationPermissions(); 79 } 80 return sInstance; 81 } 82 83 /** 84 * Creates the UI message handler. Must be called on the UI thread. 85 * @hide 86 */ createUIHandler()87 public void createUIHandler() { 88 if (mUIHandler == null) { 89 mUIHandler = new Handler() { 90 @Override 91 public void handleMessage(Message msg) { 92 // Runs on the UI thread. 93 switch (msg.what) { 94 case RETURN_ORIGINS: { 95 Map values = (Map) msg.obj; 96 Set origins = (Set) values.get(ORIGINS); 97 ValueCallback<Set> callback = (ValueCallback<Set>) values.get(CALLBACK); 98 callback.onReceiveValue(origins); 99 } break; 100 case RETURN_ALLOWED: { 101 Map values = (Map) msg.obj; 102 Boolean allowed = (Boolean) values.get(ALLOWED); 103 ValueCallback<Boolean> callback = (ValueCallback<Boolean>) values.get(CALLBACK); 104 callback.onReceiveValue(allowed); 105 } break; 106 } 107 } 108 }; 109 } 110 } 111 112 /** 113 * Creates the message handler. Must be called on the WebKit thread. 114 * @hide 115 */ createHandler()116 public void createHandler() { 117 if (mHandler == null) { 118 mHandler = new Handler() { 119 @Override 120 public void handleMessage(Message msg) { 121 // Runs on the WebKit thread. 122 switch (msg.what) { 123 case GET_ORIGINS: { 124 getOriginsImpl(); 125 ValueCallback callback = (ValueCallback) msg.obj; 126 Set origins = new HashSet(mOrigins); 127 Map values = new HashMap<String, Object>(); 128 values.put(CALLBACK, callback); 129 values.put(ORIGINS, origins); 130 postUIMessage(Message.obtain(null, RETURN_ORIGINS, values)); 131 } break; 132 case GET_ALLOWED: { 133 Map values = (Map) msg.obj; 134 String origin = (String) values.get(ORIGIN); 135 ValueCallback callback = (ValueCallback) values.get(CALLBACK); 136 getAllowedImpl(origin); 137 Map retValues = new HashMap<String, Object>(); 138 retValues.put(CALLBACK, callback); 139 retValues.put(ALLOWED, new Boolean(mAllowed)); 140 postUIMessage(Message.obtain(null, RETURN_ALLOWED, retValues)); 141 } break; 142 case CLEAR: 143 nativeClear((String) msg.obj); 144 break; 145 case ALLOW: 146 nativeAllow((String) msg.obj); 147 break; 148 case CLEAR_ALL: 149 nativeClearAll(); 150 break; 151 } 152 } 153 }; 154 155 if (mOriginsToClear != null) { 156 for (String origin : mOriginsToClear) { 157 nativeClear(origin); 158 } 159 } 160 if (mOriginsToAllow != null) { 161 for (String origin : mOriginsToAllow) { 162 nativeAllow(origin); 163 } 164 } 165 } 166 } 167 168 /** 169 * Utility function to send a message to our handler. 170 */ postMessage(Message msg)171 private void postMessage(Message msg) { 172 assert(mHandler != null); 173 mHandler.sendMessage(msg); 174 } 175 176 /** 177 * Utility function to send a message to the handler on the UI thread 178 */ postUIMessage(Message msg)179 private void postUIMessage(Message msg) { 180 if (mUIHandler != null) { 181 mUIHandler.sendMessage(msg); 182 } 183 } 184 185 /** 186 * Gets the set of origins for which Geolocation permissions are stored. 187 * Note that we represent the origins as strings. These are created using 188 * WebCore::SecurityOrigin::toString(). As long as all 'HTML 5 modules' 189 * (Database, Geolocation etc) do so, it's safe to match up origins for the 190 * purposes of displaying UI. 191 * @hide 192 */ getOrigins(ValueCallback<Set> callback)193 public void getOrigins(ValueCallback<Set> callback) { 194 if (callback != null) { 195 if (WebViewCore.THREAD_NAME.equals(Thread.currentThread().getName())) { 196 getOriginsImpl(); 197 Set origins = new HashSet(mOrigins); 198 callback.onReceiveValue(origins); 199 } else { 200 postMessage(Message.obtain(null, GET_ORIGINS, callback)); 201 } 202 } 203 } 204 205 /** 206 * Helper method to get the set of origins. 207 */ getOriginsImpl()208 private void getOriginsImpl() { 209 // Called on the WebKit thread. 210 mOrigins = nativeGetOrigins(); 211 } 212 213 /** 214 * Gets the permission state for the specified origin. 215 * @hide 216 */ getAllowed(String origin, ValueCallback<Boolean> callback)217 public void getAllowed(String origin, ValueCallback<Boolean> callback) { 218 if (callback == null) { 219 return; 220 } 221 if (origin == null) { 222 callback.onReceiveValue(null); 223 return; 224 } 225 if (WebViewCore.THREAD_NAME.equals(Thread.currentThread().getName())) { 226 getAllowedImpl(origin); 227 callback.onReceiveValue(new Boolean(mAllowed)); 228 } else { 229 Map values = new HashMap<String, Object>(); 230 values.put(ORIGIN, origin); 231 values.put(CALLBACK, callback); 232 postMessage(Message.obtain(null, GET_ALLOWED, values)); 233 } 234 } 235 236 /** 237 * Helper method to get the permission state. 238 */ getAllowedImpl(String origin)239 private void getAllowedImpl(String origin) { 240 // Called on the WebKit thread. 241 mAllowed = nativeGetAllowed(origin); 242 } 243 244 /** 245 * Clears the permission state for the specified origin. This method may be 246 * called before the WebKit thread has intialized the message handler. 247 * Messages will be queued until this time. 248 * @hide 249 */ clear(String origin)250 public void clear(String origin) { 251 // Called on the UI thread. 252 if (mHandler == null) { 253 if (mOriginsToClear == null) { 254 mOriginsToClear = new HashSet<String>(); 255 } 256 mOriginsToClear.add(origin); 257 if (mOriginsToAllow != null) { 258 mOriginsToAllow.remove(origin); 259 } 260 } else { 261 postMessage(Message.obtain(null, CLEAR, origin)); 262 } 263 } 264 265 /** 266 * Allows the specified origin. This method may be called before the WebKit 267 * thread has intialized the message handler. Messages will be queued until 268 * this time. 269 * @hide 270 */ allow(String origin)271 public void allow(String origin) { 272 // Called on the UI thread. 273 if (mHandler == null) { 274 if (mOriginsToAllow == null) { 275 mOriginsToAllow = new HashSet<String>(); 276 } 277 mOriginsToAllow.add(origin); 278 if (mOriginsToClear != null) { 279 mOriginsToClear.remove(origin); 280 } 281 } else { 282 postMessage(Message.obtain(null, ALLOW, origin)); 283 } 284 } 285 286 /** 287 * Clears the permission state for all origins. 288 * @hide 289 */ clearAll()290 public void clearAll() { 291 // Called on the UI thread. 292 postMessage(Message.obtain(null, CLEAR_ALL)); 293 } 294 295 // Native functions, run on the WebKit thread. nativeGetOrigins()296 private static native Set nativeGetOrigins(); nativeGetAllowed(String origin)297 private static native boolean nativeGetAllowed(String origin); nativeClear(String origin)298 private static native void nativeClear(String origin); nativeAllow(String origin)299 private static native void nativeAllow(String origin); nativeClearAll()300 private static native void nativeClearAll(); 301 } 302