Android 截图功能源码的分析

Android 截图功能源码的分析

一般没有修改rom的android原生系统截图功能的组合键是音量减+开机键;今天我们从源码角度来分析截图功能是如何在源码中实现的。

在android系统中,由于我们的每一个Android界面都是一个Activity,而界面的显示都是通过Window对象实现的,每个Window对象实际上都是PhoneWindow的实例,而每个PhoneWindow对象都对应一个PhoneWindowManager对象,当我们在Activity界面执行按键操作的时候,在将按键的处理操作分发到App之前,首先会回调PhoneWindowManager中的dispatchUnhandledKey方法,该方法主要用于执行当前App处理按键之前的操作,我们具体看一下该方法的实现。

/** {@inheritDoc} */
  @Override
  public KeyEvent dispatchUnhandledKey(WindowState win, KeyEvent event, int policyFlags) {
    ...
    KeyEvent fallbackEvent = null;
    if ((event.getFlags() & KeyEvent.FLAG_FALLBACK) == 0) {
      final KeyCharacterMap kcm = event.getKeyCharacterMap();
      final int keyCode = event.getKeyCode();
      final int metaState = event.getMetaState();
      final boolean initialDown = event.getAction() == KeyEvent.ACTION_DOWN
          && event.getRepeatCount() == 0;

      // Check for fallback actions specified by the key character map.
      final FallbackAction fallbackAction;
      if (initialDown) {
        fallbackAction = kcm.getFallbackAction(keyCode, metaState);
      } else {
        fallbackAction = mFallbackActions.get(keyCode);
      }

      if (fallbackAction != null) {
        ...
        final int flags = event.getFlags() | KeyEvent.FLAG_FALLBACK;
        fallbackEvent = KeyEvent.obtain(
            event.getDownTime(), event.getEventTime(),
            event.getAction(), fallbackAction.keyCode,
            event.getRepeatCount(), fallbackAction.metaState,
            event.getDeviceId(), event.getScanCode(),
            flags, event.getSource(), null);

        if (!interceptFallback(win, fallbackEvent, policyFlags)) {
          fallbackEvent.recycle();
          fallbackEvent = null;
        }

        if (initialDown) {
          mFallbackActions.put(keyCode, fallbackAction);
        } else if (event.getAction() == KeyEvent.ACTION_UP) {
          mFallbackActions.remove(keyCode);
          fallbackAction.recycle();
        }
      }
    }

    ...
    return fallbackEvent;
  }

这里我们关注一下方法体中调用的:interceptFallback方法,通过调用该方法将处理按键的操作下发到该方法中,我们继续看一下该方法的实现逻辑。

private boolean interceptFallback(WindowState win, KeyEvent fallbackEvent, int policyFlags) {
    int actions = interceptKeyBeforeQueueing(fallbackEvent, policyFlags);
    if ((actions & ACTION_PASS_TO_USER) != 0) {
      long delayMillis = interceptKeyBeforeDispatching(
          win, fallbackEvent, policyFlags);
      if (delayMillis == 0) {
        return true;
      }
    }
    return false;
  }

然后我们看到在interceptFallback方法中我们调用了interceptKeyBeforeQueueing方法,通过阅读我们我们知道该方法主要实现了对截屏按键的处理流程,这样我们继续看一下interceptKeyBeforeWueueing方法的处理:

@Override
  public int interceptKeyBeforeQueueing(KeyEvent event, int policyFlags) {
    if (!mSystemBooted) {
      // If we have not yet booted, don't let key events do anything.
      return 0;
    }

    ...
    // Handle special keys.
    switch (keyCode) {
      case KeyEvent.KEYCODE_VOLUME_DOWN:
      case KeyEvent.KEYCODE_VOLUME_UP:
      case KeyEvent.KEYCODE_VOLUME_MUTE: {
        if (mUseTvRouting) {
          // On TVs volume keys never go to the foreground app
          result &= ~ACTION_PASS_TO_USER;
        }
        if (keyCode == KeyEvent.KEYCODE_VOLUME_DOWN) {
          if (down) {
            if (interactive && !mScreenshotChordVolumeDownKeyTriggered
                && (event.getFlags() & KeyEvent.FLAG_FALLBACK) == 0) {
              mScreenshotChordVolumeDownKeyTriggered = true;
              mScreenshotChordVolumeDownKeyTime = event.getDownTime();
              mScreenshotChordVolumeDownKeyConsumed = false;
              cancelPendingPowerKeyAction();
              interceptScreenshotChord();
            }
          } else {
            mScreenshotChordVolumeDownKeyTriggered = false;
            cancelPendingScreenshotChordAction();
          }
        }
        ...

    return result;
  }

可以发现这里首先判断当前系统是否已经boot完毕,若尚未启动完毕,则所有的按键操作都将失效,若启动完成,则执行后续的操作,这里我们只是关注音量减少按键和电源按键组合的处理事件。另外这里多说一句想安卓系统的HOME按键事件,MENU按键事件,进程列表按键事件等等都是在这里实现的,后续中我们会陆续介绍这方面的内容。

回到我们的interceptKeyBeforeQueueing方法,当我用按下音量减少按键的时候回进入到:case KeyEvent.KEYCODE_VOLUME_MUTE分支并执行相应的逻辑,然后同时判断用户是否按下了电源键,若同时按下了电源键,则执行:

if (interactive && !mScreenshotChordVolumeDownKeyTriggered
                && (event.getFlags() & KeyEvent.FLAG_FALLBACK) == 0) {
              mScreenshotChordVolumeDownKeyTriggered = true;
              mScreenshotChordVolumeDownKeyTime = event.getDownTime();
              mScreenshotChordVolumeDownKeyConsumed = false;
              cancelPendingPowerKeyAction();
              interceptScreenshotChord();
            }

可以发现这里的interceptScreenshotChrod方法就是系统准备开始执行截屏操作的开始,我们继续看一下interceptcreenshotChord方法的实现。

private void interceptScreenshotChord() {
    if (mScreenshotChordEnabled
        && mScreenshotChordVolumeDownKeyTriggered && mScreenshotChordPowerKeyTriggered
        && !mScreenshotChordVolumeUpKeyTriggered) {
      final long now = SystemClock.uptimeMillis();
      if (now <= mScreenshotChordVolumeDownKeyTime + SCREENSHOT_CHORD_DEBOUNCE_DELAY_MILLIS
          && now <= mScreenshotChordPowerKeyTime
              + SCREENSHOT_CHORD_DEBOUNCE_DELAY_MILLIS) {
        mScreenshotChordVolumeDownKeyConsumed = true;
        cancelPendingPowerKeyAction();

        mHandler.postDelayed(mScreenshotRunnable, getScreenshotChordLongPressDelay());
      }
    }
  }

在方法体中我们最终会执行发送一个延迟的异步消息,请求执行截屏的操作而这里的延时时间,若当前输入框是打开状态,则延时时间为输入框关闭时间加上系统配置的按键超时时间,若当前输入框没有打开则直接是系统配置的按键超时处理时间,可看一下getScreenshotChordLongPressDelay方法的具体实现。

private long getScreenshotChordLongPressDelay() {
    if (mKeyguardDelegate.isShowing()) {
      // Double the time it takes to take a screenshot from the keyguard
      return (long) (KEYGUARD_SCREENSHOT_CHORD_DELAY_MULTIPLIER *
          ViewConfiguration.get(mContext).getDeviceGlobalActionKeyTimeout());
    }
    return ViewConfiguration.get(mContext).getDeviceGlobalActionKeyTimeout();
  }

回到我们的interceptScreenshotChord方法,发送了异步消息之后系统最终会被我们发送的Runnable对象的run方法执行;这样我们看一下Runnable类型的mScreenshotRunnable的run方法的实现:

private final Runnable mScreenshotRunnable = new Runnable() {
    @Override
    public void run() {
      takeScreenshot();
    }
  };

好吧,方法体中并未执行其他操作,直接就是调用了takeScreenshot方法,这样我们继续看一下takeScreenshot方法的实现。

private void takeScreenshot() {
    synchronized (mScreenshotLock) {
      if (mScreenshotConnection != null) {
        return;
      }
      ComponentName cn = new ComponentName("com.android.systemui",
          "com.android.systemui.screenshot.TakeScreenshotService");
      Intent intent = new Intent();
      intent.setComponent(cn);
      ServiceConnection conn = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
          synchronized (mScreenshotLock) {
            if (mScreenshotConnection != this) {
              return;
            }
            Messenger messenger = new Messenger(service);
            Message msg = Message.obtain(null, 1);
            final ServiceConnection myConn = this;
            Handler h = new Handler(mHandler.getLooper()) {
              @Override
              public void handleMessage(Message msg) {
                synchronized (mScreenshotLock) {
                  if (mScreenshotConnection == myConn) {
                    mContext.unbindService(mScreenshotConnection);
                    mScreenshotConnection = null;
                    mHandler.removeCallbacks(mScreenshotTimeout);
                  }
                }
              }
            };
            msg.replyTo = new Messenger(h);
            msg.arg1 = msg.arg2 = 0;
            if (mStatusBar != null && mStatusBar.isVisibleLw())
              msg.arg1 = 1;
            if (mNavigationBar != null && mNavigationBar.isVisibleLw())
              msg.arg2 = 1;
            try {
              messenger.send(msg);
            } catch (RemoteException e) {
            }
          }
        }
        @Override
        public void onServiceDisconnected(ComponentName name) {}
      };
      if (mContext.bindServiceAsUser(
          intent, conn, Context.BIND_AUTO_CREATE, UserHandle.CURRENT)) {
        mScreenshotConnection = conn;
        mHandler.postDelayed(mScreenshotTimeout, 10000);
      }
    }
  }

可以发现这里通过反射机制创建了一个TakeScreenshotService对象然后调用了bindServiceAsUser,这样就创建了TakeScreenshotService服务并在服务创建之后发送了一个异步消息。好了,我们看一下TakeScreenshotService的实现逻辑。

public class TakeScreenshotService extends Service {
  private static final String TAG = "TakeScreenshotService";

  private static GlobalScreenshot mScreenshot;

  private Handler mHandler = new Handler() {
    @Override
    public void handleMessage(Message msg) {
      switch (msg.what) {
        case 1:
          final Messenger callback = msg.replyTo;
          if (mScreenshot == null) {
            mScreenshot = new GlobalScreenshot(TakeScreenshotService.this);
          }
          mScreenshot.takeScreenshot(new Runnable() {
            @Override public void run() {
              Message reply = Message.obtain(null, 1);
              try {
                callback.send(reply);
              } catch (RemoteException e) {
              }
            }
          }, msg.arg1 > 0, msg.arg2 > 0);
      }
    }
  };

  @Override
  public IBinder onBind(Intent intent) {
    return new Messenger(mHandler).getBinder();
  }
}

可以发现在在TakeScreenshotService类的定义中有一个Handler成员变量,而我们在启动TakeScreentshowService的时候回发送一个异步消息,这样就会执行mHandler的handleMessage方法,然后在handleMessage方法中我们创建了一个GlobalScreenshow对象,然后执行了takeScreenshot方法,好吧,继续看一下takeScreentshot方法的执行逻辑。

/**
   * Takes a screenshot of the current display and shows an animation.
   */
  void takeScreenshot(Runnable finisher, boolean statusBarVisible, boolean navBarVisible) {
    // We need to orient the screenshot correctly (and the Surface api seems to take screenshots
    // only in the natural orientation of the device :!)
    mDisplay.getRealMetrics(mDisplayMetrics);
    float[] dims = {mDisplayMetrics.widthPixels, mDisplayMetrics.heightPixels};
    float degrees = getDegreesForRotation(mDisplay.getRotation());
    boolean requiresRotation = (degrees > 0);
    if (requiresRotation) {
      // Get the dimensions of the device in its native orientation
      mDisplayMatrix.reset();
      mDisplayMatrix.preRotate(-degrees);
      mDisplayMatrix.mapPoints(dims);
      dims[0] = Math.abs(dims[0]);
      dims[1] = Math.abs(dims[1]);
    }

    // Take the screenshot
    mScreenBitmap = SurfaceControl.screenshot((int) dims[0], (int) dims[1]);
    if (mScreenBitmap == null) {
      notifyScreenshotError(mContext, mNotificationManager);
      finisher.run();
      return;
    }

    if (requiresRotation) {
      // Rotate the screenshot to the current orientation
      Bitmap ss = Bitmap.createBitmap(mDisplayMetrics.widthPixels,
          mDisplayMetrics.heightPixels, Bitmap.Config.ARGB_8888);
      Canvas c = new Canvas(ss);
      c.translate(ss.getWidth() / 2, ss.getHeight() / 2);
      c.rotate(degrees);
      c.translate(-dims[0] / 2, -dims[1] / 2);
      c.drawBitmap(mScreenBitmap, 0, 0, null);
      c.setBitmap(null);
      // Recycle the previous bitmap
      mScreenBitmap.recycle();
      mScreenBitmap = ss;
    }

    // Optimizations
    mScreenBitmap.setHasAlpha(false);
    mScreenBitmap.prepareToDraw();

    // Start the post-screenshot animation
    startAnimation(finisher, mDisplayMetrics.widthPixels, mDisplayMetrics.heightPixels,
        statusBarVisible, navBarVisible);
  }

可以看到这里后两个参数:statusBarVisible,navBarVisible是否可见,而这两个参数在我们

PhoneWindowManager.takeScreenshot方法传递的:

if (mStatusBar != null && mStatusBar.isVisibleLw())
              msg.arg1 = 1;
            if (mNavigationBar != null && mNavigationBar.isVisibleLw())
              msg.arg2 = 1;

可见若果mStatusBar可见,则传递的statusBarVisible为true,若mNavigationBar可见,则传递的navBarVisible为true。然后我们在截屏的时候判断nStatusBar是否可见,mNavigationBar是否可见,若可见的时候则截屏同样将其截屏出来。继续回到我们的takeScreenshot方法,然后调用了:

// Take the screenshot
mScreenBitmap = SurfaceControl.screenshot((int) dims[0], (int) dims[1]);

方法,看注释,这里就是执行截屏事件的具体操作了,然后我看一下SurfaceControl.screenshot方法的具体实现,另外这里需要注意的是,截屏之后返回的是一个Bitmap对象,其实熟悉android绘制机制的童鞋应该知道android中所有显示能够显示的东西,在内存中表现都是Bitmap对象。

public static Bitmap screenshot(int width, int height) {
    // TODO: should take the display as a parameter
    IBinder displayToken = SurfaceControl.getBuiltInDisplay(
        SurfaceControl.BUILT_IN_DISPLAY_ID_MAIN);
    return nativeScreenshot(displayToken, new Rect(), width, height, 0, 0, true,
        false, Surface.ROTATION_0);
  }

好吧,这里调用的是nativeScreenshot方法,它是一个native方法,具体的实现在JNI层,这里就不做过多的介绍了。继续回到我们的takeScreenshot方法,在调用了截屏方法screentshot之后,判断是否截屏成功:

if (mScreenBitmap == null) {
      notifyScreenshotError(mContext, mNotificationManager);
      finisher.run();
      return;
    }

若截屏之后,截屏的bitmap对象为空,这里判断截屏失败,调用了notifyScreenshotError方法,发送截屏失败的notification通知。

static void notifyScreenshotError(Context context, NotificationManager nManager) {
    Resources r = context.getResources();

    // Clear all existing notification, compose the new notification and show it
    Notification.Builder b = new Notification.Builder(context)
      .setTicker(r.getString(R.string.screenshot_failed_title))
      .setContentTitle(r.getString(R.string.screenshot_failed_title))
      .setContentText(r.getString(R.string.screenshot_failed_text))
      .setSmallIcon(R.drawable.stat_notify_image_error)
      .setWhen(System.currentTimeMillis())
      .setVisibility(Notification.VISIBILITY_PUBLIC) // ok to show outside lockscreen
      .setCategory(Notification.CATEGORY_ERROR)
      .setAutoCancel(true)
      .setColor(context.getColor(
            com.android.internal.R.color.system_notification_accent_color));
    Notification n =
      new Notification.BigTextStyle(b)
        .bigText(r.getString(R.string.screenshot_failed_text))
        .build();
    nManager.notify(R.id.notification_screenshot, n);
  }

然后继续看takeScreenshot方法,判断截屏的图像是否需要旋转,若需要的话,则旋转图像:

if (requiresRotation) {
      // Rotate the screenshot to the current orientation
      Bitmap ss = Bitmap.createBitmap(mDisplayMetrics.widthPixels,
          mDisplayMetrics.heightPixels, Bitmap.Config.ARGB_8888);
      Canvas c = new Canvas(ss);
      c.translate(ss.getWidth() / 2, ss.getHeight() / 2);
      c.rotate(degrees);
      c.translate(-dims[0] / 2, -dims[1] / 2);
      c.drawBitmap(mScreenBitmap, 0, 0, null);
      c.setBitmap(null);
      // Recycle the previous bitmap
      mScreenBitmap.recycle();
      mScreenBitmap = ss;
    }

在takeScreenshot方法的最后若截屏成功,我们调用了:

// Start the post-screenshot animation
    startAnimation(finisher, mDisplayMetrics.widthPixels, mDisplayMetrics.heightPixels,
        statusBarVisible, navBarVisible);

开始截屏的动画,好吧,看一下动画效果的实现:

/**
   * Starts the animation after taking the screenshot
   */
  private void startAnimation(final Runnable finisher, int w, int h, boolean statusBarVisible,
      boolean navBarVisible) {
    // Add the view for the animation
    mScreenshotView.setImageBitmap(mScreenBitmap);
    mScreenshotLayout.requestFocus();

    // Setup the animation with the screenshot just taken
    if (mScreenshotAnimation != null) {
      mScreenshotAnimation.end();
      mScreenshotAnimation.removeAllListeners();
    }

    mWindowManager.addView(mScreenshotLayout, mWindowLayoutParams);
    ValueAnimator screenshotDropInAnim = createScreenshotDropInAnimation();
    ValueAnimator screenshotFadeOutAnim = createScreenshotDropOutAnimation(w, h,
        statusBarVisible, navBarVisible);
    mScreenshotAnimation = new AnimatorSet();
    mScreenshotAnimation.playSequentially(screenshotDropInAnim, screenshotFadeOutAnim);
    mScreenshotAnimation.addListener(new AnimatorListenerAdapter() {
      @Override
      public void onAnimationEnd(Animator animation) {
        // Save the screenshot once we have a bit of time now
        saveScreenshotInWorkerThread(finisher);
        mWindowManager.removeView(mScreenshotLayout);

        // Clear any references to the bitmap
        mScreenBitmap = null;
        mScreenshotView.setImageBitmap(null);
      }
    });
    mScreenshotLayout.post(new Runnable() {
      @Override
      public void run() {
        // Play the shutter sound to notify that we've taken a screenshot
        mCameraSound.play(MediaActionSound.SHUTTER_CLICK);

        mScreenshotView.setLayerType(View.LAYER_TYPE_HARDWARE, null);
        mScreenshotView.buildLayer();
        mScreenshotAnimation.start();
      }
    });
  }

好吧,经过着一些列的操作之后我们实现了截屏之后的动画效果了,这里暂时不分析动画效果,我们看一下动画效果之后做了哪些?还记不记的一般情况下我们截屏之后都会收到一个截屏的notification通知?这里应该也是在其AnimatorListenerAdapter的onAnimationEnd方法中实现的,也就是动画执行完成之后,我们看一下其saveScreenshotInWorkerThread方法的实现:

/**
   * Creates a new worker thread and saves the screenshot to the media store.
   */
  private void saveScreenshotInWorkerThread(Runnable finisher) {
    SaveImageInBackgroundData data = new SaveImageInBackgroundData();
    data.context = mContext;
    data.image = mScreenBitmap;
    data.iconSize = mNotificationIconSize;
    data.finisher = finisher;
    data.previewWidth = mPreviewWidth;
    data.previewheight = mPreviewHeight;
    if (mSaveInBgTask != null) {
      mSaveInBgTask.cancel(false);
    }
    mSaveInBgTask = new SaveImageInBackgroundTask(mContext, data, mNotificationManager,
        R.id.notification_screenshot).execute(data);
  }

好吧,这里主要逻辑就是构造了一个SaveImageInBackgroundTask对象,看样子发送截屏成功的通知应该是在这里实现的,我们看一下SaveImageInBackgroundTask构造方法的实现逻辑:

SaveImageInBackgroundTask(Context context, SaveImageInBackgroundData data,
      NotificationManager nManager, int nId) {
    ...

    // Show the intermediate notification
    mTickerAddSpace = !mTickerAddSpace;
    mNotificationId = nId;
    mNotificationManager = nManager;
    final long now = System.currentTimeMillis();

    mNotificationBuilder = new Notification.Builder(context)
      .setTicker(r.getString(R.string.screenshot_saving_ticker)
          + (mTickerAddSpace ? " " : ""))
      .setContentTitle(r.getString(R.string.screenshot_saving_title))
      .setContentText(r.getString(R.string.screenshot_saving_text))
      .setSmallIcon(R.drawable.stat_notify_image)
      .setWhen(now)
      .setColor(r.getColor(com.android.internal.R.color.system_notification_accent_color));

    mNotificationStyle = new Notification.BigPictureStyle()
      .bigPicture(picture.createAshmemBitmap());
    mNotificationBuilder.setStyle(mNotificationStyle);

    // For "public" situations we want to show all the same info but
    // omit the actual screenshot image.
    mPublicNotificationBuilder = new Notification.Builder(context)
        .setContentTitle(r.getString(R.string.screenshot_saving_title))
        .setContentText(r.getString(R.string.screenshot_saving_text))
        .setSmallIcon(R.drawable.stat_notify_image)
        .setCategory(Notification.CATEGORY_PROGRESS)
        .setWhen(now)
        .setColor(r.getColor(
            com.android.internal.R.color.system_notification_accent_color));

    mNotificationBuilder.setPublicVersion(mPublicNotificationBuilder.build());

    Notification n = mNotificationBuilder.build();
    n.flags |= Notification.FLAG_NO_CLEAR;
    mNotificationManager.notify(nId, n);

    // On the tablet, the large icon makes the notification appear as if it is clickable (and
    // on small devices, the large icon is not shown) so defer showing the large icon until
    // we compose the final post-save notification below.
    mNotificationBuilder.setLargeIcon(icon.createAshmemBitmap());
    // But we still don't set it for the expanded view, allowing the smallIcon to show here.
    mNotificationStyle.bigLargeIcon((Bitmap) null);
  }

可以发现在构造方法的后面狗仔了一个NotificationBuilder对象,然后发送了一个截屏成功的Notification,这样我们在截屏动画之后就收到了Notification的通知了。

总结:

一般默认情况下按下音量减少键和开机键会执行截图动作,程序执行的入口就在在PhoneWindowManager的dispatchUnhandledKey方法中;然后通过TakeScreenshotService服务执行截图逻辑;通过nativie方法获取截图的bitmap,如果失败调用失败通知栏消息,如果成功调用截图动画后发送成功通知栏消息。

如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • Android截屏截图的几种方法总结

    Android截屏 Android截屏的原理:获取具体需要截屏的区域的Bitmap,然后绘制在画布上,保存为图片后进行分享或者其它用途 一.Activity截屏 1.截Activity界面(包含空白的状态栏) /** * 根据指定的Activity截图(带空白的状态栏) * * @param context 要截图的Activity * @return Bitmap */ public static Bitmap shotActivity(Activity context) { View vie

  • Android中如何获取视频文件的截图、缩略图

    背景 公司最近要求给我负责的APP加上视频录制和发布的功能,我简单的完成了基本的录制和视频压缩功能,后来发现发布接口需要上传视频的截图,网上搜索了一下资料,在这里整理一下. 代码实现 /** * 获取视频文件截图 * * @param path 视频文件的路径 * @return Bitmap 返回获取的Bitmap */ public static Bitmap getVideoThumb(String path) { MediaMetadataRetriever media = new Me

  • Android实现截图和分享功能的代码

    先给大家展示下效果图吧 直接上代码: xml的布局: <Button android:id="@+id/btn_jp" android:layout_marginTop="10dip" android:layout_width="match_parent" android:layout_height="wrap_content" android:gravity="center" android:tex

  • Android屏幕及view的截图实例详解

    Android屏幕及view的截图实例详解 屏幕可见区域的截图 整个屏幕截图的话可以用View view = getWindow().getDecorView(); public static Bitmap getNormalViewScreenshot(View view) { view.setDrawingCacheEnabled(true); view.buildDrawingCache(); return view.getDrawingCache(); } scrollview的整体截屏

  • Android仿银行客户签名并且保存签名的截图文件并命名为本地时间

    首先需要一个自定义view用来签字使用,可以修改颜色和画笔的粗细,可以擦拭重新画 package com.android.tcm.view; import android.content.Context; import android.graphics.Bitmap; import android.graphics.Bitmap.Config; import android.graphics.Canvas; import android.graphics.Color; import androi

  • Android中通过view方式获取当前Activity的屏幕截图实现方法

    此方法是通过view的方式获取当前activity的屏幕截图,并不是framebuffer的方式,所以有一定的局限性.但是这种方法相对简单,容易理解. 首先通过下面的函数获取Bitmap格式的屏幕截图: 复制代码 代码如下: public Bitmap myShot(Activity activity) { // 获取windows中最顶层的view View view = activity.getWindow().getDecorView(); view.buildDrawingCache()

  • android截图事件监听的原理与实现

    Android系统没有对用户截屏行为提供回调的api,所以我们只能走野路子来获取用户是否截屏了.一般大家都会采用如下两种方法 1.监听截屏图片所在目录变化(FileObserver) 2.监听媒体库的变化(ContentObserver) 上面两种方法均不是万能的,需要结合使用才能达到良好的效果,首先看看如何监控目录 在android中,我们可以通过FileObserver来监听目录变化,先来看看如何使用 private static final File DIRECTORY_PICTURES

  • Android 截图功能源码的分析

    Android 截图功能源码的分析 一般没有修改rom的android原生系统截图功能的组合键是音量减+开机键:今天我们从源码角度来分析截图功能是如何在源码中实现的. 在android系统中,由于我们的每一个Android界面都是一个Activity,而界面的显示都是通过Window对象实现的,每个Window对象实际上都是PhoneWindow的实例,而每个PhoneWindow对象都对应一个PhoneWindowManager对象,当我们在Activity界面执行按键操作的时候,在将按键的处

  • elasticsearch分布式及数据的功能源码分析

    从功能上说,可以分为两部分,分布式功能和数据功能.分布式功能主要是节点集群及集群附属功能如restful借口.集群性能检测功能等,数据功能主要是索引和搜索.代码上这些功能并不是完全独立,而是由相互交叉部分.当然分布式功能是为数据功能服务,数据功能肯定也难以完全独立于分布式功能. 它的源码有以下几个特点: 模块化: 每个功能都以模块化的方式实现,最后以一个借口向外暴露,最终通过guice(google轻量级DI框架)进行管理.整个系统有30多个模块(version1.5). 接口解耦: es代码中

  • PHP+jQuery实现自动补全功能源码

    前面手工写了一个下拉自动补全功能,写的简单,只实现了鼠标选择的功能,不支持键盘选择.由于项目很多地方要用到这个功能,所以需要用心做一下.发现select2这个插件的功能可以满足当前需求. 在使用jquery插件select2的过程中遇到了一些疑惑,无论是穿json数据还是通过jsonp方式取数据,都能够正确返回.可是下拉列表中的条目却不能被选中,对鼠标和键盘选择都无效. 后来发现,select2插件在实现选中时是以数据中的id字段为准的.所以不管是json还是jsonp,ajax返回的数据都必须

  • elasticsearch索引index数据功能源码示例

    从本篇开始,对elasticsearch的介绍将进入数据功能部分(index),这一部分包括索引的创建,管理,数据索引及搜索等相关功能.对于这一部分的介绍,首先对各个功能模块的分析,然后详细分析数据索引和搜索的整个流程. 这一部分从代码包结构上可以分为:index, indices及lucene(common)几个部分.index包中的代码主要是各个功能对应于lucene的底层操作,它们的操作对象是index的shard,是elasticsearch对lucene各个功能的扩展和封装.indic

  • asp下实现代码的“运行代码”“复制代码”“保存代码”功能源码

    Function content_Code(Str) dim ary_String,i,n,n_pos ary_String=split(Str,"[ code ]") n=ubound(ary_String) If n<1 then content_Code=Str Exit function End If for i=1 to n n_pos=inStr(ary_String(i),"[/ code ]") If n_pos>0 then ary_S

  • C语言实现的统计php代码行数功能源码(支持文件夹、多目录)

    放假在家没事,睡过懒觉,看过电影,就想起来写个小程序. 统计php代码的行数,对于phper还是挺实用的.支持单个文件和目录.下面是代码和演示的例子! /**  * @date     2012-12-1  * @author bright  * @todo     统计php代码行数  */ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> #inc

  • Android ViewPager源码详细分析

    1.问题 由于Android Framework源码很庞大,所以读源码必须带着问题来读!没有问题,创造问题再来读!否则很容易迷失在无数的方法与属性之中,最后无功而返. 那么,关于ViewPager有什么问题呢? 1). setOffsreenPageLimit()方法是如何实现页面缓存的? 2). 在布局文件中,ViewPager布局内部能否添加其他View? 3). 为什么ViewPager初始化时,显示了一个页面却不会触发onPageSelected回调? 问题肯定不止这三个,但是有这三个问

  • 从源码角度分析Android的消息机制

    前言 说到Android的消息机制,那么主要的就是指的Handler的运行机制.其中包括MessageQueue以及Looper的工作过程. 在开始正文之前,先抛出两个问题: 为什么更新UI的操作要在主线程中进行? Android中为什么主线程不会因为Looper.loop()里的死循环卡死? UI线程的判断是在ViewRootImpl中的checkThread方法中完成的. 对于第一个问题,这里给一个简单的回答: 如果可以在子线程中修改UI,多线程的并发访问可能会导致UI控件的不可预期性,采用

  • 分析Android 11.0Settings源码之主界面加载

    本篇主要记录AndroidR Settings源码主界面加载流程,方便后续工作调试其流程. Settings代码路径: packages/app/Settings/ Settings代码获取: Setting 源码下载地址:https://github.com/aosp-mirror/platform_packages_apps_settings git地址:https://github.com/aosp-mirror/platform_packages_apps_settings.git 主界

  • Android开发Retrofit源码分析

    目录 项目结构 retrofit 使用 Retrofit #create ServiceMethod #parseAnnotations HttpServiceMethod#parseAnnotations 第二种 非Kotlin协程情况 DefaultCallAdapterFactory#get 第一种 Kotlin协程情况 总结 项目结构 把源码 clone 下来 , 可以看到 retrofit 整体结构如下 图 http包目录下就是一些http协议常用接口 , 比如 请求方法 url ,

随机推荐