2017-11-25 9 views
-2

APIからオブジェクトを取得するサービスがあります。そのオブジェクトの一部に画像のURLが含まれることがあります。バックエンドは現在、これらをスキャンし、get_file_contents()でそれらを処理し(PHPで)、それらをインラインデータに変換します。これは私のサーバーのスループットを大きく負荷しています。私がこれをやっているのは、後でオフラインになるようにイメージをキャッシュしたいからです。しかし、オブジェクトをレンダリングするために通常の角度を使うことができるようにしています。画像をサービスとしてロード

画像をホストしているサイトがクロスサイトリクエストをブロックしているため、ブラウザでJavascriptで$http.get()の処理を実行できません。私が考えていたのは、ブラウザに<IMG>という要素を作成することでした。サービスをロードした後にサービスを呼び出して、データを抽出してオブジェクトを処理できるようにしました。

APIワーカーがAPIオブジェクトをダウンロードする前に、いつでもアプリからgetを保存するようにサービスワーカーを制御することはできません。

私は、私のサイトからの取得をストアするためにサービスワーカーをやり直すことについて考えましたが、それは少し間違っているようでした。そして、どうにかうまくいくとは思っていませんでした。サービスワーカーをオフにするということは、サイト全体を2回ロードして完全にリフレッシュさせる必要があるということです。

誰かが私のサービスにブラウザ経由で画像データを取得する方法を教えてもらえますか?

+0

なぜ匿名のdownvote? :( –

答えて

-1

最初にCORSのサポートイメージホストが見つかった場合は、これが必要でない可能性があり、おそらく$httpコールを使用した可能性があります。

CORSをサポートするホスト(たとえば、Imgur)と同様に、ディレクティブ、サービス、およびコントローラが必要です。私もthis base64 canvas codeを使用しました。ここで

はjavascriptのコードです:

// Using this from https://stackoverflow.com/questions/934012/get-image-data-in-javascript 
function getBase64Image(img) { 
    // Create an empty canvas element 
    var canvas = document.createElement("canvas"); 
    canvas.width = img.width; 
    canvas.height = img.height; 

    // Copy the image contents to the canvas 
    var ctx = canvas.getContext("2d"); 
    ctx.drawImage(img, 0, 0); 

    // Get the data-URL formatted image 
    // Firefox supports PNG and JPEG. You could check img.src to 
    // guess the original format, but be aware the using "image/jpg" 
    // will re-encode the image. 
    var dataURL = canvas.toDataURL("image/png"); 

    return dataURL; 
    // return dataURL.replace(/^data:image\/(png|jpg);base64,/, ""); 
} 

// Used on the img tag to handle the DOM notification feeding into the service 
app.directive('notifyimgsvc', function() { 
    return {restrict : 'A', link : function(scope, element, attrs) { 
     element.bind('load', function() { 
      console.log('imgSvc::notify() image is loaded'); 
      console.log("imgSvc::notify(): " + this.src); 
      imgSvc.notifyLoad(this.src, getBase64Image(this)); 

     }); 
     element.bind('error', function() { 
      console.log('imgSvc::notify() image could not be loaded'); 
      console.log("imgSvc::notify(): " + this.src); 
     }); 
    }}; 
}); 

// A core service to handle the comms in both directions from requests to data 
app.service('imgSvc', [function(netSvc) { 
    imgSvc = this; // to avoid ambiguoity in some inner function calls 

    imgSvc.images = {}; // a cache of images 
    imgSvc.requests = []; // the requests and their callbacks 
    imgSvc.handlers = []; // handlers that will render images 

    console.log("imgSvc::init()"); 

    // Allows a controller to be notified of a request for an image and 
    // a callback to call when an image is added. There should only ever 
    // be one of these so an array is probaby not needed and any further 
    // requests should probably throw an error. 
    imgSvc.registerHandler = function(callback) { 
     console.log("imgSvc::registerHandler()"); 
     if (imgSvc.requests.length) { 
      // Already have image requests, so tell the new handler about them 
      for (var i in imgSvc.requests) { 
       callback(imgSvc.requests[i].url); 
      } 
     } 
     // Add the new handler to the stack 
     imgSvc.handlers.push(callback); 
    }; 

    // The usage function from your code, provide a callback to get notified 
    // of the data when it loads. 
    imgSvc.getImg = function(url, callback) { 
     console.log("imgSvc::getImg('" + url + "')"); 

     // If we have pre-cached it, send it back immediately. 
     if (imgSvc.images[url] != undefined) { 
      console.log("imgSvc::getImg('" + url + "'): Already have data for this one"); 
      callback(url, imgSvc.images[url]); 
      return; 
     } 

     // push an object into the request queue so we can process returned data later. 
     // Doing it this way als means you can have multiple requests before any data 
     // is returned and they all get notified easily just by looping through the array. 
     var obj = {"url" : url, "callback" : callback}; 
     if (imgSvc.handlers.length) { 
      console.log("imgSvc::getImg('" + url + "'): informing handler"); 
      for (var i in imgSvc.handlers) { 
       imgSvc.handlers[i](obj.url); 
      } 
     } 
     imgSvc.requests.push(obj); 
    }; 

    // Notification of a successful load (or fail if src == null). 
    imgSvc.notifyLoad = function(url, src) { 
     console.log("imgSvc.notifyLoad()"); 
     // Save the data to the cache so any further calls can be handled 
     // immediately without a request being created. 
     imgSvc.images[url] = src; 

     // Go though the requests list and call any callbacks that are registered. 
     if (imgSvc.requests.length) { 
      console.log("imgSvc.notifyLoadCallback('" + url + "'): scanning requests"); 
      for (var i = 0; i < imgSvc.requests.length; i++) { 
       if (imgSvc.requests[i].url == url) { 
        console.log("imgSvc.notifyLoadCallback('" + url + "'): found request"); 
        // found the request so remove it from the request list and call it 
        var req = imgSvc.requests.splice(i, 1)[0]; 
        i = i - 1; 
        console.log("imgSvc.notifyLoadCallback('" + url + "')"); 
        req.callback(url, src); 
       } else { 
        console.log("imgSvc.notifyLoadCallback('" + url + "'): skipping request for '" + imgSvc.requests[i].url + "'"); 
       } 
      } 
     } else { 
      console.log("imgSvc.notifyLoadCallback('" + url + "'): No requests present??"); 
     } 
    }; 

    // The notifiy fail is just a logging wrapper around the failure. 
    imgSvc.notifyFail = function(url) { 
     console.log("imgSvc.notifyFail()"); 
     imgSvc.notifyLoad(url, null); 
    }; 
}]); 

// A simple controller to handle the browser loading of images. 
// Could probably generate the HTML, but just doing simply here. 
app.controller('ImageSvcCtrl', ["$scope", function($scope) { 
    $scope.images = []; 
    console.log("imgSvcCtrl::init()"); 

    // Register this handler so as images are pushed to the service, 
    // this controller can render them using regular angular. 
    imgSvc.registerHandler(function(url) { 
     console.log("imgSvcCtrl::addUrlHandler('" + url + "')"); 
     // Only add it if we don't hqve it already. The caching in the 
     // service will handle multiple request for the same URL, and 
     // all associated malarkey 
     if ($scope.images.indexOf(url) == -1) { 
      $scope.images.push(url); 
     } 
    }); 

}]); 

あなたがこのために必要なHTMLは非常に簡単です:

<div data-ng-controller="ImageSvcCtrl" style="display:none;"> 
    <img data-ng-repeat="img in images" data-ng-src="{{img}}" alt="loading image" crossorigin="anonymous" notifyimgsvc /> 
</div> 

そして、あなたはこのようなあなたのコントローラ内で呼び出す:

var req_url = "https://i.imgur.com/lsRhmIp.jpg"; 
imgSvc.getImg(req_url, function(url, data) { 
    if(data) { 
     logger("MyCtrl.notify('" + url + "')"); 
    } else { 
     logger("MyCtrl.notifyFailed('" + url + "')"); 
    } 
}); 
関連する問題