KEMBAR78
TMP 36 AC | PDF | Ajax (Programming) | Java Script
0% found this document useful (0 votes)
169 views53 pages

TMP 36 AC

This document provides a high-level design for redesigning freelancer profile pages on a website. It outlines the data objects and information to be displayed on each profile page. It also discusses the frontend implementation strategy, including using local storage and integrating the frontend code with the existing backend. Comments are provided on changes to the design and SEO requirements. A prototype was delivered to demonstrate how the frontend could integrate with the backend code.

Uploaded by

skrishatguru
Copyright
© Attribution Non-Commercial (BY-NC)
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as DOCX, PDF, TXT or read online on Scribd
0% found this document useful (0 votes)
169 views53 pages

TMP 36 AC

This document provides a high-level design for redesigning freelancer profile pages on a website. It outlines the data objects and information to be displayed on each profile page. It also discusses the frontend implementation strategy, including using local storage and integrating the frontend code with the existing backend. Comments are provided on changes to the design and SEO requirements. A prototype was delivered to demonstrate how the frontend could integrate with the backend code.

Uploaded by

skrishatguru
Copyright
© Attribution Non-Commercial (BY-NC)
We take content rights seriously. If you suspect this is your content, claim it here.
Available Formats
Download as DOCX, PDF, TXT or read online on Scribd
You are on page 1/ 53

Frontend Code Design with Backend Integration (Freelancer Profile Revamp)

High Level Design (HLD) Version 3.0

Developed for Guru.com June 18th, 2013 Developed by Automated Logical Software Pvt. Ltd.
Atlogys Technical Consulting R-8, Nehru Enclave New Delhi, India 110019

Strictly Confidential

Versions
Date 05/24/2013 06/02/2013 Author Ritika Garga Ritika Garga Change Summary Original Document Revised Display pages post changes in UI design, post getting SEO feedback, post getting new css/html from Greg. Incorporates portfolio Popup with history api and lazy loading. Added sections 4.4.2 , 4.7 4.16 Updated section 5 for Display Pages Removed Build Pages. This whole section should be part of LLD and will be moved there itself from next version. Updated sections 6.1 and 6.2 06/18/2013 Ritika Garga Comments from Shankar on June 18th + notes from conf calls from Guru on all Q&A. 1 2 Version

References
# Document Name Description High level design doc for frontend coding of the portfolio pages Updated BRS document from Guru team Author Atlogys Date Shared 03/08/2013 1 UI HLD.pdf

2 BRSProfilesBuildViewv1.02 3 SEO - URL Title Meta


Desc - 28may2013.xlsx

Guru

04/30/2013

SEO excel sheet SEO word document

Guru Guru

05/29/2013 05/21/2013

4 Guru Profile - SEO


Feedback 21.05.2013.docx

1. Introduction
Guru intends to revamp the entire freelancer profile by redoing the profile UI pages, the profile related business rules as well as the information and its hierarchy as displayed on the current freelancer profile. The new approach is very UI centric and requires a big frontend change in look, feel and technology. The document describes the overall architecture & high level design suggested for the frontend coding of this feature. In doing so, the document throws light on various modules and libraries involved, and their role in creating a user experience that is optimized for speed and data. Further the document discusses the integration strategy for how this frontend will get integrated with the guru backend.

2. Comments
1. This document should be read in reference with pdf document mentioned in item #1 of the References table above. The following sections from the reference document hold true and apply here as well: Goal Core Technologies Used Techniques Testing Plugins Used Settings for Production Only Server Integration 2. [DESIGN CHANGES] This document is based on the BRS document mentioned in item #2 of the references table above. Lately (as off the week of May 20th 2013) there has been a new profile UI design and there have been changes. This HLD is now incorporating the new UI design across all pages. The BRS for the new design is pending and there may be some more changes required post the new BRS these will get adjusted in an iterative manner after we get the BRS. The completion of this HLD is however not dependent on the receipt of the new BRS. 3. [SEO] There have also been some inputs from the SEO team in the last weeks of May. These changes got finalized last week. The HLD is now addressing the SEO requirements. The SEO affects the following: Json objects being sent to/from the page Static data being sent to the page URL structure

Page interlinking query params Dynamic meta data updates and data placeholders for keywords and descriptions Data fetched with ajax vs. data fetched on pageLoad

// A second almost working mock of the integration in prototypal stage has also been delivered which shows how this frontend code can be integrated with the guru backend via making changes on the guru presenter and services layer of codebase.

3. Data Objects (per Page) for all Profile Pages


Display Pages Page Freelancer OverView Page [Top Band Information ] Name of freelancer, Website, image for vendor type, vendor type, Location of freelancer, Background cover image, Screen icon, Profile Tagline, Yearly and All Time Income, Number of quotes in last 30 days and all time, Rating (feedback/yr), Number of Reviews, jobs awarded, % of 4-5 stars, % of 3 stars, % od 1-2 stars, About Us short, About Us Long, Running Since, How we Work, Member since date, last sign up date, Address, time zone, email address, contact number Top 9 skills, services, team-members, files and documents, tested skills, Data Items Employer Status of freelancer Fav or not Actions on Page Freelancer Update profile, add/edit services, copy to drafts/unpublish/delete services Reorder tiles for services Employer Add to favorites, contact freelancer (opens a new page)

feedback popup info with bars and scores, elsewhere online

Service Details Page

[Top Band Information ] // save as overview above Service Title, Description, Hourly Rate, Minimum Rate, Sharing Link, work collection tiles, skills, industries,reviews, freedback bars, rating bars

Status of freelancer Fav or not

Edit service, clone, unpublish , delete, reorder work collection tiles Goto update profile

Contact freelancer, make fav

Skills Page

Top 8 skills, Rest of skills, [Top Band Information ] // save as overview above

Status of freelancer fav or not

Update Profile

Add to favorites

Individual Skills Page (Page fetched in ajax so top bar is not reloaded)

Service tiles, work collection tiles, Number of Skills, Number of Work Collections [Top Band Information ] // save as overview above

Status of freelancer fav or not

Update Profile

Add to favorites

Feedback Page

[Top Band Information ] // save as overview above Overall Feedback Rating, Ratings breakdown, Feedback Breakdown, Reviews, Testimonials

Status of freelancer fav or not,

Update profile, Sort, manage feedback

Add to favorites, Sort

Work Collections Page

[filter by skills]?? - TBD Published work collections, draft work collections, [Top Band Information ] // save as overview above Included in X services (for each work collection) Note that this is for freelancer also now.

Update profile, reorder tiles, copy to drafts, unpublish, delete, Update Content, Change cover, Add services and Add skills

Add to favorites

Work collections Popup (fetched in ajax)

Total number of work collections, title, freelancer name All image types (all media), found in services, skills used, industries, admirers, comments, this is nice image [Top Band Information ] // save as overview above Share link, All image types (all media), found in services, skills used, industries, admirers, comments, this is nice image Freelancer status fav or not? Update profile, edit portfolio copy to drafts/unpublish/delete, edit, update content/cover, add services/skills,

Write comments, click on this is nice image

Work details popup (per portfolio)

Write comments, click on this is nice image, add to favorites

Build Pages (Freelancer only) Page Profile About Edit (Update Mode) Profile Picture - About Section, Cover Picture - About Section Screen name, Closest Add (from scratch) Template images (options) for bgd cover and screen icon, screen name, timezone options, country/city/state Actions on Page Save/update profile picture, save/update background picture, save changes at end of form, cancel changes, publish/unpublish skill

City, Website, Make My profile, I am, Tagline, Description, Bio Country, Street, City, State, Zip, Timezone, Timezone privacy, Phone, Phone privacy, Skype., AIM , Yahoo, iChat, Chat Privacy Published vs. unpublished skill tests, files, team members, connected networks Template images (options) for bgd cover and screen icon, screen name, timezone options, country/city/state options Add New Service Form Service name, description, skills used, list of work collections already linked to the service, list of all available work collections, rate per hour, minimum budget, selected category, list of industries Cover image for work collection, select category and complete list, selected industries and complete list, selected skills, setting for comments, list of all file types (media), title, services selected,

options, list of skill tests taken

tests, edit/delete files, update photo of team member, delete team member

List of work collections to be added to this service, list of categories, list of industries,

Create new service, save service as draft, cancel, remove work collection, add work collection

Work Build Canvas Page

List of services of the freelancer, list of industries, list of categories

Save as draft, publish, cancel

Testimonials

List of testimonials

Add, edit, delete

Build Pages (Freelancer Only) Page Edit Profile Services (shows service tiles) Edit Profile Work (shows work tiles) Details List of services for this freelancer Actions on Page Edit service, copy to drafts, unpublish, delete, reorder tiles Update content, add cover, add services, add skills, copy to drafts, unpublish, delete, reorder tiles Purchase membership, upgrade membership, renew account, stop/start recurring payments Add bids

List of all work collections for this freelancer

Membership

Current membership status, receipts,

Bids

Number of Bids (including membership + purchased), increased bids, date, all time bid usage, last 30 days usage, in average 30 days usage, in same month last year usage, Last 100 bids

Profile Stats

All Time Views, Profile Conversions for Contact me was clicked,Employer Sent Message, Employer Invited to Project, Personal Website Visited, Total Marketing Score + breakdown into CAR, CER and CRR, work collections overview, services overview

Display Pages (Employer only) Page Contact Form Page Add (from scratch) List of project names with ids Actions on Page Invite to quote, send message only

4. Frontend Implementation and Strategy


4.1 Important Points on Local Storage Local Storage is based on named key/value pairs. You store data based on a named key, then you can retrieve that data with the same key. The named key is a string. The data can be any type supported by JavaScript, including strings, Booleans, integers, or floats. However, the data is actually stored as a string. If you are storing and retrieving anything other than strings, you will need to use functions like parseInt() or parseFloat() to coerce your retrieved data into the expected JavaScript datatype. We are using basket.js to store and retrieve all our javascript files to be cached in the local storage in the format required. Some important points about local storage Local storage data is stored in the hard disk , for windows the path for these files is \Users\[UserName]\AppData\Local\ . Default storage capacity (2.5 MB per origin in Google Chrome; 5 MB per origin in Mozilla Firefox, and Opera; 10 MB in Internet Explorer). User can even choose "Unlimited storage" plan for a domain in some browsers in Opera.

4.2 Code Structure Hierarchy and Config Files The javascript in the code for individual pages is arranged/called in the following manner : main.js will be called on all the HTML pages, it is where basket is used to load the javascript files specified in the config files called along with main.js . There will be a config file for each HTML page, it checks for the javascript files called in the local storage and if present, uses the cached version. Otherwise the files are loaded from the server.

The config file contains all the javascript files to be loaded for an HTML page. These include basket.js and brucelist.js that aid us in the caching of the javascript files in the browsers local storage and help us modularize and handle rendering of HTML elements effectively.

The javascript files mentioned in column 3 of the table below are loaded from the server (if not already present in the cache) on page load. Page Overview Service Details Page Skills Page Individual Skills Page Config File Main Javascript File config_overview.js overview.js config_service_details.js service_details.js config_skills.js skills.js Comments

This section is rendered after receiving data using ajax on the click event of a skills box on the Skills Page. config_feedback.js config.js config.js config_profile.js config_details.js feedback.js mainView.js newProjectView.js profile.js detailsView.js

Feedback Page Work Collections Pagge Portfolio Build (work collection canvas) Profile Build Project Detail Page

The detailsView.js is a separate javascript file specific to the work collections popup appearing on clicking the image for corresponding to a work collection on the service tile. This javascript file is loaded only on the click event explained above.

The main.js will be called on all the HTML pages, it is where basket.js callback is used to load the javascript files specified in the configuration file called on the same HTML page. 4.3 Loading Data for components on page The information related to different data objects on each page is fetched from server in json. Then this json is loaded on to the DOM (html of page) either on page load (onpageLoad() JS call) or using ajax calls triggered on a user action/event. All the repeating html elements have been modularized through the use of json objects. These are provided to brucelist (client side javascript) which contains the javascript functions to render the final HTML. This is explained in section 4.4.1 below.

The non-repeating objects will obtain their content from the server using .net placeholders in the aspx page that the HTML page is converted to. This is explained in section 4.4.2 below. For all repeatable elements (json arrays) per page, we will also need backend to send the total number of items from the server. This is not being calculated on client side. This applies across all pages. Ex. 1 of 6 work collections, so we will get the number 6 from the backend.

The manner of implementation of the sections in all the pages has been explained below in section 5. 4.4 Linking with Backend Data Transfer Layer 4.4.1 Json arrays (Repeatable Data) The concept of the way data is fetched from the server and loaded into HTML is explained below taking the example of the comments section on the Project Details Page. a. The templatized HTML snippet for comments is given below. <script type="text/x-tmpl" id="commentsTemplate"> <li class="clearfix"> <a href="#" class="avatar"><img src="{{imgUrl}}" class="rounded"></a> <div class="info"> <a href="#" class="sn">{{usrName}}</a> <timestamp title="{{commentDate}}">"{{commentDate}}"</timestamp> <p>{{commentText}}</p> </div> </li> </script> {{ }} here is brucelist specific client side code. The data inside {{ }} is populated from brucelist variables after pageLoad using the JS onPageLoad() handler. These brucelist variables hold the json data received from the server. The list of json objects provides information corresponding to each comment in a separate object. The json object for each comment is basically a mapping of the keys specified by the content inside the double curly brackets ({{}}) and the values corresponding to each of them. b. The comments data to be used to render the final HTML is obtained through the code below that fetches the data from mockup,js if the useMock value in the config file is set to true. If not, the data is

fetched from the server through an ajax call to a service. The data fetched is stored in the variable result here. $.ajax({ type: "GET", url: Services/ProjectDetails.svc/GetComments // now a hosted service call contentType: "application/json", dataType: "json", data: { ProjectID: 121 } }).done(function (result) { // show comments on page by using template var commentsData = config.dev.useMock ? PM.mockup.comments : result; //checks for location of data to be obtained

c. The populated commentData then feeds the commensTemplate referenced in HTML above (section a). commentsList.init($("#commentsList").get(0), { template: $("#commentsTemplate").html(), data: commentsData }); }); The list of json objects corresponding to each comment and the comment template format are provided as parameters to the init function of brucelist which replaces the {{}} parts of the template with the values provided in the json objects for individual comments and renders the final HTML to the DOM. After pageLoad, when you do an inspect element on the HTML of the page, you will find generated HTML for N comments where N is the total number of entries in the json comments array. This will look like: <ul class="userList" id="commentsList"> <li class="clearfix">

<a href="#" class="avatar"><img src="http://behance.vo.llnwd.net/profiles5/210170/50x75f961eee5dcb421f7cbc7f26b12c097.jpg" class="rounded"></a> <div class="info"> <a href="#" class="sn">User 1</a> <timestamp title="09-Aug-2012 at 09:09 am/pm">"09Aug"</timestamp> <p>First comment !!</p> </div> </li> </ul> d. There is some json data that comes on page load. This will be sent to a JS variable. There will be one JS variable with all json arrays for that page. This will be filled using strategy in 4.4.2 below

4.4.2 .Net ASP Placeholders (for static data from server) A very powerful and flexible way to program sites in ASP.NET is to have placeholders that are populated dynamically accordingly to information received from the server. The Page Method is shown below:

<%@ Page Language="C#" %> <script runat="server"> private void Page_Load(object sender, System.EventArgs e) { //build placeholder for(int x = 0; x <= 10; x++) { Label title = new Label(); title.Text = "Item " + x.ToString(); title.ID = "Item" + x.ToString(); Area1.Controls.Add(title); //populating the area1 placeholder on Page Load Area1.Controls.Add(new LiteralControl("<br>")); } } </script> <html> <head> </head> <body> <asp:PlaceHolder id="Area1" runat="server"></asp:PlaceHolder></P> //Placeholder in the aspx page <P> </body> </html> <P>

4.5 Compatibility of brucelist and basket across browsers Brucelist.js has been written such that all the functions used are cross browser compatible and makes it safe to use of all the browsers including IE8 and IE9. Basket.js supports locally caching scripts in any browser with local storage capabilities. As far as internet explorer is concerned IE8+ browsers support local storage . 4.6 Main Plugins Used Drag and Drop plugin. FileReader API for upload. JCrop for cropping. Basket.js for caching in the local storage. Brucelist.js for handling of DOM elements . Scribd Javascript API. 4.7 Custom Controls and User Controls For the following places in the frontend UI, custom controls will be used: Various HTML input Validations on Profile build about form, on service form (for both client and server side validations on entered user data). Attach multiple files Uploading plugin Populating list of cities, countries, closest city (auto suggest) Getting list of skills (auto suggest) The corresponding auto complete UI for the auto suggest options above. In cases of IE8 and IE9,where we cannot use the data URI based crop method (which depends on canvas html5 element), we will use the existing crop functionality on the guru site.

The custom controls are pre-compiled binaries. Since these are pre-compiled, their javascript and css cannot be extracted and cached using basket. Note there are no custom controls being used on the portfolio build canvas page. In our frontend code, there are client side validations to be done using guru custom controls. The guru custom controls is code that executes only on the server side. Hence, in frontend code, we will write comments at all places which need to be replaced with custom controls. LLD explains how to change it to using custom controls. Actual code snippets have been shared in LLD. Once the integration is done, the comments will get replaced with actual custom control tags in the .aspx files. 4.8 Scribd Javascript API, Scribd Platform API and S3 << this part may change on further research >> a. Scribd has a platform api which is a server side api only. Server side calls need to be made by guru servers to scribd servers to store user uploaded files on the scribd servers.

http://www.scribd.com/developers/platform PS: the "my_user_id" param is of utmost importance to us. http://api.scribd.com/api?method=docs.getList&api_key=758buxouy289mh4mhh1uf&my_user_id=fluff y19 The docs.upload method of scribd will be used. The api_key here is doing the user_authentication part but we will use my_user_id parameter (as explained here http://www.scribd.com/developers/platform) for authentication across all guru users. For non image files, we invoke scribd and not filereader api. This means that the actual file like a PDF file is NOT converted into a data URI. It is passed to guru servers immediately on upload and from there to the scribd servers as an http request param. This means that the scribd must return with a successful file key and only post that the scribd frame with the file can be displayed to the user. The user will wait till such time. Please note that Request signing of scribd api calls is not needed by web apps. Guru will probably need to be a paid account.

b. Post this scribd returns a key to the guru server for this particular file. This key can be used to generate a URL which can be used to fetch the document directly from scribd on any browser. This URL can be embedded and displayed on a web page (inside a scribd user control) using the scribd javascript api. Custom functions provided by this API use the id of the scribd file provided and generate the HTML for the same to provide as a value to the corresponding key in the JSON object that is passed as one of the parameters to the brucelist callback. Embedding of the scribd document will be carried out using an ajax request that has been mocked as a pre-defined file that has already been embedded and can be changed later to receive information from the actual xhr response. var scribd_doc = scribd.Document.getDoc(2520449, 'key-1127428tb3rbejns9bhr');

The following files are going to scribd: 1. Non image flies added on portfolio build canvas 2. files added on profile build about form which are not images [ have to integrate with uploadify plugin]

The following files are going to s3:

1. Image files added on portfolio build canvas, including the cover image of the portfolio. 2. Greelancer profile image Bgd cover image 3. Team member images 4. images on profile build about us page : images in attach files S3 also has a server side api only. In browsers like IE8 and IE9 where we cant use fileReaderAPIl, the uploaded image files will only be visible to the user (even for preview) only after they have been sent to s3 via guru server. So the user will have to wait till such time in IE8 and IE9.

4.9 URL Structure Freelancer vs. Employer URLs

URLS & Page Titles


Page Type View Job Page URL Guru.com/jobs/<Job Title>/ID Title <Job Title> | Job on Guru Meta Description <Project Description> Notes Limit project description in meta description to 165 characters. Tagline + Introduction limited to 165 characters will be in the meta description Limit service description in meta description to 165 characters. Limit meta description to 165 characters

Profile Overview Page

Guru.com/freelancers/<Screen Name>

<Screen Name> | Freelancer on Guru

<About Us>

Profile Service Individual Page Profile Portfolio Overview Page Profile Portfolio Individual Page Profile - Skill Overview Page

Guru.com/service/<Service Title>/<Country>/<City>/ID

<Service Title> in <City>, <Country Code> by <Screen Name> Portfolio by <Screen Name> on Guru

<Service Description>

Guru.com/freelancers/<Screen Name>/portfolio

<Portfolio Title 1>, <Portfolio Title 2>, <Portfolio Description>

Guru.com/portfolio/<Portfolio Title>/<Industry>/ID

<Portfolio Title> for <Industry> by <Screen Name> on Guru

Limit portfolio description to 165 characters.

Guru.com/freelancers/<Screen Name>/skills

<Screen Name> Expert in <Skill 1>, <Skill 2>, <Skill3> <Skill> | Skill by <Screen Name> on Guru

Other skills include <Skill 4>, <Skill 5><Skill 8> <Service Title 1>, <Service Title 2>,

Limit meta description to 165 characters

Profile - Skill Individual Page

Guru.com/freelancers/<Screen Name>/skills/<Skill>

Limit meta description to 165 characters

Profile Feedback Page

Guru.com/freelancers/<Screen Name>/reviews

Reviews, Ratings for <Screen Name> on Guru

Rated <X> of 5 stars. <X> total reviews. <Testimonial 1>, <Testimonial 2>,

Limit meta description to 165 characters

4.10 Handling of Images The server side backend team must store images in different sizes depending on the user agent string and the height and width of the browser window, the server backend must send the appropriate image. As users re-size browsers, the images will only be auto-scaled. Only on a page refresh, will the correct sized window for that screen size be sent. There are the 4 image sizes we will be storing in the backend: 128x128, 48x28, 32x32 and 16x16 127X128 will be treated as the base image. The following are the images that will be re-sized for responsive layout. Profile icon Stored in 4 sizes but used in 2 sizes for profile purposes.
For the profile, when larger than 800px show the 128 version. Less than that show 48px version.

Bgd cover image of freelancer profile Stored in 4 sizes but used in 2 sizes for profile purposes.
Greater than 600px show the large version.

Cover image of each work collection - Stored in 4 sizes but used in 2 sizes for profile purposes.
The portfolio page only uses 1 image size and lets the browser scale it automatically. Elsewhere we have a need to show the smaller covers (like adding them to services in the build pages).

Small covers can be 200px X 150px for <600 px.

Images added on canvas page - Stored in 4 sizes but used in 2 sizes for profile purposes. Greater than 600px swap in the large versions of the image. The small versions can be about 60% size.

The size of the images is autoscaled with changes in the browser window dimensions. We will not be fetching images based on window size on the resize event. We only initially fetch the most appropriately sized image given the browser width or on a page refresh.

4.11 Handling of sharing buttons on the profile build page Clicking on any of the icon divs in the sharing link section opens up a popup with a textarea to enter the sharing link corresponding to the network. Along with the textarea will be two buttons to enable and

disable the link. Disabling a particular social network greys out the icon font corresponding to the section and on clicking of save removes the image(icon-font) corresponding to the social network on the overview page.

4.12 HTML Snippets templatized and stored in a separate file as JSON The templates.js file has re-usable html snippets for images appearing commonly across many pages like drag and drop 4 arrow image, skulls double pencil icon image, pencil edit icon etc. We will add more to this as needed. For images in custom controls, please note that the error handling etc. will be done using their dlls and hence those images are not re-usable HTML snippets that need to be stored as json. 4.13 Query Params The profile build and display pages will be executed with the login session of the freelancer or employer. Hence we will always know and have the id of the freelancer and the employer. These will continue to get passed in query params like they happen now with no additional change in that logic from a backend perspective. 4.14 Mimification and baking into one JS There is not much room or requirement for this. If at all, there are any plugin files in js which are always loaded together in everypage, we may combine and bake them into one JS file. Else the basket is taking care of all modules and js file caching. It is better for us to have more number of small files that get cached using basket rather than small number of large files.

4.15 Responsiveness of UI This is by default present using media queries and is working as expected in the HTML code base received from guru team. We will ensure we carry forward the same. 4.16 Lazy Loading implementation @rohan add anything here now if you feel this desc below is incomplete. There refers to the concept of loading a subset of data on the page to begin with, and then loading the remaining data in increments as user scrolls the page vertically. The remaining data is usually something which is below the fold on initial page load and is fetched from server on ajax. This way we: Do not frontload the page with a lot of data that the user may never scroll down to see We only fetch from the server in increments as the user scrolls the page The html is loaded for a particular section of the page only when the user reaches that section

For data which is lazily loaded, we will get the json data from the server via ajax calls in increments of N where N is the next number of data items to be loaded. The html for these N items will then be generated in order from top to bottom as the user scrolls. If the html contains img requests, this means that the user will see the effect of images loading in order from top to bottom. How it is being done 1. Page Load if (#publishedtiles<=limit && #drafttiles<=limit) ->load all to the page, disable lazy loading on client side . else load first 12 published tiles, lazy loading enabled . (we will need to find out the minimum number of tiles to show on the page if #publishedtiles<12 ) 2. Lazy load client side detection - we will carry out the detection on the last tile(be it published or draft) of the page and send the number of tiles on the published and draft section, the server accordingly sends the json for the next batch of tiles. 3. Once the last batch of tiles is sent, the server sends an a specific response in the ajax call according to which we inform prevent any more detection for lazy loading. 4.In case of operations like unpublish,publish etc, we would be carrying a check on whether the number of draft tiles equals 0 and that lazy loading mode is on, if both conditions are satisfied we will not be appending the tiles to the other section. Otherwise, the tiles are appended to the other section as planned earlier.

The lazy loading is happening on the following pages: 1. Portfolio single page view and Popup view The frontend code will load a desired number of items at a time.

2. Overview page for the services blocks. Here we fetch 10 services at a time. On scroll, as the user reaches the last service, we lazily load 10 more services. For purposes of SEO, on the initial first page request to the Overview page, we will fetch the complete data to render 10 service tiles. Plus we will also get from the server the URL and Title data only for all the remaining services. This will be rendered in basic html below the footer. As the user scrolls and we send the first ajax request, we will delete this data from below the footer and then re-fetch the entire services data for next 10 tiles. This will be lazily loaded below the first set of 10 tiles and so on. The rendering of data below the footer is only done once on initial page load so Google has all this html for indexing and so that Google has the URLs for all individual services page which it may crawl. 3. Individual Services Page for the list of work collections on that page Services page. ( there are no drafts here). Here we load in increments of 12. The SEO concept applies here also for work URLs and titles. 4. Individual skills page see services blocks and work tiles both. There are no drafts here. Here we lazily load 4 services at a time and 6 portfolios at a time. Please note that the SEO concept is not needed here. 5. Work tab on the portfolio tiles. Here we load 12 at a time. The SEO concept applies here as well. Note that the main skills page has no lazy loading. There is also No lazy loading on profile build page for work and services. We will show all the drafts and published tiles on main first page load only.

// Note we will keep the code flexible so we can easily tweak and adjust the number of items to be loaded in each increment for lazy loading across various pages. 4.17 Fallbacks for IE8 and IE9 1. The canvas element used to build portfolio is not supported in IE8/IE9. We will use a different fallback. 2. The FileReader API cannot be used. So for images being added during the portfolio build process or for images added for team members in profile build, these will be immediately sent to server. In case that user cancels the edit process or leaves the page, we must delete the images from both the guru and the S3 servers. 3. The crop image functionality will need to be separate and we cannot use the concept of data uris for cropping which only works with File Reader api Images. The existing Guru controls for cropping (which may be a custom control) will need to be used. 4. Css3 enhancements can be ignored in IE 8. CSS attributes such as border-radius, transition, boxshadow, text-shadow are being used but dont need to be supported. Fallback will be straight edges no transitions

5. Instead of using media queries for responsive layout, in case of IE8, we will have a separate IE stylesheet. There will be horizontal scroll bars. 6. Drag and Dropping files from desktop to browser will not work. Fallback will be using the buttons to add files 7. IE 8 may not be able to support rearrange of files and portfolio items by clicking and dragging.

5. Request/Response data per Page


For all repeatable elements (json arrays) per page, we do not need total number page to be sent from server. This gets calculated on client side across all pages. In all these display and build pages, the frontend must get the ID of the freelancer in a js variable. This JS will be used for all back and forth ajax calls from frontend to backend. 5.1 Overview page The overview page will be available under the same url for both freelancers and employers. It will follow the pattern of www.guru.com/freelancers/<Screen Name>. 5.1.1 Components of the page to be obtained on page load by use of JSON Objects (Employer and Freelancer Both): These json objects are passed onto brucelist and not directly loaded into html from .net placeholders. There is Javascript in frontend code that extracts the objects from brucelist variables and passes that to html. See {{ }} in html at various places. Skills boxes (top 9 boxes) {id:" ",name:" ",url:" ", number:} Published Service Tiles (we will be rendering 10 initially on page load and the rest on lazy loading) { id:"", title: "", titlehref:"", // cssClass has been removed, This is the json from the server and brucelist if it needs // cssClass can do that independently on client side.

desc: "", skillsset: [ skills1:, skill2:,

skill3:, .. .. ] "", thumbnail:"", //s3 link for first work collection in service tile. Here backend must send us the 1st work collection in this service. workcollectionURL: // may get removed as we do lazy loading coding. rateperhour:, ratemin:, rating:, numberofreviews: } Note that we dont need the total number o f published services as this can be calculated on client side. Note that we get the value for the total number of work collections for each service when the user actually clicks on the work collection and Not on initial page load of this Overview page. Draft Service Tiles The draft tiles are not lazy loaded but instead are loaded all at once after the published tiles have finished loading. In case there are less than 10 published service blocks, we will load all draft tiles on page load itself. Else an ajax call will be made to load all draft tiles after all published tiles have finished loading.

For a draft service, clicking on the image will take you to the service detail page in this case the html form for edit service with data about this service pre-filled. [ See the section on profile build services section below ] Clicking on the zoom icon will open the portfolio build popup of the work collection in that service.[ see the section on the portfolio build popup below. ]

{ id:"", title: "", titlehref:"", desc: "", skillsset: [ skills1:, skill2:,

skill3:, .. .. ] "", thumbnail:"", rateperhour:, ratemin:, rating:, numberofreviews: } Title and url of all the published service tiles for SEO {title:,url:} This information will be stored near the footer and removed from the DOM as soon as the user scrolls to the end of the page. <<It may get removed on the first ajax request itself or at end. To be decided.>> Elsewhere online { id:,link:,linktype: } Meet our team { id:"", image: "", //this will be a link for the image to be fetched on Amazon S3 name:"", resumeurl: "", title:"", } Files { id:, // This is the guru Id for this file html: "", // only added on client side, not coming from server. See the writeup below. // This is the embed html. cssClass:"", // css Class is not coming from server but being added in frontend only. } The html key in the array holds the value of the html to be displayed on clicking of the corresponding button below. The html is produced by the client side scribd API using the id of the scribd document provided by the server as a parameter. Similarly we will use the share link of a video to wrap the relevant html around it and provide it to the html key. For this purpose the format of the array of json objects we will be requiring from the server will be as given below [ {id:,type:scribd,fileid:}, // this is the s3 or scribd ID.

{id:,type:video,url:}, .. ] Tested Skills // Published skills only. { id:"", name: "", year:"", score: "", }

5.1.2 Components of the page to be obtained on page load by use of placeholders in the HTML code (Freelancer & Employer both ) : screen name (Also used for title - seo) Website Location of freelancer (Corellia) Background cover image Screen icon Profile Tagline Yearly and All Time Income,Number of jobs, Number of quotes in last 30 days Rating, Number of Reviews , percentage with 4-5 stars, percentage with 3 stars and percentage with 1-2 stars What we are about short description (this is the tagline) What we are about long description also used upto 165 chars in seo for meta description. Running Since How we Work

Components of the page to be obtained on page load by use of placeholders in the HTML code (Employer Only) : The following component is additional in the case of the employer view Add to Favourites data icon status(Active/Inactive)

5.1.3 Scenarios on the page requiring Ajax calls to the server to send data (Employer) : Clicking the Add to favourites button. $.ajax({ type: "POST", url: " Services/CommonService.svc/favourite ", //service to handle the ajax call

contentType: "application/json", dataType: "json", data: {favouritestatus:, freelancer_screenname: } //data to be sent via the POST request }).done(function (result) { //html change }); The employer id for the above will be extracted through server side checks on the Session info or whichever way is suitable. Here the employer must be logged in so we know their employer ID. The backend must store this in the database for that employer. In case the employer is not logged in, we take them to the login page. For freelancer, we always have their screen names only, so must send that.

5.1.4 Scenarios on the page requiring Ajax calls to the server to receive data (Employer and Freelancer) : Clicking the zoom icon on image on the service tiles. $.ajax({ type: "POST", url: " Services/CommonService.svc /workcollectionpopup ", //service to handle the ajax call contentType: "application/json", dataType: "json", data: {serviceid: } //data to be sent via the POST request and will fetch 1st WC in that service by default . }).done(function (result) { //popup initiation }); The service id send here is the id corresponding to the tile whose image was clicked. It automatically gives data for first work collection in that service and its id. The backend here must also send back the total number of work collections for this service so this number can be shown on the work collection popup. The data received on completion of the ajax call is listed here for reference BUT is explained in more detail in Project Detail Popup section below. { files : [ files data in json ], foundinservices : [foundinservices data in json ],

skills : [skills data in json ], industry : [industry data in json ], admirers : [admirers data in json ], comments : [comments data in json ], like : true, tagline: xyz, name : xyz, collectionnumber : 1, totalcollections : 2, } To see the JSON values for corresponding keys above, please look at the JSON objects defined for the Project Details Popup below. Note : The popup that appears after the data is loaded using ajax will appear under a separate dynamic URL through the use of the HTML5 History API. It is also important to note that IE8 and IE9 dont support changing of state using the history API and we will be redirecting to the individual portfolio page if the browser is IE9 or less. Clicking on the next/previous buttons on the Work Collections popup This fires an ajax call to receive data for the next work collection. The URL is again updated using the History API. C $.ajax({ type: "POST", url: "Services/CommonService.svc/ workcollectionpopup ", //service to handle ajax call contentType: "application/json", dataType: "json", data: {service_id: , currentWCid:,nextprevstate:} //data to be sent via the POST request }).done(function (result) { //html change }); // Note that the two functions above are defined in a CommonService.svc file so they can be re-used across pages.

Request to retrieve next batch of data via lazy loading. This fetches the next batch of 10 published tiles as required and renders the HTML for the new JSON entries via brucelist consequently appending it to the previously displayed tiles. The json array for the corresponding published services is also altered with the newly fetched json elements appended to the corresponding array, This is done using window.aspect ratio calculations with respect to browser height. We track when last tile is 10-15 px above the window bottom and then trigger the ajax request.

5.1.5 Scenarios on the page requiring Ajax calls to the server to send data (Freelancer Only): Clicking on Copy to drafts, Unpublish and Delete from the dropdown corresponding to individual service tiles. The 3 functions below are also needed on the individual services page. We can make them common later when we are doing code optimization. $.ajax({ type: "POST", url: "Services/OverviewService.svc/copytodraftsservice ", //service to handle ajax call contentType: "application/json", dataType: "json", data: {srcid: } //data to be sent via the POST request }).done(function (result) { //html change }); The srcid variable sent in the ajax call is the id of the service to be copied. On the server side the published service corresponding to this id is copied to the first location amongst draft services in the database. Backend must change its ordering also for layout of tiles the next time the page is loaded. So must update the their ordering array. The same copytodrafts service will be used for ajax calls made on the clicking of the corresponding options on the edit dropdown of both published and draft service tiles. $.ajax({ type: "POST", url: "Services/OverviewService.svc/unpublishservice ", //service to handle ajax call contentType: "application/json",

dataType: "json", data: {srcid: } //data to be sent via the POST request }).done(function (result) { //html change }); The srcid variable sent in the ajax call is the id of the service to be unpublished. On the server side the published service corresponding to this id is removed and copied to the first location amongst draft services in the database. Backend must change its ordering also for layout of tiles the next time the page is loaded. So must updatetheir ordering array. $.ajax({ type: "POST", url: "Services/OverviewService.svc/deleteservice ", //service to handle ajax call contentType: "application/json", dataType: "json", data: {srcid: } //data to be sent via the POST request }).done(function (result) { //html change }); The srcid variable sent in the ajax call is the id of the service to be deleted. On the server side the published service corresponding to this id is deleted. Backend must update the ordering array. Edit dropdown on draft services [ Are these 2 doing the same thing? ] Edit services/ Publish in drafts this will be a redirect to the profile build page. From there on pageload we will do another ajax call to open the services form for this service. On the user scrolling down the page, the response to the ajax call to retrieve more data will carry information in the following format [ Published service tiles : {json of published services }, ] @rohan - // add ajax call to fetch all draft tiles data at once Reordering of the services tiles. This only happens in overview pages and then in profile build page. However since this overview page has concept of lazy loading, So keeping this in overviewservice.svc. Can

be changed later. $.ajax({ type: "POST", url: " Services/OverviewService.svc/reorderpublishedservice ", //service to handle ajax call contentType: "application/json", dataType: "json", data: {srcElement:, dstElement: } //data to be sent via the POST request }).done(function (result) { //html change }); Assuming that only the logged in freelancer can do this. Note that the employer cannot do this. This will be executed inside the javascript callback when the variable for successful swap returns true. The srcElement and dstElement are the ids of the service elements whose positions have been swapped. These ids are used in the backend to swap the places of the corresponding services. In the backend we must keep an array to maintain the ordering of services per freelancer and also update that array on this call. Please also note that the drag and drop of tiles/blocks applies within the scope of publish and draft areas separately. This means that we cannot drag and drop from published to drafts and vice versa.

5.2 Service Details page The service details page will be available under the same url for both freelancers and employers. It will follow the pattern of www.guru.com/service/<Service Title>/<Country>/<City>/ID. 5.2.1 Components of the page to be obtained on page load by use of JSON Objects (Freelancer and Employer both): Work Collection Tiles [ this is lazily loaded 12 at a time ] { id:, image: "", title: "", cssClass: "", // These two are only being added in frontend after this json is received from the server cssId: "",

url:"", } Skills and Industries used (total 10) {id:"",name:" ",url:" "}, // skills and industries are combined now. Ratings Breakdown,Feedback Breakdown { name:"", score:"", barscore:"", // its a percentage value used for css } Reviews (On the services detail page, how many do we get all at once? Is there any pagination? { id:"", rating: "", review:"", photo:"", EmployerName:"", ProjectUrl:"", ProjectName:"", date:"", } // There is a Read More which will be handled client side to show full review. No server side call here. Title and url of all the work collection tiles for SEO {title:,url:} This information will be stored near the footer and removed from the DOM as soon as the user scrolls to the end of the page.

5.2.2 Components of the page to be obtained on page load by use of placeholders in the HTML code (Freelancer and Employer) : Freelancer screen name // Also used in page title for seo purposes Website Location of freelancer (Corellia) Background cover image Screen icon Yearly and All Time Income,Number of jobs, Number of quotes in last 30 days [ also showed under guru stats below ] Rating, Number of Reviews ,percentage with 4-5 stars, 3 stars and 1-2 stars

Service Title Description // also used under meta description. Hourly Rate, Minimum Rate Attach Link. Note that this is not for social networks, its a non-repeating link. City,Country code. // Needed for page title

Components of the page to be obtained on page load by use of placeholders in the HTML code (Employer Only) : The following component is additional in the case of the employer view: Add to Favourites data icon status(Active/Inactive)

Components on the page obtained (From Above) on page load to be used to populate the meta tags on the page by use of placeholders (Freelance and Employer) : Screen Name, Service Title, City and Country Code (to be used for the meta title) Service Description (to be used for the meta description that will be limited to 165 characters)

5.2.3 Scenarios on the page requiring Ajax calls to the server to send data (Freelancer) : Please note that there are no draft work tiles on this page. Also there are no edit options on the published work tiles for the freelancer view of this page. Clicking on Clone, from the dropdown corresponding to the individual service. $.ajax({ type: "POST", url: "Services/ServDetailService.svc/cloneservice ", //service to handle ajax call contentType: "application/json", dataType: "json", data: {srcid: } //data to be sent via the POST request }).done(function (result) { //html change }); Clicking on Unpublish and Delete from the dropdown corresponding to the individual service. $.ajax({ type: "POST", url: " Services/ServDetailService.svc /unpublishservice ", //service to handle ajax call contentType: "application/json", dataType: "json",

data: {srcid: } //data to be sent via the POST request }).done(function (result) { //html change }); $.ajax({ type: "POST", url: " Services/ServDetailService.svc /deleteservice ", //service to handle ajax call contentType: "application/json", dataType: "json", data: {srcid: } //data to be sent via the POST request }).done(function (result) { //html change }); Note These functions are also there on overview page so we may combine these into a common service file. Will hash out post lazy loading.

As explained above in overview page, these calls will require backend to update its ordering array.

Reordering of the work collection tiles $.ajax({ type: "POST", url: " Services/ServDetailService.svc /reorderworkcollection ", //service to handle ajax call contentType: "application/json", dataType: "json", data: {srcElement:, dstElement: } //data to be sent via the POST request }).done(function (result) { //html change }); // this function call is with respect to this service. On the work tab page, we also have this functionality but those are specific to freelancer These functions may get combined. Will optimize later. This will be executed inside the javascript callback when the variable for successful swap returns true.

5.2.5 Scenarios on the page requiring Ajax calls to the server to receive data (Freelancer and Employer) : Clicking of any of the work collection tiles This opens up the popup for the individual work collection. After completion of the ajax call that posts the id of the work collection clicked on. $.ajax({ type: "POST", url: " Services/ServDetailService.svc /workclick", //service to handle ajax call contentType: "application/json", dataType: "json", data: {workid: } //data to be sent via the POST request }).done(function (result) { //html change }); This can goto commonservice.svc just like Overview.aspx page calls this. But here we will use workcollection ID so need to add separate call for now but will optimize and combine later. This returns the data in JSON format for the repeating elements and in an appropriate format for the placeholders on the popup that appears. The repeating elements are as the same as those specified for the Project Details popup. (also explained in Overview.aspx above) The non-repeating elements include the work collection number and the total number of work collections. For the format of data received in the response, refer to section on Projects Detail Popup. Clicking on the next/previous buttons on the Work Collections popup. This fires an ajax call to receive data for the next work collection. For the format of data received in the response, refer to section on Project Details Popup. Also mentioned in POverview.aspx above.

5.3 Skills page The skills page will be available under the same url for both freelancers and employers. It will follow the pattern of www.guru.com /freelancers/<Screen Name>/skills. 5.3.1 Components of the page to be obtained on page load by use of JSON Objects : (Freelancer and Employer) Top 9 skills. {id:"",name:" ",number:" ",url:" "} Rest of the skills {id:"",name:" ",number:" ",url:" "}

// Note there is No lazy loading on this page. 5.3.2 Components of the page to be obtained on page load by use of placeholders in the HTML code (Freelancer and Employer): screen name // a;sp needed for seo page title Website Location of freelancer (Corellia) Background cover image Screen icon Yearly and All Time Income,Number of jobs, Number of quotes in last 30 days [ also showed under guru stats below ] Rating, Number of Reviews ,percentage with 4-5 stars, 3 stars and 1-2 stars

Components of the page to be obtained on page load by use of placeholders in the HTML code (Employer) : The following component is additional in the case of the employer view Add to Favourites data icon status(Active/Inactive)

Components on the page obtained above on page load to be used to populate the meta tags on the page by use of placeholders (Freelance and Employer) : The first,second and third skill names - These will be obtained from the array of json elements for the top 8 skills obtained on page load (to be used for the meta title) The rest of the skill names in the first 8 category - These will be obtained from the array of json elements for the top 8 skills obtained on page load (to be used for the meta description that will be limited to 165 characters)

5.3.3 Scenarios on the page requiring Ajax calls to the server to send data (Employer) : Clicking the Add to favourites button (refer to overview.aspx above)

5.4 Individual Skills Section The individual skills page will be available under the same url for both freelancers and employers. It will follow the pattern of www.guru.com / freelancers/<Screen Name>/skills/<Skill>. 5.4.1 Components of the page to be obtained on page load by use of JSON Objects : (Both employer and Freelancer) Services Tiles (4 tiles will be loaded initially and the rest using lazy loading) {

id:"", cssClass: "", // added on client side only title: "", titlehref:"", desc: "", skillsset: [ skills1:, skill2:, skill3:, .. .. ] "", thumbnail:"", workcollection url: // may be removed later post lazy loading code is written rateperhour:, ratemin:, rating:, numberofreviews: } Work Collection Tiles (6 tiles loaded initially and the rest using lazy loading) { id:"", image: "", title: "", cssClass: "", // added on client side only. url:"", numberofservices:"", // this is needed for both freelancers and employers } Title and url of all the services for this skill for SEO {title:,url:} Title and url for all the work collections for this skill for SEO {title:,url:} This information will be stored near the footer and removed from the DOM as soon as the user scrolls to the end of the page.

5.4.2 Components of the page to be obtained on page load by use of placeholders in the HTML code (Freelancer and Employer View) : screen name Website Location of freelancer (Corellia) Background cover image Screen icon Yearly and All Time Income,Number of jobs, Number of quotes in last 30 days [ Rating, Number of Reviews ,percentage with 4-5 stars, 3 stars and 1-2 stars

Total Number of services, Number of Work Collections Skill Name

Components of the page to be obtained on page load by use of placeholders in the HTML code (Employer) : The following component is additional in the case of the employer view Add to Favourites data icon status(Active/Inactive)

Components on the page obtained above on page load to be used to populate the meta tags on the page by use of placeholders (Freelance and Employer) : Skill Name, Screen Name (to be used for the meta title) Service titles - These will be obtained from the array of json elements for the services obtained on page load (to be used for the meta description that will be limited to 165 characters)

5.4.3 Scenarios on the page requiring Ajax calls to the server to receive data (Freelancer and Employer) : Request to retrieve next batch of data via lazy loading. This fetches the next batch of service/portfolio info as required and renders the HTML for the new JSON entries via brucelist consequently appending it to the previously displayed tiles. The json array for the corresponding published services is also altered with the newly fetched json elements appended to the corresponding array, On the user scrolling down the page, the response to the ajax call to retrieve more data will carry information in the following format [ service tiles : {json of services }, portfolio tiles : {json of work collections },

] Clicking on image in service tile to get work collection popup (same as overview page above) Clicking on work collection tiles to open popup same as individual services page above.

5.5. Feedback page The feedback page will be available under the same url for both freelancers and employers. It will follow the pattern of www.guru.com / freelancers/<Screen Name>/reviews.

5.5.1 Components of the page to be obtained on page load by use of JSON Objects : Ratings Breakdown, Feedback Breakdown { name:"", score:"", barscore:"", }

Reviews (need 10, rest is paginated on server side, fetched default on basis of most recent first ) { id:"", rating: "", review:"", photo:"", EmployerName:"", ServiceUrl:"", ServiceName:"", ProjectUrl:"", ProjectName:"", date:"", time: } Testimonials Please note that on this feedback page, there is no lazy loading for testimonials. We get all testimonials at once on page load. { id:"", cssClass:"", // added client side only photo:"", review:"", name:"", }

5.5.2 Components of the page to be obtained on page load by use of placeholders in the HTML code (Freelancer and Employer) : Screen Name of freelance Website Location of freelancer (Corellia) Background cover image Screen icon

Yearly and All Time Income,Number of jobs, Number of quotes in last 30 days [ also showed under guru stats below ] Rating, Number of Reviews ,percentage with 4-5 stars, 3 stars and 1-2 stars

Components of the page to be obtained on page load by use of placeholders in the HTML code (Employer) : The following component is additional in the case of the employer view Add to Favourites data icon status(Active/Inactive)

Components on the page obtained above on page load to be used to populate the meta tags on the page by use of placeholders (Freelance and Employer) : Screen Name (to be used for the meta title) Overall Feedback rating, total Number of Reviews and Testimonials (to be used for the meta description that will be limited to 165 characters) Note The testimonials will be obtained from the array of json elements for the same obtained on page load

5.5.3 Scenarios on the page requiring Ajax calls to the server to receive data (Employer and Freelancer both) : Please note that the Freelancer can also sort. Sorting for the reviews based on best rated, worst rated and most recent reviews Clicking on any one of the above sort options from the dropdown send an ajax call to the server giving the freelancer screen name and the sorting mode based on the element clicked. $.ajax({ type: "POST", url: " Services/FeedbackService.svc /sortreviews ", //service to handle ajax call contentType: "application/json", dataType: "json", data: {freelancer_screenname:,sortmode:, } //data to be sent via the POST request }).done(function (result) { //html change // includes change in the destination ids for the next and previous buttons });

Note : The sortmode key sent in the json will be 0,1 or 2 based on the sort type requested and the handler service provides reviews in the requested order on completion of the ajax call as is shown in section 5.3.6.5. Please note that if we sort on Page X, it automatically goes back to the first page. The response to this contains data in the following format for all the three cases.. { reviews : [ reviews data in json ], } Pagination of reviews using the next/prev buttons. $.ajax({ type: "POST", url: " Services/FeedbackService.svc /paginatereviews ", //service to handle ajax call contentType: "application/json", dataType: "json", data: {freelancer_screenname:,pageid: } //sort mode can be figured based on the variable for the same stored on the session that is updated according on the sorting parameter . Pageid is the destination page number. }).done(function (result) { //html change }); The data received will be in the following format { reviews : [ reviews data in json ], next10url: , prev10url: , } To see the JSON values for corresponding keys above, please look at the JSON objects defined for reviews above for the same HTML page. Note : The sorting mode according to which the records are to be retrieved for the previous and next 10 records is decided according to the value saved for the same in the session. If no value is stored in the corresponding session variable, reviews are loaded with the most recent ones first. Sorting order will be saved in session,Carrying out a search for the dropdown element and assigning the variable based on it on every iteration , session storage seems better

5.5.4 Scenarios on the page requiring Ajax calls to the server to send data (Employer Only) : Clicking the Add to favourites button(refer to Overview.aspx above)

5.6 Work Collections Page The work collections/portfolio page will be available under the same url for both freelancers and employers. It will follow the pattern of www.guru.com /freelancers/<Screen Name>/portfolio. Please note there is no feature of filtering the work collections via skills. 5.6.1 Components of the page to be obtained on page load by use of JSON Objects : (Freelancer and Employer) Published Work Collection Tiles (12 will be loaded initially and the rest via lazy loading) { id:"", image: "", title: "", url:"", numberofservices:"", // included in X number of services (for employer and freelancer both) } Title and url of all the portfolio tiles for SEO {title:,url:} This information will be stored near the footer and removed from the DOM as soon as the user scrolls to the end of the page. Draft Work Collection Tiles (There is no lazy loading for draft tiles. We load them all at once after the published tiles have finished loading.) { id:"", image: "", title: "", url:"", } 5.6.2 Components of the page to be obtained on page load by use of placeholders in the HTML code (Freelancer and Employer) : screen name

Website Location of freelancer (Corellia) Background cover image Screen icon Yearly and All Time Income,Number of jobs, Number of quotes in last 30 days [ also showed under guru stats below ] Rating, Number of Reviews ,percentage with 4-5 stars, 3 stars and 1-2 stars

Components of the page to be obtained on page load by use of placeholders in the HTML code (Employer) : The following component is additional in the case of the employer view Add to Favourites data icon status(Active/Inactive)

Components on the page obtained above on page load to be used to populate the meta tags on the page by use of placeholders (Freelance and Employer) : Screen Name (to be used for the meta title) Portfolio titles - These will be obtained from the array of json elements for the published portfolio tiles obtained on page load (to be used for the meta description that will be limited to 165 characters)

5.6.3 Scenarios on the page requiring Ajax calls to the server to send data (Employer) : Make Freelancer as favourite (Refer to Overview.aspx above) 5.6.4 Scenarios on the page requiring Ajax calls to the server to send data (Employer and Freelancer) : Reordering of the work collection tiles $.ajax({ type: "POST", url: " Services/ WorkService.svc /reorderworkcollection ", //service to handle ajax call contentType: "application/json", dataType: "json", data: {srcElement:, dstElement: } //data to be sent via the POST request }).done(function (result) { //html change }); Note that this function may be common with what is defined above but this optimization can be done later.

5.6.5 Scenarios on the page requiring Ajax calls to the server to send data (Freelancer) : Clicking on the Copy to drafts, Unpublish and Delete options from the dropdown corresponding to individual work collection tiles. $.ajax({ // these functions are also on individual project page so may use common service, type: "POST", url: " Services/WorkService.svc /copytodraftswork ", //service for published/draft work tiles to handle ajax call contentType: "application/json", dataType: "json", data: {workid: } //data to be sent via the POST request }).done(function (result) { //html change }); The workid variable sent in the ajax call is the id of the work collection to be copied. On the server side the published work collection corresponding to this id is copied to the first location amongst draft work collections in the database.

$.ajax({ type: "POST", url: " Services/WorkService.svc / unpublishwork ", //service for published work tiles to handle ajax call contentType: "application/json", dataType: "json", data: {workid: } //data to be sent via the POST request }).done(function (result) { //html change }); The workid variable sent in the ajax call is the id of the work collection to be unpublished. On the server side the published work collection corresponding to this id is deleted and copied to the first location amongst draft work collections in the database.

$.ajax({ type: "POST", url: " Services/WorkService.svc / deletework ", //service for published/draft work tiles to handle ajax call contentType: "application/json", dataType: "json", data: {workid: } //data to be sent via the POST request }).done(function (result) {

//html change }); The workid variable sent in the ajax call is the id of the work collection to be deleted. On the server side the published work collection corresponding to this id is deleted. // Please note that in three functions above, the backend must update its array ordering. 5.6.6 Scenarios on the page requiring Ajax calls to the server to receive data (Freelancer and Employer ) : Request to retrieve next batch of data via lazy loading. This fetches the next batch of 12 published tiles as required and renders the HTML for the new JSON entries via brucelist consequently appending it to the previously displayed tiles. The json array for the corresponding published and draft services is also altered with the newly fetched json elements appended to the corresponding array,

Clicking on Update Content, Change cover, Add services and Add skills on the edit dropdown corresponding to the published/draft work tiles makes an ajax call to server providing the id of the project corresponding to the project tile thats dropdown optio n as clicked. $.ajax({ type: "POST", url: " Services/WorkService.svc / updatecontent ", //service to handle ajax call for Update Content, Change cover, Add services and Add skills options contentType: "application/json", dataType: "json", data: {workid: } //data to be sent via the POST request }).done(function (result) { / /html change }); The ajax function will be the same for the rest with the same service name except that for the HTML response on completion of the ajax call. This will get the entire data of the portfolio via ajax and shows the portfolio build popup. The non-repeating elements on the popup given as placeholders are also populated by the data in appropriate format received in the response. The data received will be in the following format { media : [ media data in json ], //please look at the JSON data for the same in the Project Details Popup

name: , coverimage: , services: {1,2,3}, skills: {}, industry: {}, commentstatus: } To see the JSON values for corresponding keys above, please look at the JSON objects defined in Project Details Popup. On the user scrolling down the page, the response to the ajax call to retrieve more data will carry information in the following format [ Published work tiles : {json of published work tiles }, Draft work tiles : {json of draft work tiles }, ] Please note that here clicking on any of the published work tiles will take you to the individual portfolio page.

5.7 Project Details Popup The Project Details popup will be available under the same url generated via the History API for both freelancers and employers. It will follow the pattern of www.guru.com/portfolio/<Portfolio Title>/<Industry>/ID. It will load via ajax call on the base page but the URL of the base page will be updated to be the individual portfolio url of the work opening in the popup. 5.7.1 Components of the popup to be obtained on ajax call completion by use of JSON Objects (Freelancer and Employer) Please note that the freelancer can also write comments or click on this is nice image Various File Types var media = [ { id:"", // this is the guru id imgsrc: "http://img.gawkerassets.com/img/18390vv0b325xjpg/xlarge.jpg", this is s3 link imgalt:" http://img.gawkerassets.com/img/18390vv0b325xjpg/xlarge.jpg ",

imgcaption:"A car driving around. Image is centered if smaller.", type:"image", }, { id:"", url: "http://w.soundcloud.com/player/?url=http%3A%2F%2Fapi.soundcloud.com%2Ftracks% 2F87931&amp;auto_play=false&amp;show_artwork=true&amp;color=ff7700", caption:"My favorite song", type:"music", }, { id:"", // guru id url: "http://www.scribd.com/embeds/20940714/content?start_page=1&view_mode=slidesh ow&access_key=key-1cv8cixdjkrhhrwilyue", id:"doc_44676", caption:"My favorite song", type:"scribd", }, { id:"", url: "http://player.vimeo.com/video/58915714?portrait=0", caption:"lellalenanbvopial;kje akl;jew", type:"video", } ]; Found in Services, Skills Used and Industry {id:"",name:" ",url:" "} Admirers {id:"",img:" ",url:""} Comments { id:, imgUrl: ', usrName : '', commentDate : '', commentText : ' }

5.7.2 Components of the popup to be obtained on ajax call completion by use of placeholders in the HTML code (Freelancer and Employer) : Work Collection Number/Total Work Collections. Title of work collection. 5.7.3 Scenarios on the popup requiring Ajax calls to the server to send data (Freelancer and Employer) : Clicking on the this is nice image. $.ajax({ type: "POST", url: " Services/WorkDetailService.svc / thisisnice ", //service to handle ajax call contentType: "application/json", dataType: "json", data: { workcollectionid:,nicestatus: } //the nicestatus variable will be 1 or 0 depending on whether the project has initially been liked or not }).done(function (result) { //html change }); Add comment. $.ajax({ type: "POST", url: " Services/WorkDetailService.svc / addcomment ", //service to handle ajax call contentType: "application/json", dataType: "json", data: { id:"",usrName : '', commentDate : '', commentText : '' } //data to be sent via the POST request comment date will be needed. }).done(function (result) { //html change });

5.7.4 Scenarios on the popup requiring Ajax calls to the server to receive data (Employer and Freelancer) : Request to retrieve next batch of data via lazy loading. This fetches the next batch of 7 media blocks as required and renders the HTML for the new JSON entries via brucelist consequently appending it to the previously displayed blocks. The json array for the

corresponding media blocks is also altered with the newly fetched json elements appended to the corresponding array.

5.8 Project Details Page The Project Details page will be available under the same url for both freelancers and employers. It will follow the pattern of www.guru.com/portfolio/<Portfolio Title>/<Industry>/ID. 5.8.1 Components of the page to be obtained on page load by use of JSON Objects : Refer to Json array defined in project details popup above. 5.8.2 Components of the page to be obtained on page load by use of placeholders in the HTML code (Freelancer and Employer) : Screen Name Location Rating Yearly Income Share link. website Title of work collection. Current Work Collection number,Total number of work collections (Available only for freelancer) Url of the next and previous work collection (Available only for employer) Total number of comments Number of admirers left after showing 10 Industry Name Screen Name used for page seo purposes This is nice image status On or off? Portfolio description used for seo purposes

Components on the page obtained above on page load to be used to populate the meta tags on the page by use of placeholders (Freelance and Employer) : Portfolio Title,Industry,Screen Name (to be used for the meta title) Portfolio Description (to be used for the meta description that will be limited to 165 characters)

5.8.3 Scenarios on the page requiring Ajax calls to the server to send data (Employer) : Clicking the Add to favourites button(Refer to Overview.aspx above) 5.8.3 Scenarios on the page requiring Ajax calls to the server to send data (Employer and Freelancer) :

Clicking on the this is nice image(refer to ajax call added above) Add comment(added above)

Scenarios on the page requiring Ajax calls to the server to receive data (Freelancer and Employer) : Request to retrieve next batch of data via lazy loading. This fetches the next batch of 7 media blocks as required and renders the HTML for the new JSON entries via brucelist consequently appending it to the previously displayed blocks. The json array for the corresponding media blocks is also altered with the newly fetched json elements appended to the corresponding array,

This is done using window.aspect ratio calculations with respect to browser height. We track when last block is 10-15 px above the window bottom and then trigger the ajax request. NOTE Clicking on the update content,change cover,add services and add skills actions options for the dropdown on this page will redirect the user to the work collections url with parameters appended to the url specifying the work collection id and the option type. This correspondingly opens the work collections page with an ajax call being fired to fetch details of the work collection tab specified and correspondingly showing the user the part of the popup specific to the option the user had previously selected from the dropdown.

@rohan add the necessary calls if needed. copy to drafts and unpublish are 2 different things. Add calls for them.

6. Backend Integration Logics


6.1 Presenter Layer and WebServices (REST calls) Each and every .html page above will be converted into a .aspx page. This .aspx page will have the html structure for the page along with asp .net placeholders for all static data that is to be given to frontend on pageLoad. (This data is fetched from database). Furthermore, For each .aspx page, there will also be a file defined in the: /masketplace/services/ folder which will have all the rest WCF (web services) methods to handle all ajax calls for get and post data to and from that page.
This way the services will be hosted services onwww.guru.com/services/ and the URL for ajax calls in javascript will be www.guru.com/services/Work.svc/GetProjects OR www.guru.com/services/ProjectDetails.svc/GetComments etc so you can have multiple get rest calls and multiple post rest calls. [ refer to section 4.4.1 above ]

Sample Service Files:

ProjectDetails Page -> shows the single page for one work collection. The ProjectDetails.svc Service file may have methods like: [AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)] public class ProjectService : Interfaces.IProjectService { private readonly DataAccess.DataAccessLayer commentDL = new DataAccess.DataAccessLayer(); public List<Models.CommentData> GetComments(int ProjectID) { //get comments from database by data-access layer var comments = commentDL.GetComments(); return comments; } public Models.CommentData SaveComment(Models.CommentData comment) { var result = commentDL.SaveComment(comment); return result; } } } The Work Tab Work.svc service file may have methods like: [AspNetCompatibilityRequirements(RequirementsMode = AspNetCompatibilityRequirementsMode.Allowed)] public class WorkService : Interfaces.IWorkService { private readonly DataAccess.DataAccessLayer db = new DataAccess.DataAccessLayer(); public List<Models.ProjectData> GetProjects() { var projects = db.GetProjects(); return projects; }

}
These files will do the json serialization/de-serialization and will then make direct calls to the framework backend layer code in the guru framework. There are WCF libs for this serialization.

//get comments from database by data-access layer var comments = commentDL.GetComments(); //convert object into json string result = (new JavaScriptSerializer()).Serialize(comments); //return JSON of result context.Response.ContentType = "application/json"; context.Response.Write(result);

Page Methods For all static data that is loaded via .net placeholders on pageLoad, there will also be a .aspx.cs file created for each .aspx page. This file will have a onPageLoad() method which will populate the values of all asp .net placeholders. [ refer to section 4.4.2 above ] Example, overview.html becomes overview.aspx and then there is a /marketplace/services/overview file created which will have Get and Post handlers for all ajax calls as defined in section 5.1 above. There is also a overview.aspx.cs file created which has a onPageLoad() method that send all non-json data as defined in section 5.1 above. We will call the Page method in Page load section as we are using these only for Static data. public partial class ProjectDetails : Guru.Web.Pages.Page { protected void Page_Load(object sender, EventArgs e) { Constant constant = null; try

{ if (!IsPostBack) { constant = new Constant(); constant.FillDropDown(ref ddlServiceType); // just an example } } catch (Exception objException) { lblValidation.Text = Convert.ToString(objException).Trim(); } finally { } }

// To fill a drop down list with a enum public void FillDropDown(ref DropDownList ddlDroDown) { foreach (Int32 iValue in System.Enum.GetValues(typeof(ServiceType))) { ListItem li = new ListItem(); li.Value = iValue.ToString(); li.Text = Enum.GetName(typeof(ServiceType), iValue).Replace("_", " "); ddlDroDown.Items.Add(li); li.Selected = false; li = null;

} } }

6.2 Architecture

ASPX \ HTML Page

We will use html pages for view and aspx pages for build pages.

JS Library
Pass Json Objects and return Json Objects

Presenter Class
serialize and De serialize the Json Object and Call appropritae Web service, This will be invoke by Ajax hit or page load method

Controller Layer
This will take the approproiate Web method call from Predsenter and Pass on to the Services

Service Layer
This is Service Side Code and this can be host on same server or Another Server. This will call Business Logic layer

Business Layer
This layer is used to written all Business logic and communicate with Data access layer

Data Access Layer


This Layer is used to manage all the Database Curd operations. this will commuicate with database.

SQL Server Database

You might also like