• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /**
2  * Copyright (c) 2010, 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.content;
18 
19 import android.content.Context;
20 import android.os.Message;
21 import android.os.RemoteException;
22 import android.os.Handler;
23 import android.os.IBinder;
24 import android.os.ServiceManager;
25 
26 import java.util.ArrayList;
27 
28 /**
29  * Interface to the clipboard service, for placing and retrieving text in
30  * the global clipboard.
31  *
32  * <p>
33  * You do not instantiate this class directly; instead, retrieve it through
34  * {@link android.content.Context#getSystemService}.
35  *
36  * <p>
37  * The ClipboardManager API itself is very simple: it consists of methods
38  * to atomically get and set the current primary clipboard data.  That data
39  * is expressed as a {@link ClipData} object, which defines the protocol
40  * for data exchange between applications.
41  *
42  * <div class="special reference">
43  * <h3>Developer Guides</h3>
44  * <p>For more information about using the clipboard framework, read the
45  * <a href="{@docRoot}guide/topics/clipboard/copy-paste.html">Copy and Paste</a>
46  * developer guide.</p>
47  * </div>
48  *
49  * @see android.content.Context#getSystemService
50  */
51 public class ClipboardManager extends android.text.ClipboardManager {
52     private final static Object sStaticLock = new Object();
53     private static IClipboard sService;
54 
55     private final Context mContext;
56 
57     private final ArrayList<OnPrimaryClipChangedListener> mPrimaryClipChangedListeners
58              = new ArrayList<OnPrimaryClipChangedListener>();
59 
60     private final IOnPrimaryClipChangedListener.Stub mPrimaryClipChangedServiceListener
61             = new IOnPrimaryClipChangedListener.Stub() {
62         public void dispatchPrimaryClipChanged() {
63             mHandler.sendEmptyMessage(MSG_REPORT_PRIMARY_CLIP_CHANGED);
64         }
65     };
66 
67     static final int MSG_REPORT_PRIMARY_CLIP_CHANGED = 1;
68 
69     private final Handler mHandler = new Handler() {
70         @Override
71         public void handleMessage(Message msg) {
72             switch (msg.what) {
73                 case MSG_REPORT_PRIMARY_CLIP_CHANGED:
74                     reportPrimaryClipChanged();
75             }
76         }
77     };
78 
79     /**
80      * Defines a listener callback that is invoked when the primary clip on the clipboard changes.
81      * Objects that want to register a listener call
82      * {@link android.content.ClipboardManager#addPrimaryClipChangedListener(OnPrimaryClipChangedListener)
83      * addPrimaryClipChangedListener()} with an
84      * object that implements OnPrimaryClipChangedListener.
85      *
86      */
87     public interface OnPrimaryClipChangedListener {
88 
89         /**
90          * Callback that is invoked by {@link android.content.ClipboardManager} when the primary
91          * clip changes.
92          */
onPrimaryClipChanged()93         void onPrimaryClipChanged();
94     }
95 
getService()96     static private IClipboard getService() {
97         synchronized (sStaticLock) {
98             if (sService != null) {
99                 return sService;
100             }
101             IBinder b = ServiceManager.getService("clipboard");
102             sService = IClipboard.Stub.asInterface(b);
103             return sService;
104         }
105     }
106 
107     /** {@hide} */
ClipboardManager(Context context, Handler handler)108     public ClipboardManager(Context context, Handler handler) {
109         mContext = context;
110     }
111 
112     /**
113      * Sets the current primary clip on the clipboard.  This is the clip that
114      * is involved in normal cut and paste operations.
115      *
116      * @param clip The clipped data item to set.
117      */
setPrimaryClip(ClipData clip)118     public void setPrimaryClip(ClipData clip) {
119         try {
120             if (clip != null) {
121                 clip.prepareToLeaveProcess(true);
122             }
123             getService().setPrimaryClip(clip, mContext.getOpPackageName());
124         } catch (RemoteException e) {
125             throw e.rethrowFromSystemServer();
126         }
127     }
128 
129     /**
130      * Returns the current primary clip on the clipboard.
131      */
getPrimaryClip()132     public ClipData getPrimaryClip() {
133         try {
134             return getService().getPrimaryClip(mContext.getOpPackageName());
135         } catch (RemoteException e) {
136             throw e.rethrowFromSystemServer();
137         }
138     }
139 
140     /**
141      * Returns a description of the current primary clip on the clipboard
142      * but not a copy of its data.
143      */
getPrimaryClipDescription()144     public ClipDescription getPrimaryClipDescription() {
145         try {
146             return getService().getPrimaryClipDescription(mContext.getOpPackageName());
147         } catch (RemoteException e) {
148             throw e.rethrowFromSystemServer();
149         }
150     }
151 
152     /**
153      * Returns true if there is currently a primary clip on the clipboard.
154      */
hasPrimaryClip()155     public boolean hasPrimaryClip() {
156         try {
157             return getService().hasPrimaryClip(mContext.getOpPackageName());
158         } catch (RemoteException e) {
159             throw e.rethrowFromSystemServer();
160         }
161     }
162 
addPrimaryClipChangedListener(OnPrimaryClipChangedListener what)163     public void addPrimaryClipChangedListener(OnPrimaryClipChangedListener what) {
164         synchronized (mPrimaryClipChangedListeners) {
165             if (mPrimaryClipChangedListeners.size() == 0) {
166                 try {
167                     getService().addPrimaryClipChangedListener(
168                             mPrimaryClipChangedServiceListener, mContext.getOpPackageName());
169                 } catch (RemoteException e) {
170                     throw e.rethrowFromSystemServer();
171                 }
172             }
173             mPrimaryClipChangedListeners.add(what);
174         }
175     }
176 
removePrimaryClipChangedListener(OnPrimaryClipChangedListener what)177     public void removePrimaryClipChangedListener(OnPrimaryClipChangedListener what) {
178         synchronized (mPrimaryClipChangedListeners) {
179             mPrimaryClipChangedListeners.remove(what);
180             if (mPrimaryClipChangedListeners.size() == 0) {
181                 try {
182                     getService().removePrimaryClipChangedListener(
183                             mPrimaryClipChangedServiceListener);
184                 } catch (RemoteException e) {
185                     throw e.rethrowFromSystemServer();
186                 }
187             }
188         }
189     }
190 
191     /**
192      * @deprecated Use {@link #getPrimaryClip()} instead.  This retrieves
193      * the primary clip and tries to coerce it to a string.
194      */
getText()195     public CharSequence getText() {
196         ClipData clip = getPrimaryClip();
197         if (clip != null && clip.getItemCount() > 0) {
198             return clip.getItemAt(0).coerceToText(mContext);
199         }
200         return null;
201     }
202 
203     /**
204      * @deprecated Use {@link #setPrimaryClip(ClipData)} instead.  This
205      * creates a ClippedItem holding the given text and sets it as the
206      * primary clip.  It has no label or icon.
207      */
setText(CharSequence text)208     public void setText(CharSequence text) {
209         setPrimaryClip(ClipData.newPlainText(null, text));
210     }
211 
212     /**
213      * @deprecated Use {@link #hasPrimaryClip()} instead.
214      */
hasText()215     public boolean hasText() {
216         try {
217             return getService().hasClipboardText(mContext.getOpPackageName());
218         } catch (RemoteException e) {
219             throw e.rethrowFromSystemServer();
220         }
221     }
222 
reportPrimaryClipChanged()223     void reportPrimaryClipChanged() {
224         Object[] listeners;
225 
226         synchronized (mPrimaryClipChangedListeners) {
227             final int N = mPrimaryClipChangedListeners.size();
228             if (N <= 0) {
229                 return;
230             }
231             listeners = mPrimaryClipChangedListeners.toArray();
232         }
233 
234         for (int i=0; i<listeners.length; i++) {
235             ((OnPrimaryClipChangedListener)listeners[i]).onPrimaryClipChanged();
236         }
237     }
238 }
239