From 6b65e03931e883eae2ed158947b4d36d00a370f1 Mon Sep 17 00:00:00 2001 From: Alan Date: Mon, 1 Apr 2024 05:26:53 +0000 Subject: [PATCH 01/14] Initial setup for a writing for the image gallery --- projects/gallery/index.php | 55 ++++++++++++++++++++++++++++++++++++++ 1 file changed, 55 insertions(+) create mode 100644 projects/gallery/index.php diff --git a/projects/gallery/index.php b/projects/gallery/index.php new file mode 100644 index 0000000..27f1bf7 --- /dev/null +++ b/projects/gallery/index.php @@ -0,0 +1,55 @@ + +
+
+
+
+
+

Preface

+
+

+ +

+
+
+
+

Javascript: Image Gallery

+
+

+ +

+
+
+

Concluding notes

+

+ +

+
+
+
+ +
+
+ + From 9f87c861ad3389a48b4f19ec4fc63cd31f5bc930 Mon Sep 17 00:00:00 2001 From: Alan Date: Mon, 1 Apr 2024 05:43:22 +0000 Subject: [PATCH 02/14] Placement of drafted text and markup. --- projects/gallery/index.php | 246 ++++++++++++++++++++++++++++++++++++- 1 file changed, 245 insertions(+), 1 deletion(-) diff --git a/projects/gallery/index.php b/projects/gallery/index.php index 27f1bf7..07633fd 100644 --- a/projects/gallery/index.php +++ b/projects/gallery/index.php @@ -27,7 +27,13 @@

Preface

- + The image gallery hosted within this website was self-made using pure Javascript. Prior to this implementation, the home page would send a visitor to my VSCO profile. There, they would be able to view every photo within my portfolio by navigating a gallery through the platform. This has recently changed, (noticed as-of February 2024), such that one needs to sign into the platform in order to navigate said gallery. +

+

+ I'm not a fan of being put into a silo like this. Especially considering the fact I was paying an optional subscription to use the service. Life has been filled with negative circumstances of which I have spun into something more positive. This is such a circumstance; I've taken the opportunity to develop something that fits more closely to the aesthetic of my personal site whilst also functionally being more seamless. +

+

+ This project page will discuss the differences between the behaviors of my gallery and the gallery hosted on VSCO. This will transition into discussion of implementation, followed by discussion of how to improve.


@@ -35,8 +41,246 @@

Javascript: Image Gallery

+ One thing that pushes VSCO above Instagram is its image gallery. As a web developer, it's not lost that the view of an image gallery within Instagram is fixed. That is, every image given in the view of a profile, (regardless of dimension), is fitted to a certain aspect ratio. It doesn't matter whether the uploaded image is a portrait or a landscape; when viewing the image in a profile's gallery, the image is fitted within a 1:1 frame. +

+
+ +
+

+ VSCO differs by maintaining an image's aspect ratio within a profile's gallery view. This allows an extra visual dimension that can help one profile feel apart from another. In the past, I've seen profiles of creators who create a sense of coherency through a deliberate choice of color palette within their images. This is something witnessed within both platforms. VSCO goes beyond by inviting deliberate usage of shape. +

+
+ +
+

+ Another facet that used to set VSCO apart from Instagram was the fact a visitor need not sign in to view the entirety of a profile. Unfortunately, this is no longer the case. Thus the ultimate motive of creating my own. +

+
+ +
+

+ In creating an image gallery, maintaining the aspect ratio of an image within the gallery view is a primary goal. There exists built-in functionality for CSS to help accomplish this. Images can be scaled by their aspect ratio to fit within a container. Furthermore, modern browsers provide functionality for grid support such that each column of a grid has the same length regardless of the size of its individual items. This is accomplished by providing padding for items that exist in a given row that have a lesser height than the item with the maximum height. +

+
+ +
+

+ An observer of the VSCO gallery will notice the margins between images are fixed regardless of the aspect ratio of an image. Due to the temporal prioritization of displaying images within the gallery, this poses a potential problem: images can be uploaded in such a manner that the height of one column can completely overwhelm another. This creates situations within the platform where a given column is blank while another introduces more images. This can be apparent looking at the end of an image gallery for a profile which has inconsistent image sizes. +

+
+ +
+

+ As indicated, this anomaly can occur whilst scrolling through the gallery. It seems this behavior is exacerbated by employing an algorithm which decides to load in new images based on having the bottom of the largest column reaching the bottom of the viewport. When this is triggered, a set amount of images is loaded for each column, which will lead to the gaps of the shorter columns being filled in. This creates situations where a user must scroll back up to view new images. Why the developers decided to use the column height of the largest column is beyond me. Ultimately this creates a subpar viewing experience within web browsers where more than two columns can exist. +

+

+ The image gallery I've developed improves upon the VSCO gallery by using the shortest column as a trigger for loading in new images. It also diverges from the temporal priority; It is more loose in terms of chronological ordering than the algorithm used by VSCO. Here, the a set of images are grabbed in chronological order, but are then sorted by size. This occurs such that the shortest image in a batch is placed in the column with the largest height, and the tallest image in a batch is placed within the shortest column. This approach helps keep columns balanced. +

+
+ +
+

+ Many organizations use lazy loading for image content. What this does is tell the browser to only fetch a resource once it is brought into the viewport. In the realm of near-endless scrolling, it would be too much to ask a visitor to download all the resources required in preparation of viewing a page. Thus, an image locator that is placed inside an image tag is only acted upon once the bounding box of the image is near the bottom of the viewport. +

+

+ It is not clear what is meant by "near the bottom" of a viewport. This value probably varies between web browsers and even a browsers' version. My experience using web services, (services which use lazy loading), has exposed what I like to call jagged loading artifacts for the individual elements involved. These artifacts are spurred by the fact it takes time to load a resource. There are two important facets of time here: the amount of time it takes to download the resource and the amount of time it takes the web browser to render the downloaded resource. +

+

+ The amount of time it takes to download a resource is represented by the empty space that acts as a place holder before anything is displayed. This space is very apparent on VSCO since their loading algorithm allocates the space required and fills it with a random color that fits their platform's color pallet. To allocate this space and then fill it with a color is a trivial operation from a browser's perspective, thus it is a task that completes quickly and is given priority over any other step of loading/rendering an image. This was a deliberate design choice on the platform's part. To fill these placeholder spaces with an appealing color, given the context of the platform, is a means to make the process appear more seamless*. +

+

+ The amount of time it takes the web browser to render the downloaded resource is represented by the partial fill of the actual image. How this is reflected is dependent on the usage environment and image format. Within my web browser, swaths of the image are rendered from the bottom-up. Typically, the bottom half is rendered before the first half. This usually takes less than a second, but can be noticeable. The half-rendering is why the term "jagged rendering" is used in this context. From the perspective of a user, a small span of time occurs, the first half of an image pops into view, then the next half pops into view. After an image is loaded, repeat the process for the next image to be loaded. The mental imagery I conjure for this process is a timeline describing when a new chunk of visual information is loaded in - a timeline which has a jagged saw-tooth like appearance. +

+
+ +
+

+ Obscuring the jaggedness of this loading procedure was another primary facet of this project; I wanted to substitute the partial on-screen loading behavior with a transition effect that is more smooth. Knowing this goal, a practitioner of CSS will take note of the term "transition". Such an intuition is accurate to what is happening. +

+

+ The implementation of the image gallery has two primary routines. One loads a set of images into an off-screen buffer. The images that are loaded into this buffer are initially set with an opacity value of 0. The other routine enables transition styling for the set of images that have been loaded into said buffer. This is done by adding a new set of CSS styles to an image that is in the buffer-zone where opacity is changed to to 1, (in addition to setting other transition effects). A class to which the image belongs contains transition effects within an external style sheet, which are in place after the document load and thus are active once the routine calls for a figure to be displayed. +

+
+ +
+

+ When exactly does this transition occur? "Near the bottom" of the viewport is defined explicitly within this implementation. Consider the following function that returns true should a figure reach the threshold to trigger a transition: +

+ + + +

+ This is a simple function - The point of transition is noted as being a ratio of the window's inner height. When the top of a figure reaches this point, the transition is applied. The application of this transition occurs within the routine in which we've been discussing - grid_display_agent. It is here that is_figure_bottom is applied. Before stepping into this function, it is important to understand a set of global variables. Consider the following declarations: +

+

+

+ grid_display_agent can now be examined. It receives one argument which is indicative of the amount of columns for a view. The function uses this value to retrieve a gallery from the grids global. The function then iterates through each column contained in the grid taken from grids and looks at each image contained in the buffer zone. If an image is found to trigger a display transition, set a flag indicating that more images should be placed into the buffer zone. Once each column for the given grid selection is examined, if the flag has been triggered, start loading images into the buffer zone by means of the routine that has yet to be discussed. +

+ + + +

+ Note the highlighted block above. This block-level declaration is essentially a lambda. Here, an attribute called 'loaded' is read on the figure. This attribute exists on the html element. Initially, it is set to false. The element also has a function bound to its onload event handler which will switch it to true - signifying that the entire image has been loaded within the web browser. Once this attribute is switched to true, the correct display properties are set such that it will transition into view. +

+

+ Below the highlighted block, near the end of the function, the setTimeout which triggers a call to the grid_load_agent acts as usage buffer to ensure the server isn't overwhelmed with resource requests. This scales with the size of the grid in question. It is within grid_load_agent that logic exists to start loading in the next batch of images into the buffer zone, which will then be primed in a manner that can be enacted upon by the grid_display_agent. +

+

+ The grid_load_agent is a bit more complicated than the display agent. This is where the placement logic of figures into columns exists. The logic is predicated on the fact that there exists another global data structure that acts as an image manifest. This manifest is a json structure which contains pruned data obtained from VSCO's CCPA-compliant information request. The data in the context of the gallery contains only information pertinent to an image. The schema is as follows: +

+ + + +

+ This data structure has a size; there is a quantity of images that need to be loaded in. Naturally, the logic of grid_load_agent begins by ensuring that a quantity of images that's beyond this maximum size isn't attempted to be retrieved. Whilst contemplating this and taking a look at the code itself, don't lose sight on the fact that the data structures at play operate on the fact that multiple column views exist for a given view-port. A screen of a certain size may be concerned with a view consisting of two columns. Another screen of a different size may be concerned with a view consisting of 4 columns. This is handled by the access of these data structures, such as load_counts. +

+

+ While that is in mind, note that the amount of images that may exist is a number that isn't easily divisible by a certain column count. Will all columns be filled with the same amount of images in a case where the column count is 4 and the total amount of images is some odd number? No. Thus, this base case needs to be considered before beginning the core logic of this subroutine. +

+ + + +

+ Before jumping into the core logic of grid_load_agent, the structure of the markup needs to be discussed. Inspection of the html that's in place before any javascript is executed will discover one html tag: [simple code of div]. When the html document is executed by the browser, a script tag will fill this tag based on the maximum amount of columns whomever is deploying the gallery wants in place. Ultimately, sub div containers will be placed into this tag which represents a grid of a certain column count. Contained within these sub-div tags are div tags that are representative of the individual columns. Contained within the columns are figure tags which also contain img tags. The figures are embedded in a clickable anchor tag. The logic for filling out the super-div tag is as follows: +

+ + +

+ What is missing from the above code snippet is a look at what constitutes the div representative of a given column. This logic is contained within a helper function. This helper function receives information from the manifest in addition to a grouping of styles. The nesting occurs here and the figure object is returned. It should be noted that this is where the logic is placed to allow an img tag to know when it is fully loaded. This is highlighted below:

+ + + +

+ Pivoting back to the core of the logic contained in grid_load_agent, it's important to remind ourselves that when grid_load_agent is called, it is supplied a single argument. That argument is grid_selection which is a mechanism to select a specific grid view of a certain column length - the length determined by the argument given for the parameter. Much of the machination in place selects the grid of the supplied column count, then iterates through these columns. The variable labeled boundary indicates the amount of images to be pulled in through the current call. It is bounded by the column size of the grid_selection. It can be lower than this by reasons previously discussed and shown by the set of conditions to get into the core set of logic. +

+ + + +

+ Recall the general logic of placing figures into columns. The image of lesser height needs to be paired with the column with the largest height. The image of greater height needs to be paired with the column of the smallest height. Image heights need to be known and considered in a certain order. Column heights need to be known and considered in a certain order. Let's start with the images. +

+

+ The image information is first retrieved from the manifest and then used to create a new figure. +

+ + + +

+ Recall that images can be placed within the markup environment and the browser will take liberty to adjust the size of the object. In this case, we are using display properties that force the maximum size of an image to conform to the size of a given column. Within the manifest, height and width values of the originating image are stored. These can be used to calculate the proportion. It is not enough to strictly use the height value. Two images may share the same height value, but one may be significantly wider than the other, forcing a resize within the context of an html column. This resizing will in turn change its rendered height value. Thus, the image needs to have some ratio be calculated: +

+ + + +

+ We now have an object representative of an image and a ratio representative of the amount of space it takes up within a column. These need to be placed somewhere on account of the fact this process is within an iterative structure. +

+

+ Outside of this iterative structure exists a set of declarations. Relevant to the problem at hand is the following: +

+

+ + + +

+ All that's left is to account for global info and to sort the list of keys. +

+ + + +

+ The same logic needs to applied to the grouping of columns being considered. The current height of the columns in place need to be considered and placed into a list. This list should be sorted in the opposite order of the figure_height_list so that its values may be popped in tandem with the height list of the figures for correct pairing. These data structures reflect the data structures with the figure prefix, and exist within the same scope and proximity. Noting these declarations, the familiar logic is as follows: +

+ + + +

+ With the maps in place, all that needs to be done is to iterate an amount of times equal to the amount of images being placed. For each iteration, pop a height key from both height lists. Use these keys to access their respective associative arrays and get the the relevant objects to make a figure-column pairing. +

+ + + +

+ Note that the value for iteration_index allows for a case where subsequent images can be placed into the same column. This occurs when a group of images whose combined height isn't great enough to cause a column's height to overwhelm the others. +

+

+ All these pieces are brought together to make the whole of the subroutine: +

+ + + +

+ Taking a step back, a higher look will expose a key relationship. This relationship is based on the existence of two separate abstractions which recursively interact with each other. One piece is the buffer zone in which new images are placed but hidden. The other piece is the complement where the images have been revealed. The transition from one abstraction to the other is whether or not one of their components (an image-figure) has reached the threshold described by isFigureBottom. This transition ceases once all available images have been placed from the buffer zone into its complement. +

+

+ Taking a higher look on top of these relationships, one can make the claim there exists a third abstraction - images which have not been placed in the buffer zone or its complement. A transition occurs from here into the buffer zone. The transitions cease here once there are no more images to be placed. The mutual recursion between grid_load_agent and grid_display_agent allows for continual placement of images into these abstract spaces. It is through this discussion that the base cases are also highlighted. Recursion stops when isFigureBottom produces a set of false values and when there are no more images to be placed into the buffer zone. +

+

+ The recursion involved needs to be initiated by something, though. What this is is fairly obvious - when the page finishes loading, kick off the process. The first thing this process needs to ask itself is, "how many columns should initially be displayed?" This question is answered through a subroutine called readjust_caller which sets a global variable called active_grid. This variable indicates how many columns should exist dependent on the width of the screen. This subroutine is also called within a simple event handler for the window on resize. +

+ + + +

+ It is within the call stack of the readjust_caller that a the display properties of document.getElementById('galleries')'s children is handled; the children which represent a grid view of a certain column quantity. For example, should the readjust_caller determine the width of the device's screen necessitate 3 columns, it hides the grids of column sizes 1, 2, and 4, and enables the display of the grid of column size 3. +

+

+ The result of the window's load event handler, shown above, will allow the image grid to display a set amount of images based on the height of the window. It will also place the images into each buffer zone. If one were to scroll, no more action would occur; isFigureBottom would not be called upon again. To address this, another event needs to be considered: when the screen scrolls! +

+ + +

Concluding notes

From a8723d87bf9ec97e0a11deabd362ef72f10f11b2 Mon Sep 17 00:00:00 2001 From: Alan Date: Mon, 1 Apr 2024 17:06:44 +0000 Subject: [PATCH 03/14] Added code snippes for explanations --- projects/gallery/index.php | 474 ++++++++++++++++++++++++++++++++++--- 1 file changed, 436 insertions(+), 38 deletions(-) diff --git a/projects/gallery/index.php b/projects/gallery/index.php index 07633fd..5057928 100644 --- a/projects/gallery/index.php +++ b/projects/gallery/index.php @@ -6,9 +6,9 @@ $canonical = 'https://alanmckay.blog/projects/gallery/'; -$title = 'Alan McKay | Project | Image Gallery'; +$title = 'Alan McKay | Project | Balanced Image Gallery'; -$meta['title'] = 'Alan McKay | Image Gallery'; +$meta['title'] = 'Alan McKay | Balanced Image Gallery'; $meta['description'] = ''; @@ -107,7 +107,17 @@ When exactly does this transition occur? "Near the bottom" of the viewport is defined explicitly within this implementation. Consider the following function that returns true should a figure reach the threshold to trigger a transition:

- +
+function isFigureBottom(fig_object){
+    var fig_height = fig_object.getBoundingClientRect().height;
+    var fig_top = fig_object.getBoundingClientRect().top;
+    if((window.innerHeight * .95)-fig_top > 0){
+        return true;
+    }else{
+        return false;
+    }
+}
+

This is a simple function - The point of transition is noted as being a ratio of the window's inner height. When the top of a figure reaches this point, the transition is applied. The application of this transition occurs within the routine in which we've been discussing - grid_display_agent. It is here that is_figure_bottom is applied. Before stepping into this function, it is important to understand a set of global variables. Consider the following declarations: @@ -117,16 +127,37 @@

  • The declaration region of this script includes logic which creates this nested div structure. It operates with respect to this global. To get a closer feel of this structure, one only need to consider the following block: -
      -
    • - - - -
    • -
+ +

+ +
+grids_html = document.getElementById('galleries');
+for(let i=0;i<max_column_size;i++){
+    const grid = document.createElement('div');
+    grid.setAttribute('class','image-gallery');
+    // TODO: Create a css class for the folowing properties:
+    grid.style['display'] = 'grid';
+    grid.style['grid-template-columns'] = 'repeat('+(i+1)+', minmax(0px,1fr))';
+    grid.style['align-items'] = 'start';
+    grid.style['height'] = '0px';
+    grid.style['overflow'] = 'scroll';
+    for(let j=0;j<=i;j++){
+        const column = document.createElement('div');
+        column.setAttribute('class','image-col');
+        column.style['display'] = 'grid';
+        column.style['grid-template-columns'] = 'minmax(0px,1fr)';
+        grid.appendChild(column);
+    }
+    grids_html.appendChild(grid);
+}
+delete grids_html;
+
+
+

+

  • var load_counts = []; - Contains a max_column_size amount of integer values which indicate how many images have been *loaded* for a given column. This number is incremented every time an image gets placed into the buffer zone.
  • @@ -138,14 +169,28 @@
    • col_maps[1][1] points to information of the 2nd column within a gallery consisting of 2 columns. col_maps[2][1] points to information of the 2nd column within a gallery consisting of 3 columns. The logic which primes this structure is also based on max_column_size. The code which does this is as follows: -
        -
      • - - - -
      • -
    • +
    + +
+

+ +
+for(let i=0;i<max_column_size;i++){
+    col_map = [];
+    for(let j=0;j<(i+1);j++){
+        col_map[j] = [];
+        col_map[j]['loaded'] = 0;
+        col_map[j]['displayed'] = 0;
+    }
+    col_maps.push(col_map);
+}
+
+
+

+

  • - var grids = document.getElementsBy­ClassName('image-gallery'); - an array of the html objects representative of a gallery of a certain column count. + var grids = document.getElementsBy­ClassName('image-gallery'); - an array of the html objects representative of a gallery of a certain column count. The value of this global variable is equivalent to document.getElementsBy­Id('galleries').children.
  • - grid_display_agent can now be examined. It receives one argument which is indicative of the amount of columns for a view. The function uses this value to retrieve a gallery from the grids global. The function then iterates through each column contained in the grid taken from grids and looks at each image contained in the buffer zone. If an image is found to trigger a display transition, set a flag indicating that more images should be placed into the buffer zone. Once each column for the given grid selection is examined, if the flag has been triggered, start loading images into the buffer zone by means of the routine that has yet to be discussed. + grid_display_agent can now be examined. It receives one argument which is indicative of the amount of columns for a view. The function uses this value to retrieve a gallery from the grids global. The function then iterates through each column contained in the grid taken from grids and looks at each image contained in the buffer zone. If an image is found to trigger a display transition, set a flag indicating that more images should be placed into the buffer zone. Once each column for the given grid selection is examined then, if the flag has been triggered, start loading images into the buffer zone by means of the routine that has yet to be discussed.

    -
    +
    +
     async function grid_display_agent(grid_selection){
         var load_flag = false;
         for(let i=0;i<grid_selection;i++){
    @@ -281,19 +288,21 @@ function isFigureBottom(fig_object){
             setTimeout(function(){grid_load_agent(grid_selection)},(grid_selection * 100));
         }
     }
    +
     

    - Note the highlighted block above. This block-level declaration is essentially a lambda. Here, an attribute called 'loaded' is read on the figure. This attribute exists with the html element. Initially, it is set to false. The element also has a function bound to its onload event handler which will switch it to true - signifying that the entire image has been loaded within the web browser. Once this attribute is switched to true, the correct display properties are set such that it will transition into view. + Note the highlighted block above. This block-level declaration is essentially a lambda. Here, an attribute called 'loaded' is read on the figure. This attribute exists with the html element. Initially, it is set to false. The element also has a function bound to its onload event handler which will switch it to true - signifying that the entire image has been loaded within the web browser. Once this attribute is switched to true, the correct display properties are set such that it will transition into view. This ensures that no jagged loading occurs.

    Below the highlighted block, near the end of the function, the setTimeout which triggers a call to the grid_load_agent acts as usage buffer to ensure the server isn't overwhelmed with resource requests. This scales with the size of the grid in question. It is within grid_load_agent that logic exists to start loading in the next batch of images into the buffer zone, which will then be primed in a manner that can be enacted upon by the grid_display_agent.

    - The grid_load_agent is a bit more complicated than the display agent. This is where the placement logic of figures into columns exists. The logic is predicated on the fact that there exists another global data structure that acts as an image manifest. This manifest is a json structure which contains pruned data obtained from VSCO's CCPA-compliant information request. The data in the context of the gallery contains only information pertinent to an image. The schema is as follows: + The grid_load_agent is a bit more complicated than the display agent. This is where the placement of figures into columns exists. The logic is predicated on the fact that there exists another global data structure that acts as an image manifest. This manifest is a json structure which contains pruned data obtained from VSCO's CCPA-compliant information request. The data in the context of the gallery contains only information pertinent to an image. The schema is as follows:

    -
    +
    +
     <images> ::= <image_id> <images> | ∅
     <image_id> ::= {
         "upload_date": <integer>,
    @@ -302,16 +311,18 @@ function isFigureBottom(fig_object){
         "file_name": <string>,
         "share_link": <string>,
     }
    +
     

    - This data structure has a size; there is a quantity of images that need to be loaded in. Naturally, the logic of grid_load_agent begins by ensuring that a quantity of images that's beyond this maximum size isn't attempted to be retrieved. Whilst contemplating this and taking a look at the code itself, don't lose sight on the fact that the data structures at play operate on the fact that multiple column views exist for a given view-port. A screen of a certain size may be concerned with a view consisting of two columns. Another screen of a different size may be concerned with a view consisting of 4 columns. This is handled by the access of these data structures, such as load_counts. + This data structure has a size; there is a quantity of images that need to be loaded in. Naturally, the logic of grid_load_agent begins by ensuring that a quantity of images that's beyond this maximum size isn't attempted to be retrieved. Whilst contemplating this and taking a look at the code itself, don't lose sight on the fact that the data structures at play operate on the fact that multiple column views exist different ranges of view-port sizes. A screen of a certain size may be concerned with a view consisting of two columns. Another screen of a different size may be concerned with a view consisting of 4 columns. This is handled through the access of these data structures, such as load_counts.

    While that is in mind, note that the amount of images that may exist is a number that isn't easily divisible by a certain column count. Will all columns be filled with the same amount of images in a case where the column count is 4 and the total amount of images is some odd number? No. Thus, this base case needs to be considered before beginning the core logic of this subroutine.

    -
    +
    +
     var load_count = load_counts[grid_selection-1];
     
     //Check whether we've run out of images to load:
    @@ -320,6 +331,7 @@ function isFigureBottom(fig_object){
         .
         .
     }
    +
     

    @@ -329,7 +341,8 @@ function isFigureBottom(fig_object){ What is missing from the code snippet related to filling "galleries" div is insight of what constitutes the div representative of a given column. This logic is contained within a helper function. This helper function receives information from the manifest in addition to a grouping of styles. The nesting occurs here and the figure object is returned. It should be noted that this is where the logic is placed to allow an img tag to know when it is fully loaded. This is highlighted below:

    -
    +
    +
     function create_new_figure(file_name,init_style,vsco_url,source_grid,image_number){
         var anchor = document.createElement('a');
         anchor.setAttribute('target','_blank');
    @@ -354,6 +367,7 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
         anchor.appendChild(figure);
         return anchor;
     }
    +
     
    @@ -361,7 +375,8 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe Pivoting back to the core of the logic contained in grid_load_agent, it's important to remind ourselves that when grid_load_agent is called, it is supplied a single argument. That argument is grid_selection which is a mechanism to select a specific grid view of a certain column length - the length determined by the argument given for the parameter. Much of the machination in place selects the grid of the supplied column count, then iterates through these columns. The variable labeled boundary indicates the amount of images to be pulled in through the current call. It is bounded by the column size of the grid_selection. It can be lower than this by reasons previously discussed and shown by the set of conditions to get into the core set of logic.

    -
    +
    +
     async function grid_load_agent(grid_selection){//grid_selection is not zero-based
         var load_count = load_counts[grid_selection-1];
     
    @@ -399,6 +414,7 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
     .
     .
     }
    +
     

    @@ -408,25 +424,28 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe The image information is first retrieved from the manifest and then used to create a new figure.

    -
    +
    +
     for(let i=0;i<boundary;i++){
         //Grab the next image from the manifest:
         var reference = manifest[load_count];
         var new_figure_data = {
    -        'object':create_new_figure(reference['webp_file'],{'border-top':'solid 25px white','opacity':0},reference['share_link'],grid_selection,load_count),
    +        'object':create_new_figure(reference['file_name'],{'border-top':'solid 25px white','opacity':0},reference['share_link'],grid_selection,load_count),
             'height': reference['height']
         };
         .
         .
         .
     }
    +
     

    Recall that images can be placed within the markup environment and the browser will take liberty to adjust the size of the object. In this case, we are using display properties that force the maximum size of an image to conform to the size of a given column. Within the manifest, height and width values of the originating image are stored. These can be used to calculate the proportion. It is not enough to strictly use the height value. Two images may share the same height value, but one may be significantly wider than the other, forcing a resize within the context of an html column. This resizing will in turn change its rendered height value. Thus, the image needs to have some ratio be calculated:

    -
    +
    +
     for(let i=0;i<boundary;i++){
         .
         .
    @@ -436,6 +455,7 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
         .
         .
     }
    +
     

    @@ -453,7 +473,8 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe

    -
    +
    +
     for(let i=0;i<boundary;i++){
         .
         .
    @@ -469,13 +490,15 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
         .
         .
     }
    +
     

    All that's left is to account for global info and to sort the list of keys, highlighted below.

    -
    +
    +
     for(let i=0;i<boundary;i++){
         .
         .
    @@ -503,7 +526,8 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
                             The same logic needs to applied to the grouping of columns being considered. The current height of the columns in place need to be considered and placed into a list. This list should be sorted in the opposite order of the figure_height_list so that its values may be popped in tandem with the height list of the figures for correct pairing. These data structures reflect the data structures with the figure prefix, and exist within the same scope and proximity. Noting these declarations, the familiar logic is as follows:
                         

    -
    +
    +
     var columns = grids[grid_selection-1].children;
     for(let i=0;i<grid_selection;i++){
         let column_height = columns[i].getBoundingClientRect().height
    @@ -520,13 +544,15 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
     col_h_list.sort(function(a,b){
         return a-b;
     });
    +
     

    With the maps in place, all that needs to be done is to iterate an amount of times equal to the amount of images being placed. For each iteration, pop a height key from both height lists. Use these keys to access their respective associative arrays and get the the relevant objects to make a figure-column pairing.

    -
    +
    +
     var iteration_index = 0;
     var figure_key = figure_height_list[iteration_index];
     var figure_group = figure_height_map[figure_key];
    @@ -539,22 +565,24 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
             col_index = col_h_map[col_key].pop();
             //Grab the next figure of the current height-tier:
             figure = figure_group[i];
    -        columns[col_index].appendChild(figure);
    +        columns[col_index].appendChild(figure);
             iteration_index += 1;
         }
         figure_key = figure_height_list[iteration_index];
         figure_group = figure_height_map[figure_key];
     }
    +
     

    - Note that the value for iteration_index allows for a case where subsequent images can be placed into the same column. This occurs when a group of images whose combined height isn't great enough to cause a column's height to overwhelm the others. + Note that the value for iteration_index allows for a case where subsequent images can be placed into the same column. This occurs when a group of images whose combined height isn't great enough to cause a column's height to overwhelm the others.

    All these pieces are brought together to make the whole of the subroutine:

    -
    +
    +
     async function grid_load_agent(grid_selection){//grid_selection is not zero-based
         var load_count = load_counts[grid_selection-1];
     
    @@ -652,6 +680,7 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
             }
         }
     }
    +
     

    @@ -664,7 +693,8 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe The recursion involved needs to be initiated by something, though. What this is is fairly obvious - when the page finishes loading, kick off the process. The first thing this process needs to ask itself is, "how many columns should initially be displayed?" This question is answered through a subroutine called readjust_caller which sets a global variable called active_grid. This variable indicates how many columns should exist dependent on the width of the screen. This subroutine is also called within a simple event handler for the window on resize.

    -
    +
    +
     var isMobile = window.matchMedia || window.msMatchMedia;
     isMobile = isMobile("(pointer:coarse)").matches;
     
    @@ -673,7 +703,7 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
     
     window.addEventListener('load', function () {
         initial_gallery_position = document.getElementById('galleries').getBoundingClientRect().top;
    -    readjust_caller();
    +    readjust_caller();
         grids[active_grid-1].style['overflow'] = 'inherit';
         grids[active_grid-1].style['height'] = 'inherit';
         grid_load_agent(active_grid);
    @@ -686,7 +716,7 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
     });
     
     window.onresize = function(){
    -    readjust_caller();
    +    readjust_caller();
         if(old_height < window.innerHeight){
             grid_display_agent(active_grid);
         }
    @@ -702,11 +732,12 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
                             The result of the window's load event handler, shown above, will allow the image grid to display a set amount of images based on the height of the window. It will also place the images into each buffer zone. If one were to scroll, no more action would occur; isFigureBottom would not be called upon again. To address this, another event needs to be considered: when the screen scrolls!
                         

    -
    +
    +
     window.onscroll = function(){
         if(display_counts[active_grid - 1] <= load_counts[active_grid - 1] - active_grid){
             setTimeout(function(){
    -            grid_display_agent(active_grid);
    +            grid_display_agent(active_grid);
                 let base = load_counts[active_grid - 1];
                 for(let i=1;i<=max_column_size;i++){
                     let is_not_active_column = i != active_grid;
    @@ -714,23 +745,24 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe
                     let is_multiple_of = (base % i) == 0;
                     if(is_not_active_column && has_less_loaded && is_multiple_of){
                         while(load_counts[i-1] <= base && load_counts[i-1] < manifest_size){
    -                        grid_load_agent(i);
    +                        grid_load_agent(i);
                         }
                     }
                 }
             }, (100 * active_grid));
         }
     }
    +
     

    The set of booleans that scaffold the conditional within this subroutine's loop really emphasizes on a common pattern that has not been explicitly discussed thus far. Recall that both grid_load_agent and grid_display_agent have a single parameter which represents a certain grid view. A single call will determine the transitional state for a grid of the size given as an argument and decide whether an image should move from one state of display to another. As a user scrolls through the gallery, it's obvious what is occurring for the current grid view. It is not obvious that the same decisions are being made for the grids which are out of view.

    - Thus, when a user scrolls while the active_grid is 3, (for example), checks need to occur whether or not images should be placed into the buffer zone for the other grids. + Thus, when a user scrolls while the active_grid is 3, (for example), checks need to be in place to determine whether or not images should be placed into the buffer zone for the other grids.

    - The mutually recursive structure in place will also determine whether these hidden grid views move a given figure from this zone as well. This is done by some css trickery. Instead of setting the display property of these grids to none, the height is instead set to zero and an overflow property is set to obscure a scroll-bar that may be displayed as a result. This allows the calculation of the height values of the columns in place in this hidden state - the calculations that occur within grid_load_agent. It also allows isFigureBottom to work in this context as well. + The mutually recursive structure in place will also determine whether these hidden grid views move a given figure from this zone as well. This is done by some css trickery. Instead of setting the display property of these grids to none, the height is instead set to zero and an overflow property is set to obscure a scroll-bar that may be displayed as a result. This allows the calculation of the height values of the columns within this hidden state - the calculations that occur within grid_load_agent. It also allows isFigureBottom to work in this context as well.

    Ultimately, this is a code smell. This code smell has the consequence of being affected by the whims of web browser should it decide to change rendering behavior. Future work on this script will remove the reliance of the calculation of a columns height via getBoundingClientRect() and instead populate a data structure with the height information calculated by the the new_figure_ratio within grid_load_agent. @@ -739,7 +771,7 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe A closer look at this code smell reveals the unintended behavior of the fact that grids of differing column lengths reveal images at differing rates. This is because isFigureBottom may flag true for images that are scaled differently on account of the amount of space afforded by a column. The consequence of this is when a user resizes the window, (and the resize presents a different grid-view), they may have to witness the transition effect of an image being brought into view again. This may give the false impression that the image has been reloaded into the browser and/or that may gaslight a user into thinking they have not viewed that image yet.

    - Taking a look at the deployment of the image gallery, as of date in which this writing is published (April 2024), will show that this is not an issue. Some patches have been made to the code which can be viewed via this site's github repository. + Taking a look at the deployment of the image gallery, (as of date in which this writing is published - April 2024), will show that this is not an issue. Some patches have been made to the code which can be viewed via this site's github repository.

    A closer look at the repository will also reveal additional code in place. What's not been discussed here is how to differentiate transition effects for images that are a part of the initial batch of images loaded in upon page view and the subsequent images that are loaded upon scroll. Ultimately this is a trivial adaptation, so it's encouraged the reader of this article investigate the repository's code or think of their own solution on the matter. diff --git a/style.css b/style.css index 999b7f3..756b993 100644 --- a/style.css +++ b/style.css @@ -366,6 +366,10 @@ code pre.info-code{ width:70vw; } +code pre.test-code{ + width:85vw; +} + .dynamic-figure-container{ align-items:center; } @@ -636,6 +640,10 @@ div.image-col figure{ width:65vw; } + code pre.test-code{ + width:85vw; + } + .dynamic-figure-container .dynamic-figure-view{ width:55%; } From e05285d7a6b10dec380ac62d87998d76a17a9f6c Mon Sep 17 00:00:00 2001 From: Alan Date: Tue, 2 Apr 2024 21:09:21 +0000 Subject: [PATCH 10/14] Added logic to hide the introduction based on query strings. --- photography/index.php | 12 +++++++++++- 1 file changed, 11 insertions(+), 1 deletion(-) diff --git a/photography/index.php b/photography/index.php index 4dd0691..9fb9452 100644 --- a/photography/index.php +++ b/photography/index.php @@ -22,7 +22,14 @@

    +

    On VSCO

    @@ -35,6 +42,9 @@


    +

    Photography

    @@ -55,7 +65,7 @@ for($i = 0; $i < min(($images_quantity - $initial_index),$image_count); $i++){ echo ""; echo "
    "; - echo ""; + echo ""; $image_index = $image_index + 1; echo "
    "; echo "
    "; From dca9eb4fbae208ee99df0835612c74376f32c952 Mon Sep 17 00:00:00 2001 From: Alan Date: Tue, 2 Apr 2024 21:16:35 +0000 Subject: [PATCH 11/14] Concluding text. --- projects/gallery/index.php | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/projects/gallery/index.php b/projects/gallery/index.php index d609cb0..dfd42bc 100644 --- a/projects/gallery/index.php +++ b/projects/gallery/index.php @@ -35,10 +35,13 @@

    This project page will discuss the differences between the behaviors of my gallery and the gallery hosted on VSCO. This will transition into discussion of implementation, followed by discussion of how to improve.

    +

    + A working implementation can be viewed here. +


    -

    Javascript: Image Gallery

    +

    Javascript: Balanced Image Gallery

    @@ -776,11 +779,17 @@ function create_new_figure(file_name,init_style,vsco_url,source_grid,image_numbe

    A closer look at the repository will also reveal additional code in place. What's not been discussed here is how to differentiate transition effects for images that are a part of the initial batch of images loaded in upon page view and the subsequent images that are loaded upon scroll. Ultimately this is a trivial adaptation, so it's encouraged the reader of this article investigate the repository's code or think of their own solution on the matter.

    +

    + Another side effect related to the flaws discussed so far is that a user with a slower internet speed will be required to commit to a larger amount of files upon the initial load. The reason of this is because it takes more time for the rendering environment to allocate space within each individual column. This allows for a greater amount of calls to isFigureBottom to pass, which in turn will prompt the script to load in more assets. This primarily becomes an issue when the time it takes to get a response to the server is measured around 1 second (1000ms) or higher. +

    +

    + This implementation works for my use-case. To fix these issues will require some code refactoring. Anyone who has been able to keep up to speed with the architecture and code discussed thus far will likely have seen places in which a good refactor can be helpful. These refactors, and any subsequent cleaning up of logical errors, will be the next step in terms of this project. The goal is to eventually have a package that can be used as a web component elsewhere. Any progress made on this front will be cataloged within the concluding notes section of this writing. Eventually a repository dedicated to this component will be produced along with documentation in terms of how to deploy it. +


    Concluding notes

    - + Check back later for updates. - Alan


    From 19aa093160d6e5996e97a160b7a8728b8a65fe0e Mon Sep 17 00:00:00 2001 From: Alan Date: Tue, 2 Apr 2024 21:19:37 +0000 Subject: [PATCH 12/14] Meta description added. --- projects/gallery/index.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/projects/gallery/index.php b/projects/gallery/index.php index dfd42bc..4927469 100644 --- a/projects/gallery/index.php +++ b/projects/gallery/index.php @@ -10,7 +10,7 @@ $meta['title'] = 'Alan McKay | Balanced Image Gallery'; -$meta['description'] = ''; +$meta['description'] = 'Project description detailing facets of interest pertaining to building a responsive and balanced image gallery as an alternative to options given through social media.'; $meta['url'] = 'https://alanmckay.blog/projects/gallery/'; From fdfea19740cf7bb94bfded2a5f1ac5e4094ed9e1 Mon Sep 17 00:00:00 2001 From: Alan Date: Tue, 2 Apr 2024 21:21:29 +0000 Subject: [PATCH 13/14] Added link to /gallery/ --- projects/index.php | 1 + 1 file changed, 1 insertion(+) diff --git a/projects/index.php b/projects/index.php index eeed959..9cda2bb 100644 --- a/projects/index.php +++ b/projects/index.php @@ -28,6 +28,7 @@ Icon for blog link Projects + Javascript: Balanced Image Gallery Research: Privacy and Dataflow Project: Social Computing Web Development: Aquatint Image Processor From c568a802167786ee967aa003732d91d56e833964 Mon Sep 17 00:00:00 2001 From: Alan Date: Tue, 2 Apr 2024 21:22:54 +0000 Subject: [PATCH 14/14] Added projects/gallery to sitemap. --- sitemap.xml | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/sitemap.xml b/sitemap.xml index 6abf62d..4a51566 100644 --- a/sitemap.xml +++ b/sitemap.xml @@ -125,7 +125,7 @@ - 2024-02-09 + 2024-04-02 @@ -209,5 +209,14 @@ 2024-02-09 + + + https://alanmckay.blog/projects/gallery/ + + + + 2024-04-02 + +