The Wind Waker Randomizer dv_im

Link to Github archive
https://github.com/DualVission/archive-twwrando-dv_im
Repository archived 2022/7/30

Intro

One evening, I found myself in a stream with many community members I knew apart from the lead moderator and the streamer himself who was playing a custom branch of the Wind Waker Randomizer. With the assistance of j_im, I was able to translate my understanding of Python for Maya (PyMEL) to general scripting, as well as a better grasp of GitHub. Through this, I have learned much about cooperation, communication, and conceptualization through reading the preexisting scripts, the raw base game PPC assembly, and aid from others, like CrainWWR and tanjo3. This is how the project originally named v_im was born.

A Brief Overview of Game Randomizers

A randomizer is a class of game modification that changes some aspect of gameplay in an unexpected manner, often checked by a system known as logic. Action and adventure titles often get something known as an item location or check randomizer, in which collectables and their locations are shuffled among each other, creating a new and often unique gameplay experience through many of the same forms of engagement as metroidvanias or roguelikes. Logic, for this kind of randomizer, is a set of rules that define what collectibles are then required to reach a specific item location. For example, in The Legend of Zelda: Ocarina of Time, players must have a Deku shield and a Kokiri blade before they can access any location within the Great Deku Tree. Using logic, the randomizer would then decide nowhere within the Great Deku Tree is a valid location for either the Deku Shield or Kokiri blade but would be for the Fairy ocarina or any other number of items.

Kindling

DGod63 is a Twitch streamer known for playing randomizers of The Legend of Zelda franchise, who I found one night on which sometime in late May 2020 because of lost hours during the early SARS-2 pandemic. However, he was running the randomizer from source, allowing him to make additional modifications which in the past were denied by the sole developer of the randomizer. The modifications to the code enabled him to change the required number of dungeons for race mode, a setting which made all non-required dungeons illogical, effectively empty in the eyes of the player, locations and made required dungeons end in a hard-required item, an item needed to beat the game. At the then-current official 1.8.0, players either played without race mode or could only have 4 dungeons, not upwards of the 6 possible dungeons in the game or fewer.

Also around this time, another popular randomizer was transitioning its community season from 3 to 4, changing its hint policies making them less helpful. In the next official update of Ocarina of Time Randomizer, hints could only be set to this more conservative option, forcing players to either use an older build or use this new policy. A forked build, known as the Roman Build or Roman Branch, rapidly gained popularity as it allowed for either hint policy without disabling other newer features. The Roman Build acted, up to this point, as a testing ground for features that were highly requested, but that the main developers didn’t want to provide. Often the course of action was a feature being added to the Roman Build, it would be brought to the main branch, then the Roman Build would switch to the more concise version created by the main developers.

A Spark of Energy

I sought answers as to how I could also do variable race mode in the Wind Waker Randomizer, I had some knowledge of Python so it wouldn’t be too terrible, right? When I messaged DGod, his answer was simple, having only edited one number in the code as instructed by j_im, the community moderator. I followed the trail and j_im told me much the same, edit this number on this file to the desired amount. I, while now having my answer, was dissatisfied with this solution, as it would not provide easy access to anyone who wouldn’t or couldn’t run from source. I began testing and playing with a piece of software attempting to edit the Wind Waker Randomizer’s UI. But I began to realize why LagoLunatic, its dev, omitted this option, at least up to 6 dungeons; the higher number of required dungeons increased the number of hard required items that would be called up. In cases where players start with the full Triforce of Courage and were playing swordless, that would only leave the three progressive bows and the hookshot to be pulled instead. 4 items for 4 dungeons. This list would need to be extended to encapsulate the 2 additional requirements. However, in side effect of increasing the number of required dungeons, the number of hard required items also increased. As such, the power bracelets and iron boots were also added to the list, as they each only locked Earth and Wind Temple respectively.

I was coming to finalize this rough idea of a small update to 1.8.0, but I felt lacking a name, would be forced to be associated with my name. Another concern I had was competing with Lago’s main branch. It would effectively create a self-cannibalizing market between the two branches, which is not what I would want. I simply wanted to push Lago to add this feature to the randomizer. Without much thought, the name v_im was selected, a basic portmanteau of j_im’s and my screennames. Between d_im and v_im, the answer was fairly obvious, and I felt it best represented the amount of work that each party had made towards this idea. Before I released this on the world, I checked to make sure I wasn’t saying something terribly offensive, only to find that “vim and vigor” was a turn of phrase. Vim means with, of, or pertaining to energy. However, I was not prepared for what would happen.

An Open Flame

DGod was not much larger than other streams I was watching at the time, but as the pandemic continued, his audience grew. A small project intended to make a few peoples’ lives easier was now on display for a large audience. A direct link at the command input of “!v_im”. As the idea that there was more to the Wind Waker Randomizer, more people began airing ideas they felt was missing or lacking. I had not completed a small task but checked a small box on a much larger list. It was known I was effectively unemployed and capable of basic level programming/scripting. Despite what I had thought originally, it was a good feeling, helping bring joy to people. A new version but not many changes in retrospect, it was becoming more clear v_im was becoming more the community and me and less j_im. It was a great name but it was lacking, at least in terms of representing my brand. The answer was a simple rebranding.

An additional “d” was prefixed to the existing name, it would prevent confusion as v_im is just dv_im prior to 1.8.0d. This naming schema matched my other scripting projects, at least on an internal naming level. All my script files up to this point were named “dv_” followed by its function (eg DV Artistic Material Generator is dv_artistic_material_creator). With a subtle name change, also came an imagery change. The use of Hyrule Warriors: Definitive Edition’s Sail of Red Lions might have been a bit vain, effectively treating dv_im as a level up to its source branch which uses the Swift Sail, which acts as the level of weapon proceeding the Sail of Red Lions. However, the sail itself I found hard to read at lower resolutions like in-game, so a question mark was used in place of its imagery.

Playing with Fire

Luckily, I was kept grounded by my shortcomings. I am not a programmer by trade or study, errors and oversights were not an uncommon sight. That still didn’t stop the steam of desired features, one I had been longing for some time. The unedited the Wind Waker Randomizer logic is what can be called a glitchless or trickless logic set. Every expectation it had of the player was that in base game. Players would never be expected to store the item-get cutscene hitbox to sneak around an area. However, all prior attempts of glitched logic came with one tradeoff, they only would exist in place of standard logic. I needed to find a solution that would leave regular logic usable. The UI would be easy, but the backend would be difficult. I found two points where logic could best be edited before it was loaded and towards the end of loading. I decided to go for the latter, referring to it as “logic injection” as it would interrupt and replace some bits of data, requiring the least editing and redundant information for anyone creating a custom logic type.

This caught the attention of community member CrainWWR, who was a the Wind Waker HD speedrunner. The logic sets I had implemented all were fairly basic, representing a scale from standard to what most randomizer players do already. However, Crain had a different idea, what is the fewest required items to get to any particular check. Sure, no logic existed, where there were no checks, but what if there was a checking system, but it only ensured the seed was beatable, but just barely. It was a bad idea due to high risks of bottlenecking, but my primary drive was player options. I was going to play this, but someone might. Though this may have been what brought Crain onboard, he is still a large part of the team and a heavy hitter for dv_im coding.

Throwing Shadows Forward

This project naturally spun to be much bigger than the sum of its parts, dv_im still represents somewhat of a success story. In my eyes, it is still not finished. Going forward, there are a few things Crain, Tanjo, and I have in mind: alternative game modes, alternative objectives, more checks, retooling, and player choice. The next project I am looking at is rebuilding entrance randomization, increasing a player’s ability to choose which locations are included or omitted. As is now, players can only choose to do all cave entrances, dungeons, both, all together, or none. While this is fine and dandy, this means a player would be expected to visit Star Island even if they have combat caves disabled, which is odd in comparison to most other things in the randomizer.

As a long-term goal, Crain was hoping to implement “pot rando”, a sort of alternative to the current boat navigation system. This system is far from even usable; however, we have created a system to create stable branching graphs that will be the core of this. Crain designed it first in Java, I recreated it in Python, and Tanjo optimized and improved it.

dv_im exclusive features:

  • Starting Conditions:
    • Dungeon Mode
    • Editable base health
    • Hints
    • Randomizable dungeon number
    • Randomizable starting Triforce number
    • Randomizable starting items
    • Text kerning
  • Randomization:
    • Additional boss rewards
    • Editable item pools
    • Logic Types (including custom)
    • Not Logic
    • Randomizable Wind Waker, Ballad of the Gales, Wind’s Requiem, Song of Passing, and Swift Sail
    • Remove dungeon nonprogressive items
    • Removable health increase items
    • 28 Additional Checks
  • Customization:
    • Animation editing
    • Disable parts of model items
    • Editable spoiler log output
    • Sail choice

DV Artistic Material Generator v1.3.2

Latest Release
Help Page
Text Defined Format Help Page

This is a script for Maya 2017+ for use with Solid Angle Arnold 5.1+, Pixar RenderMan 22.2+, and more for Maya.

This script functions with Maya 2016, Arnold 4, RenderMan 21 and other renderers; however, complete automation is not guaranteed.

Download the current version of the script.
Download the text defined format reference sheet.
Download the archive ZIP with both.

This script allows users to generate and populate popularly used channels on physically based renderers materials. With its UI, users can select file location as well as export for multiple renderers. With text defined format, advanced users can also connect to their own renderers.

This is for artistic metallic, dielectric, and glass objects: hard opaque plastics, soft rubbers, clothing, cutlery, skin, wood, bottles, etc.

Change Log
  • Fixed ramp luminance issue for spcColor
  • Fixed error dialogue for file pathing
  • Adjusted UI widths
  • Added bdBase attribute type
  • Added Python core “os”
  • Added TDF
  • Added UI associated with TDF
  • Changed attribute list formating for TDF
  • Small changes for sssMock future-proofing

Free Maya Rig – Ageo – Anthropomorphic Alligator

This is an asset for non-commercial-use with Maya 2017.2+ with textures for PBR including RenderMan or Arnold.

This asset may function with earlier versions of Maya, but functionality is not guaranteed.

Download the current version of the asset from OneDrive.

Known issues include leg popping on some animation resolution solutions, this varies between devices and some instances.

This asset is only for use in non-commercial space, such as portfolio work or education.

Alligator/Ageo is a creation of RAKUROBIT and modeled by Seanna Alanes.
Textures by Seanna Alanes and Zachary Yarnot.

DV Artistic Material Generator v1.3.0

Latest Release
Help Page
Text Defined Format Help Page

This is a script for Maya 2017+ for use with Solid Angle Arnold 5.1+ and Pixar RenderMan 22.2+ for Maya.

This script functions with Maya 2016, Arnold 4, and RenderMan 21; however, complete automation is not guaranteed.

Download the current version of the script.
Download the text defined format reference sheet.
Download the archive ZIP with both.

Download this version of the script.

This script allows users to generate and populate popularly used channels on physically based renderers materials. With its UI, users can select file location as well as export for multiple renderers.

This is for artistic metallic, dielectric, and glass objects: hard opaque plastics, soft rubbers, clothing, cutlery, skin, wood, bottles, etc.

DV Glass PxrSurface Material Generator v1.0.0

This is a script for Maya 2017 and Maya 2018 for use with RenderMan for Maya 22.0 onward.
This script functions with RenderMan for Maya 21.0 onward and Maya 2016; however, complete automation is not guaranteed.

Help Page

Download script here

This script allows users to generate and populate popularly used channels on PxrSurface or PxrLayer materials. With its UI, users can select file location as well as advanced options, such as color correction model types and normal map direction.

This is for artistic clear or translucent objects: glass cups, plastic bottles, eye surfaces, etc.

DV Artistic PxrSurface Material Generator Legacy v1.2.2

This is a script for Maya 2017 and Maya 2018 for use with RenderMan for Maya 22.0 onward.
This script functions with RenderMan for Maya 21.0 onward and Maya 2016; however, complete automation is not guaranteed.

Help Page

Download script here

This script allows users to generate and populate popularly used channels on PxrSurface or PxrLayer materials. With its UI, users can select file location as well as advanced options, such as specular model types and normal map direction.

This is for artistic metallic objects: hard opaque plastics, soft rubbers, clothing, cutlery, etc.

Fixes and changes from legacy version 1.0.0:

  • Added file selection option
  • Added channel description
  • Added error codes
  • Broke up UI into channels
  • Added PxrLayer functionality
  • Added mass creation options
  • Removed RenderMan texture attribute adding
  • Used file nodes over PxrFile nodes
  • Creates lambert material for viewport visibility

DV Artistic PxrSurface Material Generator

This is a script for Maya 2016, Maya 2017, and Maya 2018 for use with Renderman for Maya Versions 21.0 through 22.0.

Help page

Download the script here.

This script allows users to generate and populate popularly used channels on PxrSurface materials. No need to add Renderman Texture attributes or set color space. A variety of settings are available for select use.

This is for artistic metallic objects: hard opaque plastics, soft rubbers, clothing, cutlery, etc.

if(`window -exists DVpxrArtMatWindow`) {deleteUI DVpxrArtMatWindow ;} ;
window -rtf true -title "DualVission Artistic PxrSurface Master Window" DVpxrArtMatWindow ;
//"By selecting the one of the buttons below, excluding the cancel function, \nyou have verified you have enabled Renderman_for_Maya plugin." ;
columnLayout ;
textFieldGrp -l "Name of Material" -ed true -tx "defaultMat" matNameField ;
frameLayout -l "Map Selection" -cll false -cl false -w 392 ;
columnLayout ;
checkBoxGrp -l "Diffuse Map" -ed true -value1 true diffuseSelectBox ;
checkBoxGrp -l "Specular Face Map" -ed true -value1 true specFaceSelectBox ;
checkBoxGrp -l "Specular Edge Map" -ed true -value1 true specEdgeSelectBox ;
checkBoxGrp -l "Specular Roughness Map" -ed true -value1 true specRoughSelectBox ;
checkBoxGrp -l "Normal Map" -ed true -value1 true normSelectBox ;
setParent..;
setParent..;
frameLayout -l "Map Connection" -cll true -cl true -w 392 ;
columnLayout ;
checkBoxGrp -l "Connect Maps" -ed true -value1 false connectMapSelectBox ;
textFieldGrp -l "Relative File Path" -ed true -tx "sourceimages/" relativeFilePath ;
textFieldGrp -l "File Extension" -ed true -tx ".png" fileExtension ;
frameLayout -l "Naming Convention \(Cannot Be Changed\)" -cll true -cl false ;
textFieldGrp -l "Diffuse Map" -ed false -tx "_diffuse" ;
textFieldGrp -l "Specular Face Map" -ed false -tx "_specFace" ;
textFieldGrp -l "Specular Edge Map" -ed false -tx "_specEdge" ;
textFieldGrp -l "Specular Roughness Map" -ed false -tx "_specRough" ;
textFieldGrp -l "Normal Map" -ed false -tx "_normal" ;
setParent..;
setParent..;
setParent..;
frameLayout -l "Advanced Properties" -cll true -cl true -w 392 ;
frameLayout -l "Models and Directions" -cll false -cl false -w 392 ;
rowLayout -nc 2 ;
checkBoxGrp -l "Use Ggx Model \nover Beckmann Model" -ed true -value1 false ggxSelectBox ;
checkBoxGrp -l "OpenGL Normal Direction \nover DirectX Direction" -ed true -value1 false normDirectSelectBox ;
setParent..;
setParent..;
frameLayout -l "Roughness" -cll false -cl false -w 392 ;
rowLayout -nc 2 ;
checkBoxGrp -l "Connect Roughness \ninto Diffuse" -ed true -value1 false diffRoughSelectBox ;
checkBoxGrp -l "Use Rough Specular \nover Primary Specular" -ed true -value1 false primarySpecSelectBox ;
setParent..;
setParent..;
frameLayout -l "Force Version" -cll false -cl false -w 392 ;
columnLayout ;
checkBoxGrp -l "Force Version Number" -ed true -value1 false forceSelectBox ;
textFieldGrp -l "Override Number" -ed true -tx "" numberNameField;
setParent..;
setParent..;
setParent..;
frameLayout -l "Finalize" -cll false -cl false -w 392 ;
rowLayout -nc 3 ;
button -l "Create" -c "DVpxrArtMatCreator; deleteUI DVpxrArtMatWindow; " -w 131 ;
button -l "Apply" -c "DVpxrArtMatCreator" -w 131 ;
button -l "Cancel" -c "deleteUI DVpxrArtMatWindow" -w 130 ;
setParent..;
setParent..;
showWindow ;
proc DVpxrArtMatCreator(){
//Declare variable used throughout script
string $nameMat = `textFieldGrp -q -tx matNameField` ;
//Declare state of lines
float $testVersion ;
if(`checkBoxGrp -q -value1 forceSelectBox` == true)
{$testVersion = `textFieldGrp -q -tx numberNameField` ;}
else
{$testVersion = `rman getversion` ;} ;
int $testDiffuse ;
if(`checkBoxGrp -q -value1 diffuseSelectBox` == true)
{$testDiffuse = 1 ;}
else
{$testDiffuse = 0 ;} ;
int $testSpecFace ;
if(`checkBoxGrp -q -value1 specFaceSelectBox` == true)
{$testSpecFace = 1 ;}
else
{$testSpecFace = 0 ;} ;
int $testSpecEdge ;
if(`checkBoxGrp -q -value1 specEdgeSelectBox` == true)
{$testSpecEdge = 1 ;}
else
{$testSpecEdge = 0 ;} ;
int $testSpecRough ;
if(`checkBoxGrp -q -value1 specRoughSelectBox` == true)
{$testSpecRough = 1 ;}
else
{$testSpecRough = 0 ;} ;
int $testPrimary ;
if(`checkBoxGrp -q -value1 primarySpecSelectBox` == true)
{$testPrimary = 0 ;}
else
{$testPrimary = 1 ;} ;
int $testNormal ;
if(`checkBoxGrp -q -value1 normSelectBox` == true)
{$testNormal = 1 ;}
else
{$testNormal = 0 ;} ;
int $testGXG ;
if(`checkBoxGrp -q -value1 ggxSelectBox` == true)
{$testGXG = 1 ;}
else
{$testGXG = 0 ;} ;
int $testDiffRough ;
if(`checkBoxGrp -q -value1 diffRoughSelectBox` == true)
{$testDiffRough = 1 ;}
else
{$testDiffRough = 0 ;} ;
int $testNormDirect ;
if(`checkBoxGrp -q -value1 normDirectSelectBox` == true)
{$testNormDirect = 1 ;}
else
{$testNormDirect = 0 ;} ;
int $testConnect ;
string $filePath = `textFieldGrp -q -tx relativeFilePath` ;
string $fileType = `textFieldGrp -q -tx fileExtension` ;
if(`checkBoxGrp -q -value1 connectMapSelectBox` == true)
{$testConnect = 1 ;}
else
{$testConnect = 0 ;} ;
//create and connect pxrsurface nodes
shadingNode -asShader PxrSurface -name $nameMat ;
sets -renderable true -noSurfaceShader true -empty -name ($nameMat + "_SG");
connectAttr -f ($nameMat + ".outColor") ($nameMat + "_SG.surfaceShader");
if($testVersion < 22.0){
//test if a material's diffuse is enabled
if($testDiffuse == 1){
//Declare section variables
string $nameDiffuse = ($nameMat + "_diffuse") ;
string $nameDiffusePlace2d = ($nameDiffuse + "_Place2d") ;
//create and connect texture nodes
shadingNode -asTexture -isColorManaged file -name $nameDiffuse ;
shadingNode -asUtility place2dTexture -name $nameDiffusePlace2d ;
connectAttr -f ($nameDiffusePlace2d + ".coverage") ($nameDiffuse + ".coverage") ;
connectAttr -f ($nameDiffusePlace2d + ".translateFrame") ($nameDiffuse + ".translateFrame") ;
connectAttr -f ($nameDiffusePlace2d + ".rotateFrame") ($nameDiffuse + ".rotateFrame") ;
connectAttr -f ($nameDiffusePlace2d + ".mirrorU") ($nameDiffuse + ".mirrorU") ;
connectAttr -f ($nameDiffusePlace2d + ".mirrorV") ($nameDiffuse + ".mirrorV") ;
connectAttr -f ($nameDiffusePlace2d + ".stagger") ($nameDiffuse + ".stagger") ;
connectAttr -f ($nameDiffusePlace2d + ".wrapU") ($nameDiffuse + ".wrapU") ;
connectAttr -f ($nameDiffusePlace2d + ".wrapV") ($nameDiffuse + ".wrapV") ;
connectAttr -f ($nameDiffusePlace2d + ".repeatUV") ($nameDiffuse + ".repeatUV") ;
connectAttr -f ($nameDiffusePlace2d + ".offset") ($nameDiffuse + ".offset") ;
connectAttr -f ($nameDiffusePlace2d + ".rotateUV") ($nameDiffuse + ".rotateUV") ;
connectAttr -f ($nameDiffusePlace2d + ".noiseUV") ($nameDiffuse + ".noiseUV") ;
connectAttr -f ($nameDiffusePlace2d + ".vertexUvOne") ($nameDiffuse + ".vertexUvOne") ;
connectAttr -f ($nameDiffusePlace2d + ".vertexUvTwo") ($nameDiffuse + ".vertexUvTwo") ;
connectAttr -f ($nameDiffusePlace2d + ".vertexUvThree") ($nameDiffuse + ".vertexUvThree") ;
connectAttr -f ($nameDiffusePlace2d + ".vertexCameraOne") ($nameDiffuse + ".vertexCameraOne") ;
connectAttr ($nameDiffusePlace2d + ".outUV") ($nameDiffuse + ".uv") ;
connectAttr ($nameDiffusePlace2d + ".outUvFilterSize") ($nameDiffuse + ".uvFilterSize") ;
rmanAddAttr $nameDiffuse rman__tx2dFilter "0" ;
rmanAddAttr $nameDiffuse rman__tx2dSwidth "0" ;
rmanAddAttr $nameDiffuse rman__tx2dTwidth "0" ;
rmanAddAttr $nameDiffuse rman__tx2dLerp "0" ;
rmanAddAttr $nameDiffuse rman__applysRGB "1" ;
rmanAddAttr $nameDiffuse rman__unpremultiply "0" ;
//connect texture nodes to master nodes
connectAttr -f ($nameDiffuse + ".outColor") ($nameMat + ".diffuseColor");
if($testConnect == 1){
setAttr -type "string" ($nameDiffuse + ".fileTextureName") ($filePath + $nameDiffuse + $fileType) ;
} ;
} ;
//test if material is regular or rough specular
if($testPrimary == 1){
if($testGXG == 1){setAttr -type "string" ($nameMat + ".specularModelType") "Ggx" ;} ;
if($testSpecFace == 1){
string $nameSpecFace = ($nameMat + "_specFace") ;
string $nameSpecFacePlace2d = ($nameSpecFace + "_Place2d") ;
shadingNode -asTexture -isColorManaged file -name $nameSpecFace ;
shadingNode -asUtility place2dTexture -name $nameSpecFacePlace2d ;
connectAttr -f ($nameSpecFacePlace2d + ".coverage") ($nameSpecFace + ".coverage") ;
connectAttr -f ($nameSpecFacePlace2d + ".translateFrame") ($nameSpecFace + ".translateFrame") ;
connectAttr -f ($nameSpecFacePlace2d + ".rotateFrame") ($nameSpecFace + ".rotateFrame") ;
connectAttr -f ($nameSpecFacePlace2d + ".mirrorU") ($nameSpecFace + ".mirrorU") ;
connectAttr -f ($nameSpecFacePlace2d + ".mirrorV") ($nameSpecFace + ".mirrorV") ;
connectAttr -f ($nameSpecFacePlace2d + ".stagger") ($nameSpecFace + ".stagger") ;
connectAttr -f ($nameSpecFacePlace2d + ".wrapU") ($nameSpecFace + ".wrapU") ;
connectAttr -f ($nameSpecFacePlace2d + ".wrapV") ($nameSpecFace + ".wrapV") ;
connectAttr -f ($nameSpecFacePlace2d + ".repeatUV") ($nameSpecFace + ".repeatUV") ;
connectAttr -f ($nameSpecFacePlace2d + ".offset") ($nameSpecFace + ".offset") ;
connectAttr -f ($nameSpecFacePlace2d + ".rotateUV") ($nameSpecFace + ".rotateUV") ;
connectAttr -f ($nameSpecFacePlace2d + ".noiseUV") ($nameSpecFace + ".noiseUV") ;
connectAttr -f ($nameSpecFacePlace2d + ".vertexUvOne") ($nameSpecFace + ".vertexUvOne") ;
connectAttr -f ($nameSpecFacePlace2d + ".vertexUvTwo") ($nameSpecFace + ".vertexUvTwo") ;
connectAttr -f ($nameSpecFacePlace2d + ".vertexUvThree") ($nameSpecFace + ".vertexUvThree") ;
connectAttr -f ($nameSpecFacePlace2d + ".vertexCameraOne") ($nameSpecFace + ".vertexCameraOne") ;
connectAttr ($nameSpecFacePlace2d + ".outUV") ($nameSpecFace + ".uv") ;
connectAttr ($nameSpecFacePlace2d + ".outUvFilterSize") ($nameSpecFace + ".uvFilterSize") ;
rmanAddAttr $nameSpecFace rman__tx2dFilter "0" ;
rmanAddAttr $nameSpecFace rman__tx2dSwidth "0" ;
rmanAddAttr $nameSpecFace rman__tx2dTwidth "0" ;
rmanAddAttr $nameSpecFace rman__tx2dLerp "0" ;
rmanAddAttr $nameSpecFace rman__applysRGB "1" ;
rmanAddAttr $nameSpecFace rman__unpremultiply "0" ;
connectAttr -f ($nameSpecFace + ".outColor") ($nameMat + ".specularFaceColor");
if($testConnect == 1){
setAttr -type "string" ($nameSpecFace + ".fileTextureName") ($filePath + $nameSpecFace + $fileType) ;
} ;
} ;
if($testSpecEdge == 1){
string $nameSpecEdge = ($nameMat + "_specEdge") ;
string $nameSpecEdgePlace2d = ($nameSpecEdge + "_Place2d") ;
shadingNode -asTexture -isColorManaged file -name $nameSpecEdge ;
shadingNode -asUtility place2dTexture -name $nameSpecEdgePlace2d ;
connectAttr -f ($nameSpecEdgePlace2d + ".coverage") ($nameSpecEdge + ".coverage") ;
connectAttr -f ($nameSpecEdgePlace2d + ".translateFrame") ($nameSpecEdge + ".translateFrame") ;
connectAttr -f ($nameSpecEdgePlace2d + ".rotateFrame") ($nameSpecEdge + ".rotateFrame") ;
connectAttr -f ($nameSpecEdgePlace2d + ".mirrorU") ($nameSpecEdge + ".mirrorU") ;
connectAttr -f ($nameSpecEdgePlace2d + ".mirrorV") ($nameSpecEdge + ".mirrorV") ;
connectAttr -f ($nameSpecEdgePlace2d + ".stagger") ($nameSpecEdge + ".stagger") ;
connectAttr -f ($nameSpecEdgePlace2d + ".wrapU") ($nameSpecEdge + ".wrapU") ;
connectAttr -f ($nameSpecEdgePlace2d + ".wrapV") ($nameSpecEdge + ".wrapV") ;
connectAttr -f ($nameSpecEdgePlace2d + ".repeatUV") ($nameSpecEdge + ".repeatUV") ;
connectAttr -f ($nameSpecEdgePlace2d + ".offset") ($nameSpecEdge + ".offset") ;
connectAttr -f ($nameSpecEdgePlace2d + ".rotateUV") ($nameSpecEdge + ".rotateUV") ;
connectAttr -f ($nameSpecEdgePlace2d + ".noiseUV") ($nameSpecEdge + ".noiseUV") ;
connectAttr -f ($nameSpecEdgePlace2d + ".vertexUvOne") ($nameSpecEdge + ".vertexUvOne") ;
connectAttr -f ($nameSpecEdgePlace2d + ".vertexUvTwo") ($nameSpecEdge + ".vertexUvTwo") ;
connectAttr -f ($nameSpecEdgePlace2d + ".vertexUvThree") ($nameSpecEdge + ".vertexUvThree") ;
connectAttr -f ($nameSpecEdgePlace2d + ".vertexCameraOne") ($nameSpecEdge + ".vertexCameraOne") ;
connectAttr ($nameSpecEdgePlace2d + ".outUV") ($nameSpecEdge + ".uv") ;
connectAttr ($nameSpecEdgePlace2d + ".outUvFilterSize") ($nameSpecEdge + ".uvFilterSize") ;
rmanAddAttr $nameSpecEdge rman__tx2dFilter "0" ;
rmanAddAttr $nameSpecEdge rman__tx2dSwidth "0" ;
rmanAddAttr $nameSpecEdge rman__tx2dTwidth "0" ;
rmanAddAttr $nameSpecEdge rman__tx2dLerp "0" ;
rmanAddAttr $nameSpecEdge rman__applysRGB "1" ;
rmanAddAttr $nameSpecEdge rman__unpremultiply "0" ;
connectAttr -f ($nameSpecEdge + ".outColor") ($nameMat + ".specularEdgeColor");
if($testConnect == 1){
setAttr -type "string" ($nameSpecEdge + ".fileTextureName") ($filePath + $nameSpecEdge + $fileType) ;
} ;
} ;
if($testSpecRough == 1){
string $nameRough = ($nameMat + "_specRough") ;
string $nameRoughPlace2d = ($nameRough + "_Place2d") ;
shadingNode -asTexture -isColorManaged file -name $nameRough ;
shadingNode -asUtility place2dTexture -name $nameRoughPlace2d ;
connectAttr -f ($nameRoughPlace2d + ".coverage") ($nameRough + ".coverage") ;
connectAttr -f ($nameRoughPlace2d + ".translateFrame") ($nameRough + ".translateFrame") ;
connectAttr -f ($nameRoughPlace2d + ".rotateFrame") ($nameRough + ".rotateFrame") ;
connectAttr -f ($nameRoughPlace2d + ".mirrorU") ($nameRough + ".mirrorU") ;
connectAttr -f ($nameRoughPlace2d + ".mirrorV") ($nameRough + ".mirrorV") ;
connectAttr -f ($nameRoughPlace2d + ".stagger") ($nameRough + ".stagger") ;
connectAttr -f ($nameRoughPlace2d + ".wrapU") ($nameRough + ".wrapU") ;
connectAttr -f ($nameRoughPlace2d + ".wrapV") ($nameRough + ".wrapV") ;
connectAttr -f ($nameRoughPlace2d + ".repeatUV") ($nameRough + ".repeatUV") ;
connectAttr -f ($nameRoughPlace2d + ".offset") ($nameRough + ".offset") ;
connectAttr -f ($nameRoughPlace2d + ".rotateUV") ($nameRough + ".rotateUV") ;
connectAttr -f ($nameRoughPlace2d + ".noiseUV") ($nameRough + ".noiseUV") ;
connectAttr -f ($nameRoughPlace2d + ".vertexUvOne") ($nameRough + ".vertexUvOne") ;
connectAttr -f ($nameRoughPlace2d + ".vertexUvTwo") ($nameRough + ".vertexUvTwo") ;
connectAttr -f ($nameRoughPlace2d + ".vertexUvThree") ($nameRough + ".vertexUvThree") ;
connectAttr -f ($nameRoughPlace2d + ".vertexCameraOne") ($nameRough + ".vertexCameraOne") ;
connectAttr ($nameRoughPlace2d + ".outUV") ($nameRough + ".uv") ;
connectAttr ($nameRoughPlace2d + ".outUvFilterSize") ($nameRough + ".uvFilterSize") ;
rmanAddAttr $nameRough rman__tx2dFilter "0" ;
rmanAddAttr $nameRough rman__tx2dSwidth "0" ;
rmanAddAttr $nameRough rman__tx2dTwidth "0" ;
rmanAddAttr $nameRough rman__tx2dLerp "0" ;
rmanAddAttr $nameRough rman__applysRGB "0" ;
rmanAddAttr $nameRough rman__unpremultiply "0" ;
connectAttr -f ($nameRough + ".outAlpha") ($nameMat + ".specularRoughness");
if($testDiffRough == 1){connectAttr -f ($nameRough + ".outAlpha") ($nameMat + ".diffuseRoughness"); } ;
if($testConnect == 1){
setAttr -type "string" ($nameRough + ".fileTextureName") ($filePath + $nameRough + $fileType) ;
setAttr -type "string" ($nameRough + ".colorSpace") "Raw" ;
} ;
} ;
}
else {
if($testGXG == 1){setAttr -type "string" ($nameMat + ".specularModelType") "Ggx" ;} ;
if($testSpecFace == 1){
string $nameSpecFace = ($nameMat + "_specFace") ;
string $nameSpecFacePlace2d = ($nameSpecFace + "_Place2d") ;
shadingNode -asTexture -isColorManaged file -name $nameSpecFace ;
shadingNode -asUtility place2dTexture -name $nameSpecFacePlace2d ;
connectAttr -f ($nameSpecFacePlace2d + ".coverage") ($nameSpecFace + ".coverage") ;
connectAttr -f ($nameSpecFacePlace2d + ".translateFrame") ($nameSpecFace + ".translateFrame") ;
connectAttr -f ($nameSpecFacePlace2d + ".rotateFrame") ($nameSpecFace + ".rotateFrame") ;
connectAttr -f ($nameSpecFacePlace2d + ".mirrorU") ($nameSpecFace + ".mirrorU") ;
connectAttr -f ($nameSpecFacePlace2d + ".mirrorV") ($nameSpecFace + ".mirrorV") ;
connectAttr -f ($nameSpecFacePlace2d + ".stagger") ($nameSpecFace + ".stagger") ;
connectAttr -f ($nameSpecFacePlace2d + ".wrapU") ($nameSpecFace + ".wrapU") ;
connectAttr -f ($nameSpecFacePlace2d + ".wrapV") ($nameSpecFace + ".wrapV") ;
connectAttr -f ($nameSpecFacePlace2d + ".repeatUV") ($nameSpecFace + ".repeatUV") ;
connectAttr -f ($nameSpecFacePlace2d + ".offset") ($nameSpecFace + ".offset") ;
connectAttr -f ($nameSpecFacePlace2d + ".rotateUV") ($nameSpecFace + ".rotateUV") ;
connectAttr -f ($nameSpecFacePlace2d + ".noiseUV") ($nameSpecFace + ".noiseUV") ;
connectAttr -f ($nameSpecFacePlace2d + ".vertexUvOne") ($nameSpecFace + ".vertexUvOne") ;
connectAttr -f ($nameSpecFacePlace2d + ".vertexUvTwo") ($nameSpecFace + ".vertexUvTwo") ;
connectAttr -f ($nameSpecFacePlace2d + ".vertexUvThree") ($nameSpecFace + ".vertexUvThree") ;
connectAttr -f ($nameSpecFacePlace2d + ".vertexCameraOne") ($nameSpecFace + ".vertexCameraOne") ;
connectAttr ($nameSpecFacePlace2d + ".outUV") ($nameSpecFace + ".uv") ;
connectAttr ($nameSpecFacePlace2d + ".outUvFilterSize") ($nameSpecFace + ".uvFilterSize") ;
rmanAddAttr $nameSpecFace rman__tx2dFilter "0" ;
rmanAddAttr $nameSpecFace rman__tx2dSwidth "0" ;
rmanAddAttr $nameSpecFace rman__tx2dTwidth "0" ;
rmanAddAttr $nameSpecFace rman__tx2dLerp "0" ;
rmanAddAttr $nameSpecFace rman__applysRGB "1" ;
rmanAddAttr $nameSpecFace rman__unpremultiply "0" ;
connectAttr -f ($nameSpecFace + ".outColor") ($nameMat + ".roughSpecularFaceColor");
if($testConnect == 1){
setAttr -type "string" ($nameSpecFace + ".fileTextureName") ($filePath + $nameSpecFace + $fileType) ;
} ;
} ;
if($testSpecEdge == 1){
string $nameSpecEdge = ($nameMat + "_specEdge") ;
string $nameSpecEdgePlace2d = ($nameSpecEdge + "_Place2d") ;
shadingNode -asTexture -isColorManaged file -name $nameSpecEdge ;
shadingNode -asUtility place2dTexture -name $nameSpecEdgePlace2d ;
connectAttr -f ($nameSpecEdgePlace2d + ".coverage") ($nameSpecEdge + ".coverage") ;
connectAttr -f ($nameSpecEdgePlace2d + ".translateFrame") ($nameSpecEdge + ".translateFrame") ;
connectAttr -f ($nameSpecEdgePlace2d + ".rotateFrame") ($nameSpecEdge + ".rotateFrame") ;
connectAttr -f ($nameSpecEdgePlace2d + ".mirrorU") ($nameSpecEdge + ".mirrorU") ;
connectAttr -f ($nameSpecEdgePlace2d + ".mirrorV") ($nameSpecEdge + ".mirrorV") ;
connectAttr -f ($nameSpecEdgePlace2d + ".stagger") ($nameSpecEdge + ".stagger") ;
connectAttr -f ($nameSpecEdgePlace2d + ".wrapU") ($nameSpecEdge + ".wrapU") ;
connectAttr -f ($nameSpecEdgePlace2d + ".wrapV") ($nameSpecEdge + ".wrapV") ;
connectAttr -f ($nameSpecEdgePlace2d + ".repeatUV") ($nameSpecEdge + ".repeatUV") ;
connectAttr -f ($nameSpecEdgePlace2d + ".offset") ($nameSpecEdge + ".offset") ;
connectAttr -f ($nameSpecEdgePlace2d + ".rotateUV") ($nameSpecEdge + ".rotateUV") ;
connectAttr -f ($nameSpecEdgePlace2d + ".noiseUV") ($nameSpecEdge + ".noiseUV") ;
connectAttr -f ($nameSpecEdgePlace2d + ".vertexUvOne") ($nameSpecEdge + ".vertexUvOne") ;
connectAttr -f ($nameSpecEdgePlace2d + ".vertexUvTwo") ($nameSpecEdge + ".vertexUvTwo") ;
connectAttr -f ($nameSpecEdgePlace2d + ".vertexUvThree") ($nameSpecEdge + ".vertexUvThree") ;
connectAttr -f ($nameSpecEdgePlace2d + ".vertexCameraOne") ($nameSpecEdge + ".vertexCameraOne") ;
connectAttr ($nameSpecEdgePlace2d + ".outUV") ($nameSpecEdge + ".uv") ;
connectAttr ($nameSpecEdgePlace2d + ".outUvFilterSize") ($nameSpecEdge + ".uvFilterSize") ;
rmanAddAttr $nameSpecEdge rman__tx2dFilter "0" ;
rmanAddAttr $nameSpecEdge rman__tx2dSwidth "0" ;
rmanAddAttr $nameSpecEdge rman__tx2dTwidth "0" ;
rmanAddAttr $nameSpecEdge rman__tx2dLerp "0" ;
rmanAddAttr $nameSpecEdge rman__applysRGB "1" ;
rmanAddAttr $nameSpecEdge rman__unpremultiply "0" ;
connectAttr -f ($nameSpecEdge + ".outColor") ($nameMat + ".roughSpecularEdgeColor");
if($testConnect == 1){
setAttr -type "string" ($nameSpecEdge + ".fileTextureName") ($filePath + $nameSpecEdge + $fileType) ;
} ;
} ;
if($testSpecRough == 1){
string $nameRough = ($nameMat + "_specRough") ;
string $nameRoughPlace2d = ($nameRough + "_Place2d") ;
shadingNode -asTexture -isColorManaged file -name $nameRough ;
shadingNode -asUtility place2dTexture -name $nameRoughPlace2d ;
connectAttr -f ($nameRoughPlace2d + ".coverage") ($nameRough + ".coverage") ;
connectAttr -f ($nameRoughPlace2d + ".translateFrame") ($nameRough + ".translateFrame") ;
connectAttr -f ($nameRoughPlace2d + ".rotateFrame") ($nameRough + ".rotateFrame") ;
connectAttr -f ($nameRoughPlace2d + ".mirrorU") ($nameRough + ".mirrorU") ;
connectAttr -f ($nameRoughPlace2d + ".mirrorV") ($nameRough + ".mirrorV") ;
connectAttr -f ($nameRoughPlace2d + ".stagger") ($nameRough + ".stagger") ;
connectAttr -f ($nameRoughPlace2d + ".wrapU") ($nameRough + ".wrapU") ;
connectAttr -f ($nameRoughPlace2d + ".wrapV") ($nameRough + ".wrapV") ;
connectAttr -f ($nameRoughPlace2d + ".repeatUV") ($nameRough + ".repeatUV") ;
connectAttr -f ($nameRoughPlace2d + ".offset") ($nameRough + ".offset") ;
connectAttr -f ($nameRoughPlace2d + ".rotateUV") ($nameRough + ".rotateUV") ;
connectAttr -f ($nameRoughPlace2d + ".noiseUV") ($nameRough + ".noiseUV") ;
connectAttr -f ($nameRoughPlace2d + ".vertexUvOne") ($nameRough + ".vertexUvOne") ;
connectAttr -f ($nameRoughPlace2d + ".vertexUvTwo") ($nameRough + ".vertexUvTwo") ;
connectAttr -f ($nameRoughPlace2d + ".vertexUvThree") ($nameRough + ".vertexUvThree") ;
connectAttr -f ($nameRoughPlace2d + ".vertexCameraOne") ($nameRough + ".vertexCameraOne") ;
connectAttr ($nameRoughPlace2d + ".outUV") ($nameRough + ".uv") ;
connectAttr ($nameRoughPlace2d + ".outUvFilterSize") ($nameRough + ".uvFilterSize") ;
rmanAddAttr $nameRough rman__tx2dFilter "0" ;
rmanAddAttr $nameRough rman__tx2dSwidth "0" ;
rmanAddAttr $nameRough rman__tx2dTwidth "0" ;
rmanAddAttr $nameRough rman__tx2dLerp "0" ;
rmanAddAttr $nameRough rman__applysRGB "0" ;
rmanAddAttr $nameRough rman__unpremultiply "0" ;
connectAttr -f ($nameRough + ".outAlpha") ($nameMat + ".roughSpecularRoughness");
if($testDiffRough == 1){connectAttr -f ($nameRough + ".outAlpha") ($nameMat + ".diffuseRoughness"); } ;
if($testConnect == 1){
setAttr -type "string" ($nameRough + ".fileTextureName") ($filePath + $nameRough + $fileType) ;
setAttr -type "string" ($nameRough + ".colorSpace") "Raw" ;
} ;
} ;
} ;
if($testNormal == 1)
{
string $nameNormal = ($nameMat + "_normal") ;
shadingNode -asTexture PxrNormalMap -name $nameNormal ;
connectAttr -f ($nameNormal + ".resultN") ($nameMat + ".bumpNormal") ;
if($testNormDirect == 1)
{
rmanSetAttr $nameNormal orientation 0;
}
else
{
rmanSetAttr $nameNormal orientation 1;
} ;
if($testConnect == 1){
setAttr -type "string" ($nameNormal + ".filename") ($filePath + $nameNormal + $fileType) ;
} ;
} ;
}
else {
if($testDiffuse == 1){
string $nameDiffuse = ($nameMat + "_diffuse") ;
shadingNode -asTexture PxrTexture -name $nameDiffuse ;
setAttr ($nameDiffuse + ".linearize") 1;
connectAttr -f ($nameDiffuse + ".resultRGB") ($nameMat + ".diffuseColor") ;
if($testConnect == 1){
setAttr -type "string" ($nameDiffuse + ".filename") ($filePath + $nameDiffuse + $fileType) ;
} ;
} ;
if($testPrimary == 1){
if($testGXG == 1){setAttr -type "string" ($nameMat + ".specularModelType") "Ggx" ;} ;
if($testSpecFace == 1){
string $nameSpecFace = ($nameMat + "_specFace") ;
shadingNode -asTexture PxrTexture -name $nameSpecFace ;
setAttr ($nameSpecFace + ".linearize") 1;
connectAttr -f ($nameSpecFace + ".resultRGB") ($nameMat + ".specularFaceColor") ;
if($testConnect == 1){
setAttr -type "string" ($nameSpecFace + ".filename") ($filePath + $nameSpecFace + $fileType) ;
} ;
} ;
if($testSpecEdge == 1){
string $nameSpecEdge = ($nameMat + "_specEdge") ;
shadingNode -asTexture PxrTexture -name $nameSpecEdge ;
setAttr ($nameSpecEdge + ".linearize") 1;
connectAttr -f ($nameSpecEdge + ".resultRGB") ($nameMat + ".specularEdgeColor") ;
if($testConnect == 1){
setAttr -type "string" ($nameSpecEdge + ".filename") ($filePath + $nameSpecEdge + $fileType) ;
} ;
} ;
if($testSpecRough == 1){
string $nameSpecRough = ($nameMat + "_specRough") ;
shadingNode -asTexture PxrTexture -name $nameSpecRough ;
setAttr ($nameSpecRough + ".linearize") 0;
connectAttr -f ($nameSpecRough + ".resultA") ($nameMat + ".specularRoughness") ;
if($testDiffRough == 1){connectAttr -f ($nameSpecRough + ".resultA") ($nameMat + ".diffuseRoughness"); } ;
if($testConnect == 1){
setAttr -type "string" ($nameSpecRough + ".filename") ($filePath + $nameSpecRough + $fileType) ;
} ;
} ;
}
else{
if($testGXG == 1){setAttr -type "string" ($nameMat + ".roughSpecularModelType") "Ggx" ;} ;
if($testSpecFace == 1){
string $nameSpecFace = ($nameMat + "_specFace") ;
shadingNode -asTexture PxrTexture -name $nameSpecFace ;
setAttr ($nameSpecFace + ".linearize") 1;
connectAttr -f ($nameSpecFace + ".resultRGB") ($nameMat + ".roughSpecularFaceColor") ;
if($testConnect == 1){
setAttr -type "string" ($nameSpecFace + ".filename") ($filePath + $nameSpecFace + $fileType) ;
} ;
} ;
if($testSpecEdge == 1){
string $nameSpecEdge = ($nameMat + "_specEdge") ;
shadingNode -asTexture PxrTexture -name $nameSpecEdge ;
setAttr ($nameSpecEdge + ".linearize") 1;
connectAttr -f ($nameSpecEdge + ".resultRGB") ($nameMat + ".roughSpecularEdgeColor") ;
if($testConnect == 1){
setAttr -type "string" ($nameSpecEdge + ".filename") ($filePath + $nameSpecEdge + $fileType) ;
} ;
} ;
if($testSpecRough == 1){
string $nameSpecRough = ($nameMat + "_specRough") ;
shadingNode -asTexture PxrTexture -name $nameSpecRough ;
setAttr ($nameSpecRough + ".linearize") 0;
connectAttr -f ($nameSpecRough + ".resultA") ($nameMat + ".roughSpecularRoughness") ;
if($testDiffRough == 1){connectAttr -f ($nameSpecRough + ".resultA") ($nameMat + ".diffuseRoughness"); } ;
if($testConnect == 1){
setAttr -type "string" ($nameSpecRough + ".filename") ($filePath + $nameSpecRough + $fileType) ;
} ;
} ;
} ;
if($testNormal == 1){
string $nameNormal = ($nameMat + "_normal") ;
shadingNode -asTexture PxrNormalMap -name $nameNormal ;
connectAttr -f ($nameNormal + ".resultN") ($nameMat + ".bumpNormal") ;
if($testNormDirect == 1)
{
rmanSetAttr $nameNormal orientation 0;
}
else
{
rmanSetAttr $nameNormal orientation 1;
} ;
if($testConnect == 1){
setAttr -type "string" ($nameNormal + ".filename") ($filePath + $nameNormal + $fileType) ;
} ;
} ;
} ;
} ;

Update 17/09/2018 22:44 US EDT: Fixes button size not translating correctly
Update 17/09/2018 23:58 US EDT: Fixes typo

DV Clone Master Script

This script for Maya 2016, Maya 2017, and Maya 2018 will grab objects parented to world or world oriented/centered nodes.

Download the script here.

/*
Limitations
Script can only mirror objects parented to world or world oriented and centered objects
For more concerns, please email me
*/
{
//Finds objects and lists them
string $nodes0[] = `ls -selection`;
//Clones and lists clones of selected objects
duplicate -rr ;
string $nodes1[] = `ls -selection`;
//Groups and lists group
doGroup 0 1 1 ;
string $group[] = `ls -selection`;
for ($node in $group)
{
//Loop through each object and obtain its node
string $node[] = `listRelatives $node`;
float $dimension = `getAttr ($node[0] + ".scaleX")` ;
//Set the x-scale attribute of each shape node to its negative counterpart
setAttr ($node[0] + ".scaleX") (-1*$dimension) ;
}
//Selects group
select -r $group ;
FreezeTransformations ;
//Parents objects to world
parent -world $nodes1 ;
select -cl ;
//Selects and deletes created groups
select -r $group ;
delete ;
//Selects and freezes transforms on affected objects
select -r $nodes0 ;
select -tgl $nodes1 ;
FreezeTransformations ;
}
// Script by Zachary Yarnot the DualVission
// Script for non-commercial and personal use only
// https://yarnot.com/Zach
// zach@yarnot.com
print(" Objects cloned and mirrored. Transforms frozen. ; ") ;

Update 06/09/2018 21:27 US EDT: Fixes typo in release preventing script from functioning with more than one nodes selected.

Website Comparing iPhone 7 Plus and Pixel XL

Link (http://yarnot.com/phoneproj/)

Image displaying the devices side by side

Image displaying the difference in image quality

All copyrights are held by their respective owners.
All content is rendered within 3DS Max® under an educational license.
Zachary Yarnot and DualVission do not hold any rights to these owners’ contents.
Autodesk® and 3DS Max® are registered trademarks of Autodesk, Inc.
iPhone® is a trademark of Apple, Inc.
Pixel™ is a trademark of Google LLC and Alphabet, Inc.
Zachary Yarnot and DualVission are in no way related to or endorsed by these companies – or creators – or their brands. The actions of said persons are not in any way connected to or encouraged by other mentioned parties.

Works Cited

Apple, Inc. Technical Specifications. 2016. 18 Nov 2016. <www.apple.com/iphone-7/specs/>.

—. This is 7. 2016. 18 Nov 2016. <www.apple.com/iphone-7/?afid=p238%7CswRB8Ie77-dc_mtid_20925d2q39172_pcrid_137641212607_&cid=wwa-us-kwgo-iphone-slid->.

Brownlee, Marques. “Google Assistant vs Siri! (2016).” 21 Oct 2016. YouTube. 20 Nov 2016. <www.youtube.com/watch?v=JFiu5rfnhzo>.

Centers for Disease Control and Prevention. “Distracted Driving.” n.d. Injury Prevention & Control: Motor Vehicle Safety. 20 Nov 2016. <www.cdc.gov/motorvehiclesafety/distracted_driving/>.

Google, Inc. Meet Pixel, Phone by Google. 2016. 18 Nov 2016. <madeby.google.com/phone/>.

—. Tech Specs. 2016. 18 Nov 2016. <madeby.google.com/phone/specs/>.

GSMArena.com. “Apple iPhone 7 Plus.” 2016. GSMArena. 20 Nov 2016. <www.gsmarena.com/apple_iphone_7_plus-8065.php>.

—. “Google Pixel XL.” 2016. GSMArena. 20 Nov 2016. <www.gsmarena.com/google_pixel_xl-8345.php>.

—. Photo quality comparison. n.d. 20 Nov 2016. <www.gsmarena.com/piccmp.php3?idType=1&idPhone1=8345&idPhone2=8065&idPhone3=7945>.

JerryRigEverything. “Smart Phone Durability Tests – JerryRigEverything.” 16 Nov 2016. YouTube. 18 Nov 2016. <www.youtube.com/playlist?list=PL0vZL9uwyfOHdcsT4QDrvFuMOmFPQjIPA>.

Rahimi, David. “Google Pixel XL vs. iPhone 7 Plus.” 25 Oct 2016. YouTube. Ed. PhoneBuff. 20 Nov 2016. <www.youtube.com/watch?v=Na0JRDMAoKM>.

Verizon. “iPhone® 7 Plus.” 2016. Verizon. Web. 20 Nov 2016. <www.verizonwireless.com/smartphones/apple-iphone-7-plus/>.

—. “Pixel XL, Phone by Google.” 2016. Verizon. Web. 20 Nov 2016. <www.verizonwireless.com/smartphones/google-pixel-xl/>.