mirror of
https://github.com/NativeScript/NativeScript.git
synced 2025-08-16 11:42:04 +08:00
Rewrote image-cache to use the native image caching features, i.e. LruCache and NSCache.
This commit is contained in:
@ -50,7 +50,11 @@ export class RedditViewModel extends observable.Observable {
|
|||||||
} else if (redditAppViewModel.cache) {
|
} else if (redditAppViewModel.cache) {
|
||||||
var url = this._source.thumbnail;
|
var url = this._source.thumbnail;
|
||||||
|
|
||||||
var imgSource = redditAppViewModel.cache.get(url);
|
var imgSource: imageSource.ImageSource;
|
||||||
|
var image = redditAppViewModel.cache.get(url);
|
||||||
|
if (image) {
|
||||||
|
imgSource = imageSource.fromNativeSource(image);
|
||||||
|
}
|
||||||
|
|
||||||
if (imgSource) {
|
if (imgSource) {
|
||||||
this._thumbnailImageSource = imgSource;
|
this._thumbnailImageSource = imgSource;
|
||||||
@ -61,11 +65,12 @@ export class RedditViewModel extends observable.Observable {
|
|||||||
redditAppViewModel.cache.push({
|
redditAppViewModel.cache.push({
|
||||||
key: url,
|
key: url,
|
||||||
url: url,
|
url: url,
|
||||||
completed: (result: imageSource.ImageSource, key: string) => {
|
completed: (image: any, key: string) => {
|
||||||
if (url === key) {
|
if (url === key) {
|
||||||
this.isLoading = false;
|
this.isLoading = false;
|
||||||
this._thumbnailImageSource = result;
|
var imgSource = imageSource.fromNativeSource(image);
|
||||||
this.notify({ object: this, eventName: observable.knownEvents.propertyChange, propertyName: THUMBNAIL_IMAGE_SOURCE, value: result });
|
this._thumbnailImageSource = imgSource;
|
||||||
|
this.notify({ object: this, eventName: observable.knownEvents.propertyChange, propertyName: THUMBNAIL_IMAGE_SOURCE, value: imgSource });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -19,22 +19,22 @@ export function test_DummyTestForSnippetOnly() {
|
|||||||
//// Enable download while not scrolling
|
//// Enable download while not scrolling
|
||||||
cache.enableDownload();
|
cache.enableDownload();
|
||||||
|
|
||||||
var src: imageSource.ImageSource;
|
var imgSouce: imageSource.ImageSource;
|
||||||
var url = "https://github.com/NativeScript.png";
|
var url = "https://github.com/NativeScript.png";
|
||||||
//// Try to read the image from the cache
|
//// Try to read the image from the cache
|
||||||
var result = cache.get(url);
|
var image = cache.get(url);
|
||||||
if (result) {
|
if (image) {
|
||||||
//// If present -- use it.
|
//// If present -- use it.
|
||||||
src = result;
|
imgSouce = imageSource.fromNativeSource(image);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
//// If not present -- request its download.
|
//// If not present -- request its download.
|
||||||
cache.push({
|
cache.push({
|
||||||
key: url,
|
key: url,
|
||||||
url: url,
|
url: url,
|
||||||
completed: (result: imageSource.ImageSource, key: string) => {
|
completed: (image: any, key: string) => {
|
||||||
if (url === key) {
|
if (url === key) {
|
||||||
src = result;
|
imgSouce = imageSource.fromNativeSource(image);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
@ -9,7 +9,7 @@ export module knownEvents {
|
|||||||
export interface DownloadRequest {
|
export interface DownloadRequest {
|
||||||
url: string;
|
url: string;
|
||||||
key: string;
|
key: string;
|
||||||
completed?: (result: imageSource.ImageSource, key: string) => void;
|
completed?: (image: any, key: string) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Cache extends observable.Observable implements definition.Cache {
|
export class Cache extends observable.Observable implements definition.Cache {
|
||||||
@ -17,7 +17,6 @@ export class Cache extends observable.Observable implements definition.Cache {
|
|||||||
public maxRequests = 5;
|
public maxRequests = 5;
|
||||||
private _enabled = true;
|
private _enabled = true;
|
||||||
|
|
||||||
private _cache = {};
|
|
||||||
private _pendingDownloads = {};
|
private _pendingDownloads = {};
|
||||||
private _queue: Array<DownloadRequest> = [];
|
private _queue: Array<DownloadRequest> = [];
|
||||||
private _currentDownloads = 0;
|
private _currentDownloads = 0;
|
||||||
@ -98,47 +97,42 @@ export class Cache extends observable.Observable implements definition.Cache {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public get(key: string): imageSource.ImageSource {
|
public get(key: string): any {
|
||||||
var value = this._cache[key];
|
// This method is intended to be overridden by the android and ios implementations
|
||||||
if (value) {
|
throw new Error("Abstract");
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
return undefined;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public set(key: string, source: imageSource.ImageSource): void {
|
public set(key: string, image: any): void {
|
||||||
this._cache[key] = source;
|
// This method is intended to be overridden by the android and ios implementations
|
||||||
|
throw new Error("Abstract");
|
||||||
}
|
}
|
||||||
|
|
||||||
public remove(key: string): void {
|
public remove(key: string): void {
|
||||||
delete this._cache[key];
|
// This method is intended to be overridden by the android and ios implementations
|
||||||
|
throw new Error("Abstract");
|
||||||
}
|
}
|
||||||
|
|
||||||
public clear() {
|
public clear() {
|
||||||
var keys = Object.keys(this._cache);
|
// This method is intended to be overridden by the android and ios implementations
|
||||||
var i;
|
throw new Error("Abstract");
|
||||||
var length = keys.length;
|
|
||||||
|
|
||||||
for (i = 0; i < length; i++) {
|
|
||||||
delete this._cache[keys[i]];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* tslint:disable:no-unused-variable */
|
/* tslint:disable:no-unused-variable */
|
||||||
public _downloadCore(request: definition.DownloadRequest) {
|
public _downloadCore(request: definition.DownloadRequest) {
|
||||||
// This method is intended to be overridden by the android and ios implementations
|
// This method is intended to be overridden by the android and ios implementations
|
||||||
|
throw new Error("Abstract");
|
||||||
}
|
}
|
||||||
/* tslint:enable:no-unused-variable */
|
/* tslint:enable:no-unused-variable */
|
||||||
|
|
||||||
public _onDownloadCompleted(key: string, result: imageSource.ImageSource) {
|
public _onDownloadCompleted(key: string, image: any) {
|
||||||
var request = <DownloadRequest>this._pendingDownloads[key];
|
var request = <DownloadRequest>this._pendingDownloads[key];
|
||||||
|
|
||||||
this._cache[request.key] = result;
|
this.set(request.key, image);
|
||||||
|
|
||||||
this._currentDownloads--;
|
this._currentDownloads--;
|
||||||
|
|
||||||
if (request.completed) {
|
if (request.completed) {
|
||||||
request.completed(result, request.key);
|
request.completed(image, request.key);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.hasListeners(knownEvents.downloaded)) {
|
if (this.hasListeners(knownEvents.downloaded)) {
|
||||||
@ -146,7 +140,7 @@ export class Cache extends observable.Observable implements definition.Cache {
|
|||||||
eventName: knownEvents.downloaded,
|
eventName: knownEvents.downloaded,
|
||||||
object: this,
|
object: this,
|
||||||
key: key,
|
key: key,
|
||||||
image: result
|
image: image
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -156,7 +150,7 @@ export class Cache extends observable.Observable implements definition.Cache {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private _shouldDownload(request: definition.DownloadRequest, onTop: boolean): boolean {
|
private _shouldDownload(request: definition.DownloadRequest, onTop: boolean): boolean {
|
||||||
if (request.key in this._cache || request.key in this._pendingDownloads) {
|
if (this.get(request.key) || request.key in this._pendingDownloads) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,20 +1,38 @@
|
|||||||
import common = require("ui/image-cache/image-cache-common");
|
import common = require("ui/image-cache/image-cache-common");
|
||||||
import imageSource = require("image-source");
|
|
||||||
|
|
||||||
module.exports.knownEvents = common.knownEvents;
|
module.exports.knownEvents = common.knownEvents;
|
||||||
|
|
||||||
|
class LruBitmapCache extends android.util.LruCache<string, android.graphics.Bitmap> {
|
||||||
|
constructor(cacheSize: number) {
|
||||||
|
super(cacheSize);
|
||||||
|
return global.__native(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected sizeOf(key: string, bitmap: android.graphics.Bitmap): number {
|
||||||
|
// The cache size will be measured in kilobytes rather than
|
||||||
|
// number of items.
|
||||||
|
var result = Math.round(bitmap.getByteCount() / 1024);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
export class Cache extends common.Cache {
|
export class Cache extends common.Cache {
|
||||||
private _callback: any;
|
private _callback: any;
|
||||||
|
private _cache: LruBitmapCache;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
super();
|
super();
|
||||||
|
|
||||||
|
var maxMemory = java.lang.Runtime.getRuntime().maxMemory() / 1024;
|
||||||
|
var cacheSize = maxMemory / 8;
|
||||||
|
this._cache = new LruBitmapCache(cacheSize);
|
||||||
|
|
||||||
var that = new WeakRef(this);
|
var that = new WeakRef(this);
|
||||||
this._callback = new (<any>com).tns.Async.CompleteCallback({
|
this._callback = new (<any>com).tns.Async.CompleteCallback({
|
||||||
onComplete: function (result: any, context: any) {
|
onComplete: function (result: any, context: any) {
|
||||||
var instance = that.get();
|
var instance = that.get();
|
||||||
if (instance) {
|
if (instance) {
|
||||||
instance._onBitmapDownloaded(result, context);
|
instance._onDownloadCompleted(context, result)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -24,11 +42,20 @@ export class Cache extends common.Cache {
|
|||||||
(<any>com).tns.Async.DownloadImage(request.url, this._callback, request.key);
|
(<any>com).tns.Async.DownloadImage(request.url, this._callback, request.key);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* tslint:disable:no-unused-variable */
|
public get(key: string): any {
|
||||||
private _onBitmapDownloaded(result: android.graphics.Bitmap, context: any) {
|
var result = this._cache.get(key);
|
||||||
// as a context we are receiving the key of the request
|
return result;
|
||||||
var source = imageSource.fromNativeSource(result);
|
}
|
||||||
this._onDownloadCompleted(context, source);
|
|
||||||
|
public set(key: string, image: any): void {
|
||||||
|
this._cache.put(key, image);
|
||||||
|
}
|
||||||
|
|
||||||
|
public remove(key: string): void {
|
||||||
|
this._cache.remove(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
public clear() {
|
||||||
|
this._cache.evictAll();
|
||||||
}
|
}
|
||||||
/* tslint:enable:no-unused-variable */
|
|
||||||
}
|
}
|
8
ui/image-cache/image-cache.d.ts
vendored
8
ui/image-cache/image-cache.d.ts
vendored
@ -20,7 +20,7 @@ declare module "ui/image-cache" {
|
|||||||
/**
|
/**
|
||||||
* An optional function to be called when the download is complete.
|
* An optional function to be called when the download is complete.
|
||||||
*/
|
*/
|
||||||
completed?: (result: imageSource.ImageSource, key: string) => void;
|
completed?: (image: any, key: string) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -57,11 +57,11 @@ declare module "ui/image-cache" {
|
|||||||
/**
|
/**
|
||||||
* Gets the image for the specified key. May be undefined if the key is not present in the cache.
|
* Gets the image for the specified key. May be undefined if the key is not present in the cache.
|
||||||
*/
|
*/
|
||||||
get(key: string): imageSource.ImageSource;
|
get(key: string): any;
|
||||||
/**
|
/**
|
||||||
* Sets the image for the specified key.
|
* Sets the image for the specified key.
|
||||||
*/
|
*/
|
||||||
set(key: string, source: imageSource.ImageSource): void;
|
set(key: string, image: any): void;
|
||||||
/**
|
/**
|
||||||
* Removes the cache for the specified key.
|
* Removes the cache for the specified key.
|
||||||
*/
|
*/
|
||||||
@ -73,7 +73,7 @@ declare module "ui/image-cache" {
|
|||||||
|
|
||||||
//@private
|
//@private
|
||||||
_downloadCore(request: DownloadRequest);
|
_downloadCore(request: DownloadRequest);
|
||||||
_onDownloadCompleted(key: string, result: imageSource.ImageSource);
|
_onDownloadCompleted(key: string, image: any);
|
||||||
//@endprivate
|
//@endprivate
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,15 +1,39 @@
|
|||||||
import common = require("ui/image-cache/image-cache-common");
|
import common = require("ui/image-cache/image-cache-common");
|
||||||
import imageSource = require("image-source");
|
import httpRequest = require("http/http-request");
|
||||||
|
|
||||||
module.exports.knownEvents = common.knownEvents;
|
module.exports.knownEvents = common.knownEvents;
|
||||||
|
|
||||||
export class Cache extends common.Cache {
|
export class Cache extends common.Cache {
|
||||||
|
private _cache: NSCache;
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
|
||||||
|
this._cache = new NSCache();
|
||||||
|
}
|
||||||
|
|
||||||
public _downloadCore(request: common.DownloadRequest) {
|
public _downloadCore(request: common.DownloadRequest) {
|
||||||
// TODO: WeakRef?
|
|
||||||
var that = this;
|
var that = this;
|
||||||
imageSource.fromUrl(request.url).
|
httpRequest.request({ url: request.url, method: "GET" })
|
||||||
then(function (value) {
|
.then(response => {
|
||||||
that._onDownloadCompleted(request.key, value);
|
var image = UIImage.imageWithData(response.content.raw);
|
||||||
|
that._onDownloadCompleted(request.key, image);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public get(key: string): any {
|
||||||
|
return this._cache.objectForKey(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
public set(key: string, image: any): void {
|
||||||
|
this._cache.setObjectForKey(image, key);
|
||||||
|
}
|
||||||
|
|
||||||
|
public remove(key: string): void {
|
||||||
|
this._cache.removeObjectForKey(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
public clear() {
|
||||||
|
this._cache.removeAllObjects();
|
||||||
|
}
|
||||||
}
|
}
|
Reference in New Issue
Block a user